| // Copyright 2019 The Fuchsia Authors. All rights reserved. |
| // Use of this source code is governed by a BSD-style license that can be |
| // found in the LICENSE file. |
| |
| package artifacts |
| |
| import ( |
| "fmt" |
| "io/ioutil" |
| "log" |
| "os" |
| "path/filepath" |
| "strings" |
| |
| "fuchsia.googlesource.com/host_target_testing/util" |
| ) |
| |
| // Archive allows interacting with the build artifact repository. |
| type Archive struct { |
| // Archive will store all downloaded artifacts into this directory. |
| dir string |
| |
| // lkgb (typically found in $FUCHSIA_DIR/prebuilt/tools/lkgb/lkgb) is |
| // used to look up the latest build id for a given builder. |
| lkgbPath string |
| |
| // artifacts (typically found in $FUCHSIA_DIR/prebuilt/tools/artifacts/artifacts) |
| // is used to download artifacts for a given build id. |
| artifactsPath string |
| } |
| |
| // NewArchive creates a new Archive. |
| func NewArchive(lkgbPath string, artifactsPath string, dir string) *Archive { |
| return &Archive{ |
| dir: dir, |
| lkgbPath: lkgbPath, |
| artifactsPath: artifactsPath, |
| } |
| } |
| |
| // GetBuildByName looks up a build artifact by the given name. |
| func (a *Archive) GetBuildByName(name string) (*Build, error) { |
| id, err := a.LookupBuildID(name) |
| if err != nil { |
| return nil, err |
| } |
| |
| return a.GetBuildByID(id) |
| } |
| |
| // GetBuildByID looks up a build artifact by the given id. |
| func (a *Archive) GetBuildByID(id string) (*Build, error) { |
| return &Build{ID: id, archive: a}, nil |
| } |
| |
| // LookupBuildID looks up the latest build id for a given builder. |
| func (a *Archive) LookupBuildID(builderName string) (string, error) { |
| stdout, stderr, err := util.RunCommand(a.lkgbPath, builderName) |
| if err != nil { |
| return "", fmt.Errorf("lkgb failed: %s: %s", err, string(stderr)) |
| } |
| return strings.TrimRight(string(stdout), "\n"), nil |
| } |
| |
| // Download an artifact from the build id `buildID` named `src` and write it |
| // into a directory `dst`. |
| func (a *Archive) download(buildID string, src string) (string, error) { |
| basename := filepath.Base(src) |
| buildDir := filepath.Join(a.dir, buildID) |
| path := filepath.Join(buildDir, basename) |
| |
| // Skip downloading if the file is already present in the build dir. |
| if _, err := os.Stat(path); err == nil { |
| return path, nil |
| } |
| |
| log.Printf("downloading %s to %s", src, path) |
| |
| if err := os.MkdirAll(buildDir, 0755); err != nil { |
| return "", err |
| } |
| |
| // We don't want to leak files if we are interrupted during a download. |
| // So we'll initally download into a temporary file, and only once it |
| // succeeds do we rename it into the real destination. |
| tmpfile, err := ioutil.TempFile(a.dir, basename) |
| defer func() { |
| if tmpfile != nil { |
| os.Remove(tmpfile.Name()) |
| } |
| }() |
| |
| args := []string{ |
| "cp", |
| "-build", buildID, |
| "-src", src, |
| "-dst", tmpfile.Name(), |
| } |
| |
| _, stderr, err := util.RunCommand(a.artifactsPath, args...) |
| if err != nil { |
| if len(stderr) != 0 { |
| return "", fmt.Errorf("artifacts failed: %s: %s", err, string(stderr)) |
| } |
| return "", fmt.Errorf("artifacts failed: %s", err) |
| } |
| |
| // Now that we've downloaded the file, do an atomic swap of the filename into place. |
| if err := os.Rename(tmpfile.Name(), path); err != nil { |
| return "", err |
| } |
| tmpfile = nil |
| |
| return path, nil |
| } |