blob: 7c3edcdaf4fae5370f95af8f36089c90f00c04c3 [file] [log] [blame]
// Copyright 2020 The LUCI Authors.
//
// Licensed under the Apache License, Version 2.0 (the "License");
// you may not use this file except in compliance with the License.
// You may obtain a copy of the License at
//
// http://www.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
// See the License for the specific language governing permissions and
// limitations under the License.
package lib
import (
"fmt"
"log"
"os"
"os/exec"
"path/filepath"
"time"
"github.com/maruel/subcommands"
"go.chromium.org/luci/client/isolate"
"go.chromium.org/luci/common/errors"
"go.chromium.org/luci/common/system/filesystem"
)
// CmdRun returns an object for the `run` subcommand.
func CmdRun() *subcommands.Command {
return &subcommands.Command{
UsageLine: "run <options> -- <extra args>",
ShortDesc: "Runs the test executable in an isolated (temporary) directory",
LongDesc: `All the dependencies are mapped into the temporary directory and the directory is cleaned up after the target exits. Argument processing stops at -- and these arguments are appended to the command line of the target to run.
For example, use: isolate run -isolated foo.isolate -- --gtest_filter=Foo.Bar`,
CommandRun: func() subcommands.CommandRun {
r := runRun{}
r.baseCommandRun.Init()
r.isolateFlags.Init(&r.Flags)
r.Flags.StringVar(&r.relativeCwd, "relative-cwd", ".", "Relative cwd when running isolated.")
return &r
},
}
}
type runRun struct {
baseCommandRun
isolateFlags
relativeCwd string
}
func (r *runRun) Parse(a subcommands.Application, args []string) error {
if err := r.baseCommandRun.Parse(); err != nil {
return err
}
cwd, err := os.Getwd()
if err != nil {
return err
}
if err := r.isolateFlags.Parse(cwd); err != nil {
return err
}
return nil
}
func (r *runRun) Run(a subcommands.Application, args []string, _ subcommands.Env) int {
if err := r.Parse(a, args); err != nil {
fmt.Fprintf(a.GetErr(), "%s: %s\n", a.GetName(), err)
return 1
}
if err := r.main(a, args); err != nil {
fmt.Fprintf(a.GetErr(), "%s: %s\n", a.GetName(), err)
return 1
}
return 0
}
func (r *runRun) main(a subcommands.Application, args []string) error {
deps, rootDir, _, err := isolate.ProcessIsolate(&r.ArchiveOptions)
if err != nil {
return errors.Annotate(err, "failed to process isolate").Err()
}
var cleanupErr error
if err := (&filesystem.TempDir{
Dir: filepath.Dir(rootDir),
Prefix: time.Now().Format("2006-01-02"),
CleanupErrFunc: func(tdir string, err error) {
cleanupErr = errors.Annotate(err, "failed to clean up %s", tdir).Err()
},
}).With(func(outDir string) error {
if err := recreateTree(outDir, rootDir, deps); err != nil {
return errors.Annotate(err, "failed to recreate tree").Err()
}
if len(args) == 0 {
return errors.Reason("command cannot be empty").Err()
}
cwd := filepath.Clean(filepath.Join(outDir, r.relativeCwd))
// |cwd| should never exist because it is under the temporary directory |outDir|.
err = filesystem.MakeDirs(cwd)
if err != nil {
return errors.Annotate(err, "failed to create cwd=%s", cwd).Err()
}
log.Printf("Running %v, cwd=%s\n", args, cwd)
cmd := exec.Command(args[0], args[1:]...)
cmd.Dir = cwd
cmd.Stdout = os.Stdout
cmd.Stderr = os.Stderr
if err := cmd.Run(); err != nil {
return errors.Annotate(err, "failed to run: %v", args).Err()
}
return nil
}); err != nil {
return err
}
return cleanupErr
}