blob: 9f14bdd7c6beb53dc34d767414cb6859a74ead65 [file] [log] [blame]
Patrice Arruda219eef32020-06-01 17:29:30 +00001// Copyright 2020 Google Inc. All rights reserved.
2//
3// Licensed under the Apache License, Version 2.0 (the "License");
4// you may not use this file except in compliance with the License.
5// You may obtain a copy of the License at
6//
7// http://www.apache.org/licenses/LICENSE-2.0
8//
9// Unless required by applicable law or agreed to in writing, software
10// distributed under the License is distributed on an "AS IS" BASIS,
11// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
12// See the License for the specific language governing permissions and
13// limitations under the License.
14
15package build
16
17// This file contains the functionality to upload data from one location to
18// another.
19
20import (
usta4f5d2c12022-10-28 23:32:01 -040021 "fmt"
Patrice Arruda219eef32020-06-01 17:29:30 +000022 "io/ioutil"
23 "os"
24 "path/filepath"
25 "time"
26
Patrice Arruda7cc20742020-06-10 18:48:01 +000027 "android/soong/ui/metrics"
Dan Willemsen4591b642021-05-24 14:24:12 -070028
29 "google.golang.org/protobuf/proto"
Patrice Arruda219eef32020-06-01 17:29:30 +000030
31 upload_proto "android/soong/ui/metrics/upload_proto"
32)
33
34const (
Patrice Arruda92dc64f2020-11-16 16:29:16 -080035 // Used to generate a raw protobuf file that contains information
36 // of the list of metrics files from host to destination storage.
Patrice Arruda219eef32020-06-01 17:29:30 +000037 uploadPbFilename = ".uploader.pb"
38)
39
Patrice Arruda7cc20742020-06-10 18:48:01 +000040var (
Patrice Arruda92dc64f2020-11-16 16:29:16 -080041 // For testing purpose.
42 tmpDir = ioutil.TempDir
Patrice Arruda7cc20742020-06-10 18:48:01 +000043)
44
Patrice Arruda7d235cc2020-12-09 22:43:26 +000045// pruneMetricsFiles iterates the list of paths, checking if a path exist.
46// If a path is a file, it is added to the return list. If the path is a
47// directory, a recursive call is made to add the children files of the
48// path.
49func pruneMetricsFiles(paths []string) []string {
50 var metricsFiles []string
51 for _, p := range paths {
52 fi, err := os.Stat(p)
53 // Some paths passed may not exist. For example, build errors protobuf
54 // file may not exist since the build was successful.
55 if err != nil {
56 continue
57 }
58
59 if fi.IsDir() {
usta4f5d2c12022-10-28 23:32:01 -040060 if l, err := ioutil.ReadDir(p); err != nil {
61 _, _ = fmt.Fprintf(os.Stderr, "Failed to find files under %s\n", p)
62 } else {
Patrice Arruda7d235cc2020-12-09 22:43:26 +000063 files := make([]string, 0, len(l))
64 for _, fi := range l {
65 files = append(files, filepath.Join(p, fi.Name()))
66 }
67 metricsFiles = append(metricsFiles, pruneMetricsFiles(files)...)
68 }
69 } else {
70 metricsFiles = append(metricsFiles, p)
71 }
72 }
73 return metricsFiles
74}
75
Yu Liu6e13b402021-07-27 14:29:06 -070076// UploadMetrics uploads a set of metrics files to a server for analysis.
77// The metrics files are first copied to a temporary directory
78// and the uploader is then executed in the background to allow the user/system
79// to continue working. Soong communicates to the uploader through the
80// upload_proto raw protobuf file.
Patrice Arruda7d235cc2020-12-09 22:43:26 +000081func UploadMetrics(ctx Context, config Config, simpleOutput bool, buildStarted time.Time, paths ...string) {
Patrice Arruda7cc20742020-06-10 18:48:01 +000082 ctx.BeginTrace(metrics.RunSetupTool, "upload_metrics")
83 defer ctx.EndTrace()
84
Patrice Arruda219eef32020-06-01 17:29:30 +000085 uploader := config.MetricsUploaderApp()
Patrice Arruda219eef32020-06-01 17:29:30 +000086 if uploader == "" {
Patrice Arruda92dc64f2020-11-16 16:29:16 -080087 // If the uploader path was not specified, no metrics shall be uploaded.
Patrice Arruda219eef32020-06-01 17:29:30 +000088 return
89 }
90
Patrice Arruda7d235cc2020-12-09 22:43:26 +000091 // Several of the files might be directories.
92 metricsFiles := pruneMetricsFiles(paths)
Patrice Arruda219eef32020-06-01 17:29:30 +000093 if len(metricsFiles) == 0 {
94 return
95 }
96
Patrice Arruda7cc20742020-06-10 18:48:01 +000097 // The temporary directory cannot be deleted as the metrics uploader is started
98 // in the background and requires to exist until the operation is done. The
99 // uploader can delete the directory as it is specified in the upload proto.
Patrice Arruda92dc64f2020-11-16 16:29:16 -0800100 tmpDir, err := tmpDir("", "upload_metrics")
Patrice Arruda7cc20742020-06-10 18:48:01 +0000101 if err != nil {
102 ctx.Fatalf("failed to create a temporary directory to store the list of metrics files: %v\n", err)
103 }
104
105 for i, src := range metricsFiles {
106 dst := filepath.Join(tmpDir, filepath.Base(src))
107 if _, err := copyFile(src, dst); err != nil {
108 ctx.Fatalf("failed to copy %q to %q: %v\n", src, dst, err)
109 }
110 metricsFiles[i] = dst
111 }
112
Patrice Arruda219eef32020-06-01 17:29:30 +0000113 // For platform builds, the branch and target name is hardcoded to specific
114 // values for later extraction of the metrics in the data metrics pipeline.
115 data, err := proto.Marshal(&upload_proto.Upload{
Patrice Arruda73c790f2020-07-13 23:01:18 +0000116 CreationTimestampMs: proto.Uint64(uint64(buildStarted.UnixNano() / int64(time.Millisecond))),
Patrice Arruda219eef32020-06-01 17:29:30 +0000117 CompletionTimestampMs: proto.Uint64(uint64(time.Now().UnixNano() / int64(time.Millisecond))),
118 BranchName: proto.String("developer-metrics"),
119 TargetName: proto.String("platform-build-systems-metrics"),
120 MetricsFiles: metricsFiles,
Patrice Arruda7cc20742020-06-10 18:48:01 +0000121 DirectoriesToDelete: []string{tmpDir},
Patrice Arruda219eef32020-06-01 17:29:30 +0000122 })
123 if err != nil {
124 ctx.Fatalf("failed to marshal metrics upload proto buffer message: %v\n", err)
125 }
126
Patrice Arruda7cc20742020-06-10 18:48:01 +0000127 pbFile := filepath.Join(tmpDir, uploadPbFilename)
Patrice Arruda219eef32020-06-01 17:29:30 +0000128 if err := ioutil.WriteFile(pbFile, data, 0644); err != nil {
129 ctx.Fatalf("failed to write the marshaled metrics upload protobuf to %q: %v\n", pbFile, err)
130 }
Patrice Arruda219eef32020-06-01 17:29:30 +0000131
Patrice Arruda7cc20742020-06-10 18:48:01 +0000132 // Start the uploader in the background as it takes several milliseconds to start the uploader
Patrice Arruda92dc64f2020-11-16 16:29:16 -0800133 // and prepare the metrics for upload. This affects small shell commands like "lunch".
Patrice Arruda7cc20742020-06-10 18:48:01 +0000134 cmd := Command(ctx, config, "upload metrics", uploader, "--upload-metrics", pbFile)
Patrice Arrudaf445ba12020-07-28 17:49:01 +0000135 if simpleOutput {
Patrice Arruda7cc20742020-06-10 18:48:01 +0000136 cmd.RunOrFatal()
137 } else {
138 cmd.RunAndStreamOrFatal()
139 }
Patrice Arruda219eef32020-06-01 17:29:30 +0000140}