Commit f005fedf authored by Andrew Gerrand's avatar Andrew Gerrand

misc/dashboard/builder: add timeout to all external command invocations

Fixes #4083.

R=golang-dev, r
CC=golang-dev
https://golang.org/cl/6498136
parent 0aad3cdc
...@@ -6,14 +6,16 @@ package main ...@@ -6,14 +6,16 @@ package main
import ( import (
"bytes" "bytes"
"fmt"
"io" "io"
"log" "log"
"os" "os"
"os/exec" "os/exec"
"time"
) )
// run is a simple wrapper for exec.Run/Close // run is a simple wrapper for exec.Run/Close
func run(envv []string, dir string, argv ...string) error { func run(timeout time.Duration, envv []string, dir string, argv ...string) error {
if *verbose { if *verbose {
log.Println("run", argv) log.Println("run", argv)
} }
...@@ -21,7 +23,10 @@ func run(envv []string, dir string, argv ...string) error { ...@@ -21,7 +23,10 @@ func run(envv []string, dir string, argv ...string) error {
cmd.Dir = dir cmd.Dir = dir
cmd.Env = envv cmd.Env = envv
cmd.Stderr = os.Stderr cmd.Stderr = os.Stderr
return cmd.Run() if err := cmd.Start(); err != nil {
return err
}
return waitWithTimeout(timeout, cmd)
} }
// runLog runs a process and returns the combined stdout/stderr, // runLog runs a process and returns the combined stdout/stderr,
...@@ -29,7 +34,7 @@ func run(envv []string, dir string, argv ...string) error { ...@@ -29,7 +34,7 @@ func run(envv []string, dir string, argv ...string) error {
// process combined stdout and stderr output, exit status and error. // process combined stdout and stderr output, exit status and error.
// The error returned is nil, if process is started successfully, // The error returned is nil, if process is started successfully,
// even if exit status is not successful. // even if exit status is not successful.
func runLog(envv []string, logfile, dir string, argv ...string) (string, int, error) { func runLog(timeout time.Duration, envv []string, logfile, dir string, argv ...string) (string, int, error) {
if *verbose { if *verbose {
log.Println("runLog", argv) log.Println("runLog", argv)
} }
...@@ -56,8 +61,23 @@ func runLog(envv []string, logfile, dir string, argv ...string) (string, int, er ...@@ -56,8 +61,23 @@ func runLog(envv []string, logfile, dir string, argv ...string) (string, int, er
return "", 1, startErr return "", 1, startErr
} }
exitStatus := 0 exitStatus := 0
if err := cmd.Wait(); err != nil { if err := waitWithTimeout(timeout, cmd); err != nil {
exitStatus = 1 // TODO(bradfitz): this is fake. no callers care, so just return a bool instead. exitStatus = 1 // TODO(bradfitz): this is fake. no callers care, so just return a bool instead.
} }
return b.String(), exitStatus, nil return b.String(), exitStatus, nil
} }
func waitWithTimeout(timeout time.Duration, cmd *exec.Cmd) error {
errc := make(chan error, 1)
go func() {
errc <- cmd.Wait()
}()
var err error
select {
case <-time.After(timeout):
cmd.Process.Kill()
err = fmt.Errorf("timed out after %v", timeout)
case err = <-errc:
}
return err
}
...@@ -54,6 +54,8 @@ var ( ...@@ -54,6 +54,8 @@ var (
buildCmd = flag.String("cmd", filepath.Join(".", allCmd), "Build command (specify relative to go/src/)") buildCmd = flag.String("cmd", filepath.Join(".", allCmd), "Build command (specify relative to go/src/)")
failAll = flag.Bool("fail", false, "fail all builds") failAll = flag.Bool("fail", false, "fail all builds")
parallel = flag.Bool("parallel", false, "Build multiple targets in parallel") parallel = flag.Bool("parallel", false, "Build multiple targets in parallel")
buildTimeout = flag.Duration("buildTimeout", 60*time.Minute, "Maximum time to wait for builds and tests")
cmdTimeout = flag.Duration("cmdTimeout", 2*time.Minute, "Maximum time to wait for an external command")
verbose = flag.Bool("v", false, "verbose") verbose = flag.Bool("v", false, "verbose")
) )
...@@ -220,7 +222,7 @@ func (b *Builder) build() bool { ...@@ -220,7 +222,7 @@ func (b *Builder) build() bool {
// Look for hash locally before running hg pull. // Look for hash locally before running hg pull.
if _, err := fullHash(goroot, hash[:12]); err != nil { if _, err := fullHash(goroot, hash[:12]); err != nil {
// Don't have hash, so run hg pull. // Don't have hash, so run hg pull.
if err := run(nil, goroot, hgCmd("pull")...); err != nil { if err := run(*cmdTimeout, nil, goroot, hgCmd("pull")...); err != nil {
log.Println("hg pull failed:", err) log.Println("hg pull failed:", err)
return false return false
} }
...@@ -243,12 +245,12 @@ func (b *Builder) buildHash(hash string) error { ...@@ -243,12 +245,12 @@ func (b *Builder) buildHash(hash string) error {
defer os.RemoveAll(workpath) defer os.RemoveAll(workpath)
// clone repo // clone repo
if err := run(nil, workpath, hgCmd("clone", goroot, "go")...); err != nil { if err := run(*cmdTimeout, nil, workpath, hgCmd("clone", goroot, "go")...); err != nil {
return err return err
} }
// update to specified revision // update to specified revision
if err := run(nil, filepath.Join(workpath, "go"), hgCmd("update", hash)...); err != nil { if err := run(*cmdTimeout, nil, filepath.Join(workpath, "go"), hgCmd("update", hash)...); err != nil {
return err return err
} }
...@@ -261,7 +263,7 @@ func (b *Builder) buildHash(hash string) error { ...@@ -261,7 +263,7 @@ func (b *Builder) buildHash(hash string) error {
cmd = filepath.Join(srcDir, cmd) cmd = filepath.Join(srcDir, cmd)
} }
startTime := time.Now() startTime := time.Now()
buildLog, status, err := runLog(b.envv(), logfile, srcDir, cmd) buildLog, status, err := runLog(*buildTimeout, b.envv(), logfile, srcDir, cmd)
runTime := time.Now().Sub(startTime) runTime := time.Now().Sub(startTime)
if err != nil { if err != nil {
return fmt.Errorf("%s: %s", *buildCmd, err) return fmt.Errorf("%s: %s", *buildCmd, err)
...@@ -353,28 +355,22 @@ func (b *Builder) buildSubrepo(goRoot, pkg, hash string) (string, error) { ...@@ -353,28 +355,22 @@ func (b *Builder) buildSubrepo(goRoot, pkg, hash string) (string, error) {
} }
// fetch package and dependencies // fetch package and dependencies
log, status, err := runLog(env, "", goRoot, goTool, "get", "-d", pkg) log, status, err := runLog(*cmdTimeout, env, "", goRoot, goTool, "get", "-d", pkg)
if err == nil && status != 0 { if err == nil && status != 0 {
err = fmt.Errorf("go exited with status %d", status) err = fmt.Errorf("go exited with status %d", status)
} }
if err != nil { if err != nil {
// 'go get -d' will fail for a subrepo because its top-level
// directory does not contain a go package. No matter, just
// check whether an hg directory exists and proceed.
hgDir := filepath.Join(goRoot, "src/pkg", pkg, ".hg")
if fi, e := os.Stat(hgDir); e != nil || !fi.IsDir() {
return log, err return log, err
} }
}
// hg update to the specified hash // hg update to the specified hash
pkgPath := filepath.Join(goRoot, "src/pkg", pkg) pkgPath := filepath.Join(goRoot, "src/pkg", pkg)
if err := run(nil, pkgPath, hgCmd("update", hash)...); err != nil { if err := run(*cmdTimeout, nil, pkgPath, hgCmd("update", hash)...); err != nil {
return "", err return "", err
} }
// test the package // test the package
log, status, err = runLog(env, "", goRoot, goTool, "test", "-short", pkg+"/...") log, status, err = runLog(*buildTimeout, env, "", goRoot, goTool, "test", "-short", pkg+"/...")
if err == nil && status != 0 { if err == nil && status != 0 {
err = fmt.Errorf("go exited with status %d", status) err = fmt.Errorf("go exited with status %d", status)
} }
...@@ -475,7 +471,7 @@ func commitWatcher() { ...@@ -475,7 +471,7 @@ func commitWatcher() {
} }
func hgClone(url, path string) error { func hgClone(url, path string) error {
return run(nil, *buildroot, hgCmd("clone", url, path)...) return run(*cmdTimeout, nil, *buildroot, hgCmd("clone", url, path)...)
} }
func hgRepoExists(path string) bool { func hgRepoExists(path string) bool {
...@@ -532,14 +528,14 @@ func commitPoll(key, pkg string) { ...@@ -532,14 +528,14 @@ func commitPoll(key, pkg string) {
} }
} }
if err := run(nil, pkgRoot, hgCmd("pull")...); err != nil { if err := run(*cmdTimeout, nil, pkgRoot, hgCmd("pull")...); err != nil {
log.Printf("hg pull: %v", err) log.Printf("hg pull: %v", err)
return return
} }
const N = 50 // how many revisions to grab const N = 50 // how many revisions to grab
data, _, err := runLog(nil, "", pkgRoot, hgCmd("log", data, _, err := runLog(*cmdTimeout, nil, "", pkgRoot, hgCmd("log",
"--encoding=utf-8", "--encoding=utf-8",
"--limit="+strconv.Itoa(N), "--limit="+strconv.Itoa(N),
"--template="+xmlLogTemplate)..., "--template="+xmlLogTemplate)...,
...@@ -627,7 +623,7 @@ func addCommit(pkg, hash, key string) bool { ...@@ -627,7 +623,7 @@ func addCommit(pkg, hash, key string) bool {
// fullHash returns the full hash for the given Mercurial revision. // fullHash returns the full hash for the given Mercurial revision.
func fullHash(root, rev string) (string, error) { func fullHash(root, rev string) (string, error) {
s, _, err := runLog(nil, "", root, s, _, err := runLog(*cmdTimeout, nil, "", root,
hgCmd("log", hgCmd("log",
"--encoding=utf-8", "--encoding=utf-8",
"--rev="+rev, "--rev="+rev,
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment