From a8460f7aa9523266febfca9df7a45025b7460152 Mon Sep 17 00:00:00 2001 From: Nate Finch Date: Fri, 21 Sep 2018 07:31:15 -0400 Subject: [PATCH] Ensure we print an error if running the compiled magefile fails (#171) * print an error if one comes back from running compiled binary * rework the code a little bit and make sure mage -version prints to stdout * export CmdRan and use it so we only log an error of not being able to run the magefile --- mage/main.go | 49 ++++++++++++++++++++++++++--------------------- mage/main_test.go | 13 +++++++++++++ sh/cmd.go | 10 ++++++++-- 3 files changed, 48 insertions(+), 24 deletions(-) diff --git a/mage/main.go b/mage/main.go index dd4fef86..3ebf2a08 100644 --- a/mage/main.go +++ b/mage/main.go @@ -99,7 +99,8 @@ type Invocation struct { // files in the given directory with the given args (do not include the command // name in the args). func ParseAndRun(stdout, stderr io.Writer, stdin io.Reader, args []string) int { - log := log.New(stderr, "", 0) + errlog := log.New(stderr, "", 0) + out := log.New(stdout, "", 0) inv, cmd, err := Parse(stderr, stdout, args) inv.Stderr = stderr inv.Stdin = stdin @@ -107,35 +108,35 @@ func ParseAndRun(stdout, stderr io.Writer, stdin io.Reader, args []string) int { return 0 } if err != nil { - log.Println("Error:", err) + errlog.Println("Error:", err) return 2 } switch cmd { case Version: - log.Println("Mage Build Tool", gitTag) - log.Println("Build Date:", timestamp) - log.Println("Commit:", commitHash) - log.Println("built with:", runtime.Version()) + out.Println("Mage Build Tool", gitTag) + out.Println("Build Date:", timestamp) + out.Println("Commit:", commitHash) + out.Println("built with:", runtime.Version()) return 0 case Init: if err := generateInit(inv.Dir); err != nil { - log.Println("Error:", err) + errlog.Println("Error:", err) return 1 } - log.Println(initFile, "created") + out.Println(initFile, "created") return 0 case Clean: if err := removeContents(inv.CacheDir); err != nil { - log.Println("Error:", err) + out.Println("Error:", err) return 1 } if err := removeContents(filepath.Join(inv.CacheDir, mainfileSubdir)); err != nil { - log.Println("Error:", err) + out.Println("Error:", err) return 1 } - log.Println(inv.CacheDir, "cleaned") + out.Println(inv.CacheDir, "cleaned") return 0 case CompileStatic: return Invoke(inv) @@ -258,7 +259,7 @@ Options: // Invoke runs Mage with the given arguments. func Invoke(inv Invocation) int { - log := log.New(inv.Stderr, "", 0) + errlog := log.New(inv.Stderr, "", 0) if inv.GoCmd == "" { inv.GoCmd = "go" } @@ -271,18 +272,18 @@ func Invoke(inv Invocation) int { files, err := Magefiles(inv.Dir, inv.GoCmd, inv.Stderr, inv.Debug) if err != nil { - log.Println("Error determining list of magefiles:", err) + errlog.Println("Error determining list of magefiles:", err) return 1 } if len(files) == 0 { - log.Println("No .go files marked with the mage build tag in this directory.") + errlog.Println("No .go files marked with the mage build tag in this directory.") return 1 } debug.Printf("found magefiles: %s", strings.Join(files, ", ")) exePath, err := ExeName(inv.GoCmd, inv.CacheDir, files) if err != nil { - log.Println("Error getting exe name:", err) + errlog.Println("Error getting exe name:", err) return 1 } if inv.CompileOut != "" { @@ -308,7 +309,7 @@ func Invoke(inv Invocation) int { debug.Println("ignoring existing executable") } else { debug.Println("Running existing exe") - return RunCompiled(inv, exePath) + return RunCompiled(inv, exePath, errlog) } case os.IsNotExist(err): debug.Println("no existing exe, creating new") @@ -329,14 +330,14 @@ func Invoke(inv Invocation) int { debug.Println("parsing files") info, err := parse.Package(inv.Dir, fnames) if err != nil { - log.Println("Error parsing magefiles:", err) + errlog.Println("Error parsing magefiles:", err) return 1 } main := filepath.Join(inv.Dir, mainfile) err = GenerateMainfile(main, inv.CacheDir, info) if err != nil { - log.Println("Error:", err) + errlog.Println("Error:", err) return 1 } if !inv.Keep { @@ -344,7 +345,7 @@ func Invoke(inv Invocation) int { } files = append(files, main) if err := Compile(inv.Dir, inv.GoCmd, exePath, files, inv.Debug, inv.Stderr, inv.Stdout); err != nil { - log.Println("Error:", err) + errlog.Println("Error:", err) return 1 } if !inv.Keep { @@ -360,7 +361,7 @@ func Invoke(inv Invocation) int { return 0 } - return RunCompiled(inv, exePath) + return RunCompiled(inv, exePath, errlog) } func moveMainToCache(cachedir, main, hash string) { @@ -610,7 +611,7 @@ func generateInit(dir string) error { } // RunCompiled runs an already-compiled mage command with the given args, -func RunCompiled(inv Invocation, exePath string) int { +func RunCompiled(inv Invocation, exePath string, errlog *log.Logger) int { debug.Println("running binary", exePath) c := exec.Command(exePath, inv.Args...) c.Stderr = inv.Stderr @@ -634,7 +635,11 @@ func RunCompiled(inv Invocation, exePath string) int { c.Env = append(c.Env, fmt.Sprintf("MAGEFILE_TIMEOUT=%s", inv.Timeout.String())) } debug.Print("running magefile with mage vars:\n", strings.Join(filter(c.Env, "MAGEFILE"), "\n")) - return sh.ExitStatus(c.Run()) + err := c.Run() + if !sh.CmdRan(err) { + errlog.Printf("failed to run compiled magefile: %v", err) + } + return sh.ExitStatus(err) } func filter(list []string, prefix string) []string { diff --git a/mage/main_test.go b/mage/main_test.go index 3854825c..7c8c4860 100644 --- a/mage/main_test.go +++ b/mage/main_test.go @@ -750,6 +750,19 @@ func TestInvalidAlias(t *testing.T) { } } +func TestRunCompiledPrintsError(t *testing.T) { + stderr := &bytes.Buffer{} + logger := log.New(stderr, "", 0) + code := RunCompiled(Invocation{}, "thiswon'texist", logger) + if code != 1 { + t.Errorf("expected code 1 but got %v", code) + } + + if strings.TrimSpace(stderr.String()) == "" { + t.Fatal("expected to get output to stderr when a run fails, but got nothing.") + } +} + func TestClean(t *testing.T) { if err := os.RemoveAll(mg.CacheDir()); err != nil { t.Error("error removing cache dir:", err) diff --git a/sh/cmd.go b/sh/cmd.go index 23fc3722..2cddfb52 100644 --- a/sh/cmd.go +++ b/sh/cmd.go @@ -128,10 +128,16 @@ func run(env map[string]string, stdout, stderr io.Writer, cmd string, args ...st c.Stdin = os.Stdin log.Println("exec:", cmd, strings.Join(args, " ")) err = c.Run() - return cmdRan(err), ExitStatus(err), err + return CmdRan(err), ExitStatus(err), err } -func cmdRan(err error) bool { +// CmdRan examines the error to determine if it was generated as a result of a +// command running via os/exec.Command. If the error is nil, or the command ran +// (even if it exited with a non-zero exit code), CmdRan reports true. If the +// error is an unrecognized type, or it is an error from exec.Command that says +// the command failed to run (usually due to the command not existing or not +// being executable), it reports false. +func CmdRan(err error) bool { if err == nil { return true }