perkeep/misc/buildbot/bot.go

1286 lines
28 KiB
Go

/*
Copyright 2012 The Camlistore 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.
*/
// The buildbot binary is Camlistore's continuous builder.
package main
import (
"bytes"
"flag"
"fmt"
"html/template"
"io/ioutil"
"log"
"math/rand"
"net/http"
"os"
"os/exec"
"os/signal"
"path/filepath"
"regexp"
"runtime"
"strings"
"sync"
"syscall"
"time"
)
const (
interval = 60 * time.Second // polling frequency
warmup = 60 * time.Second // duration before we test if devcam server has started properly
historySize = 30
)
var (
fast = flag.Bool("fast", false, "run dummy steps instead of the actual tasks, for debugging")
gotipdir = flag.String("gotip", "./go", "path to the Go tip tree")
help = flag.Bool("h", false, "show this help")
host = flag.String("host", "0.0.0.0:8080", "listening hostname and port")
nocleanup = flag.Bool("nocleanup", false, "do not clean up the tmp gopath when failing")
verbose = flag.Bool("verbose", false, "print what's going on")
)
var (
testFile = []string{"AUTHORS", "CONTRIBUTORS"}
camliHeadHash string
cachedCamliRoot string
camliRoot string
camputCacheDir string
currentTask task
dbg *debugger
defaultDir string
defaultPATH string
// doBuildCamli0 is for when we're in the go 1 test suite, and
// doBuildCamli1 for the go tip test suite. They are set when
// the camli source tree has been updated and indicate that
// 'make' should be run.
// doBuildGo indicates the go source tree has changed, hence
// go tip should be rebuilt, and in the go tip test suite case
// camlistore should be cleaned (pkg and bin) and rebuilt.
doBuildGo, doBuildCamli0, doBuildCamli1 bool
goPath string
go1Path string
goTipPath string
goTipDir string
goTipHash string
lastErr error
historylk sync.Mutex
currentTestSuite *testSuite
history History
// For "If-Modified-Since" requests on the status page.
// Updated every time a new test suite starts or ends.
lastModified time.Time
)
var NameToCmd = map[string]string{
"prepRepo1": "hg pull",
"prepRepo2": "hg update -C default",
"prepRepo3": "hg --config extensions.purge= purge --all",
"prepRepo4": "hg log -r tip --template {node}",
"prepRepo5": "git reset --hard HEAD",
"prepRepo6": "git clean -Xdf",
"prepRepo7": "git pull",
"prepRepo8": "git rev-parse HEAD",
"buildGoTip1": "./make.bash",
"buildCamli1": "go run make.go -v",
"buildCamli2": "go build -o devcam ./dev/devcam/",
"buildCamli3": "make presubmit",
"runCamli": "./devcam server --wipe --mysql",
"hitCamliUi1": "http://localhost:3179/ui/",
"camget": "./dev-camget ",
"camput1": "./dev-camput file --permanode " + testFile[0],
"camput2": "./dev-camput file --vivify " + testFile[1],
"camput3": "./dev-camput file --filenodes pkg",
}
func usage() {
fmt.Fprintf(os.Stderr, "\t buildbot \n")
flag.PrintDefaults()
os.Exit(2)
}
type debugger struct {
lg *log.Logger
}
func (dbg *debugger) Printf(format string, v ...interface{}) {
if *verbose {
dbg.lg.Printf(format, v...)
}
}
func (dbg *debugger) Println(v ...interface{}) {
if v == nil {
return
}
if *verbose {
dbg.lg.Println(v...)
}
}
func setup() {
var err error
defaultDir, err = os.Getwd()
if err != nil {
log.Fatal(err)
}
defaultPATH = os.Getenv("PATH")
if defaultPATH == "" {
log.Fatal("PATH not set")
}
dbg = &debugger{log.New(os.Stderr, "", log.LstdFlags)}
// check go tip tree
goTipDir, err = filepath.Abs(*gotipdir)
if err != nil {
log.Fatalf("Problem with Go tip dir: %v", err)
}
if _, err := os.Stat(goTipDir); err != nil {
log.Fatalf("Problem with Go tip dir: %v", err)
}
// setup temp gopath(s)
goTipPath, err = ioutil.TempDir("", "camlibot-gotippath")
if err != nil {
log.Fatalf("problem with tempdir: %v", err)
}
srcDir := filepath.Join(goTipPath, "src")
err = os.Mkdir(srcDir, 0755)
if err != nil {
log.Fatalf("problem with src dir: %v", err)
}
go1Path, err = ioutil.TempDir("", "camlibot-go1path")
if err != nil {
log.Fatalf("problem with tempdir: %v", err)
}
srcDir = filepath.Join(go1Path, "src")
err = os.Mkdir(srcDir, 0755)
if err != nil {
log.Fatalf("problem with src dir: %v", err)
}
goPath = go1Path
err = os.Setenv("GOPATH", goPath)
if err != nil {
log.Fatalf("problem setting up GOPATH: %v", err)
}
// set up the camlistore tree
camliRoot = filepath.Join(srcDir, "camlistore.org")
cacheDir := filepath.Join(os.TempDir(), "camlibot-cache")
cachedCamliRoot = filepath.Join(cacheDir, "camlistore.org")
if _, err := os.Stat(cachedCamliRoot); err != nil {
// git clone
if _, err := os.Stat(cacheDir); err != nil {
err = os.Mkdir(cacheDir, 0755)
if err != nil {
log.Fatalf("problem with cache dir: %v", err)
}
}
setCurrentTask("git clone https://camlistore.googlesource.com/camlistore " + camliRoot)
_, err = runCmd(getCurrentTask().Cmd)
if err != nil {
log.Fatalf("problem with git clone: %v", err)
}
} else {
// get cache
err = os.Rename(cachedCamliRoot, camliRoot)
if err != nil {
log.Fatal(err)
}
}
// recording camput cache dir, so we can clean it up fast everytime
homeDir := os.Getenv("HOME")
if homeDir == "" {
log.Fatal("HOME not set")
}
camputCacheDir = filepath.Join(homeDir, ".cache", "camlistore")
}
func cleanTempGopath() {
if *nocleanup {
return
}
err := os.Rename(camliRoot, cachedCamliRoot)
if err != nil {
panic(err)
}
err = os.RemoveAll(go1Path)
if err != nil {
panic(err)
}
err = os.RemoveAll(goTipPath)
if err != nil {
panic(err)
}
}
func switchGoPath(isTip bool) {
if isTip {
goPath = goTipPath
} else {
goPath = go1Path
}
newRoot := filepath.Join(goPath, "src", "camlistore.org")
if newRoot != camliRoot {
err := os.Rename(camliRoot, newRoot)
if err != nil {
panic(err)
}
camliRoot = newRoot
err = os.Setenv("GOPATH", goPath)
if err != nil {
log.Fatalf("problem setting up GOPATH: %v", err)
}
}
}
func handleSignals() {
c := make(chan os.Signal)
sigs := []os.Signal{syscall.SIGINT, syscall.SIGTERM}
signal.Notify(c, sigs...)
for {
sig := <-c
sysSig, ok := sig.(syscall.Signal)
if !ok {
log.Fatal("Not a unix signal")
}
switch sysSig {
case syscall.SIGINT, syscall.SIGTERM:
log.Printf("SIGINT: cleaning up %v before terminating", goPath)
cleanTempGopath()
os.Exit(0)
default:
panic("should not get other signals here")
}
}
}
func prepRepo() {
doBuildGo, doBuildCamli0, doBuildCamli1 = false, false, false
if *fast {
if currentTestSuite == nil {
currentTestSuite = &testSuite{
Run: make([]*task, 0, 1),
GoHash: goTipHash,
CamliHash: camliHeadHash,
IsTip: false,
}
}
return
}
// gotip
err := os.Chdir(goTipDir)
if err != nil {
log.Fatal(err)
}
defer func() {
err := os.Chdir(defaultDir)
if err != nil {
panic(err)
}
}()
tasks := []string{
NameToCmd["prepRepo1"],
NameToCmd["prepRepo2"],
NameToCmd["prepRepo4"],
}
hash := ""
for _, v := range tasks {
setCurrentTask(v)
out, err := runCmd(v)
if err != nil {
if v == NameToCmd["prepRepo1"] {
dbg.Printf("Go repo could not be updated: %v\n", err)
continue
}
log.Fatal(err)
}
hash = strings.TrimRight(out, "\n")
}
dbg.Println("previous head in go tree: " + goTipHash)
dbg.Println("current head in go tree: " + hash)
if hash != "" && hash != goTipHash {
goTipHash = hash
doBuildGo = true
dbg.Println("Changes in go tree detected, go tip will be rebuilt")
}
// camli
err = os.Chdir(camliRoot)
if err != nil {
log.Fatal(err)
}
tasks = []string{
NameToCmd["prepRepo5"],
NameToCmd["prepRepo6"],
NameToCmd["prepRepo7"],
NameToCmd["prepRepo8"],
}
hash = ""
for _, v := range tasks {
setCurrentTask(v)
out, err := runCmd(v)
if err != nil {
if v == NameToCmd["prepRepo7"] {
dbg.Printf("Camli repo could not be updated: %v\n", err)
continue
}
log.Fatal(err)
}
hash = strings.TrimRight(out, "\n")
}
dbg.Println("previous head in camli tree: " + camliHeadHash)
dbg.Println("current head in camli tree: " + hash)
if hash != "" && hash != camliHeadHash {
camliHeadHash = hash
doBuildCamli0, doBuildCamli1 = true, true
dbg.Println("Changes in camli tree detected, camlistore will be rebuilt")
}
}
func addToPATH(gobin string) {
splitter := ":"
switch runtime.GOOS {
case "windows":
splitter = ";"
case "plan9":
panic("unsupported")
}
p := gobin + splitter + defaultPATH
err := os.Setenv("PATH", p)
if err != nil {
log.Fatalf("Could not set PATH to %v: %v", p, err)
}
}
func restorePATH() {
err := os.Setenv("PATH", defaultPATH)
if err != nil {
log.Fatalf("Could not set PATH to %v: %v", defaultPATH, err)
}
}
type task struct {
lk sync.Mutex
// actual command that is run for this task. it includes the command and
// all the arguments, space separated. shell metacharacters not supported.
Cmd string
start time.Time
Duration time.Duration
}
func getCurrentTask() *task {
currentTask.lk.Lock()
defer currentTask.lk.Unlock()
return &task{Cmd: currentTask.Cmd, start: currentTask.start}
}
func setCurrentTask(cmd string) {
currentTask.lk.Lock()
defer currentTask.lk.Unlock()
currentTask.Cmd = cmd
currentTask.start = time.Now()
}
type testSuite struct {
Run []*task
CamliHash string
GoHash string
failedTask int
Err error
Start time.Time
IsTip bool
}
type History [][2]*testSuite
func addRun(tsk *task, tskErr error) {
historylk.Lock()
defer historylk.Unlock()
duration := time.Now().Sub(tsk.start)
tsk.Duration = duration
if len(currentTestSuite.Run) == 0 {
currentTestSuite.Start = tsk.start
}
if tskErr != nil && currentTestSuite.Err == nil {
currentTestSuite.Err = tskErr
currentTestSuite.failedTask = len(currentTestSuite.Run)
} else {
currentTestSuite.failedTask = -1
}
currentTestSuite.Run = append(currentTestSuite.Run, tsk)
}
func addTestSuite() {
historylk.Lock()
defer historylk.Unlock()
if len(currentTestSuite.Run) < 1 {
return
}
if !currentTestSuite.IsTip {
// Go 1
entry := [2]*testSuite{currentTestSuite, nil}
history = append(history, entry)
} else {
// Go tip
entry := [2]*testSuite{history[len(history)-1][0], currentTestSuite}
if len(history) > historySize {
history = append(history[1:historySize], entry)
} else {
history[len(history)-1] = entry
}
}
}
func runCmd(tsk string) (string, error) {
dbg.Println(getCurrentTask().Cmd)
fields := strings.Fields(tsk)
var args []string
if len(fields) > 1 {
args = fields[1:]
}
cmd := exec.Command(fields[0], args...)
var stdout, stderr bytes.Buffer
cmd.Stdout = &stdout
cmd.Stderr = &stderr
err := cmd.Run()
if err != nil {
return "", fmt.Errorf("%v: %v\n", stdout.String(), stderr.String())
}
return stdout.String(), nil
}
func gofast(n int) {
for i := 0; i < n; i++ {
tsk := &task{start: time.Now()}
fail := rand.Intn(10)
if fail < 1 {
dbg.Println("random fail")
addRun(tsk, fmt.Errorf("random fail"))
continue
}
addRun(tsk, nil)
}
}
func buildGoTip() error {
if *fast {
gofast(3)
return nil
}
err := os.Chdir(goTipDir)
if err != nil {
log.Fatal(err)
}
defer func() {
err := os.Chdir(defaultDir)
if err != nil {
panic(err)
}
}()
tasks := []string{
NameToCmd["prepRepo3"],
}
for _, v := range tasks {
setCurrentTask(v)
tsk := getCurrentTask()
_, err := runCmd(tsk.Cmd)
addRun(tsk, err)
if err != nil {
return err
}
}
err = os.Chdir(filepath.Join(goTipDir, "src"))
if err != nil {
log.Fatal(err)
}
defer func() {
err := os.Chdir(defaultDir)
if err != nil {
panic(err)
}
}()
setCurrentTask(NameToCmd["buildGoTip1"])
tsk := getCurrentTask()
_, err = runCmd(tsk.Cmd)
addRun(tsk, err)
if err != nil {
return err
}
return nil
}
func buildCamli(isTip bool) error {
if *fast {
gofast(5)
return nil
}
switchGoPath(isTip)
err := os.Chdir(camliRoot)
if err != nil {
log.Fatal(err)
}
defer func() {
err := os.Chdir(defaultDir)
if err != nil {
panic(err)
}
}()
if doBuildGo && isTip {
if *verbose {
dbg.Println("cleaning up gopath/pkg and gopath/bin for full rebuild")
}
// erase and rebuild everything
pkgdir := filepath.Join("..", "..", "pkg")
err := os.RemoveAll(pkgdir)
if err != nil {
log.Fatalf("failed to remove %v: %v", pkgdir, err)
}
bindir := filepath.Join("..", "..", "bin")
err = os.RemoveAll(bindir)
if err != nil {
log.Fatalf("failed to remove %v: %v", bindir, err)
}
}
if *verbose {
setCurrentTask("go version")
out, err := runCmd(getCurrentTask().Cmd)
if err != nil {
log.Fatalf("failed to run 'go version': %v", err)
}
out = strings.TrimRight(out, "\n")
dbg.Printf("Building camlistore in %v with: %v\n", goPath, out)
}
tasks := []string{}
if doBuildCamli0 || doBuildCamli1 || (doBuildGo && isTip) {
tasks = append(tasks, NameToCmd["buildCamli1"], NameToCmd["buildCamli2"])
}
tasks = append(tasks, NameToCmd["buildCamli3"])
for _, v := range tasks {
setCurrentTask(v)
tsk := getCurrentTask()
_, err := runCmd(tsk.Cmd)
addRun(tsk, err)
if err != nil {
return err
}
}
if doBuildGo && isTip {
doBuildGo = false
doBuildCamli1 = false
}
if isTip {
doBuildCamli1 = false
} else {
doBuildCamli0 = false
}
return nil
}
func runCamli() (*os.Process, error) {
if *fast {
gofast(1)
return nil, nil
}
err := os.Chdir(camliRoot)
if err != nil {
log.Fatal(err)
}
defer func() {
err := os.Chdir(defaultDir)
if err != nil {
panic(err)
}
}()
setCurrentTask(NameToCmd["runCamli"])
dbg.Println(getCurrentTask().Cmd)
fields := strings.Fields(getCurrentTask().Cmd)
args := fields[1:]
cmd := exec.Command(fields[0], args...)
var output []byte
errc := make(chan error, 1)
go func() {
output, err = cmd.CombinedOutput()
errc <- err
}()
select {
case err := <-errc:
dbg.Printf("dev server DEAD:\n%s\n", output)
tsk := getCurrentTask()
addRun(tsk, err)
return nil, fmt.Errorf("%v: server failed to start\n", tsk.Cmd)
case <-time.After(warmup):
dbg.Println("devcam server OK")
addRun(getCurrentTask(), nil)
}
return cmd.Process, nil
}
func killCamli(proc *os.Process) {
if *fast {
return
}
dbg.Println("killing dev server")
err := proc.Signal(os.Interrupt)
if err != nil {
log.Fatalf("Could not kill server: %v", err)
}
dbg.Println("")
}
func hitURL(url string) error {
setCurrentTask(fmt.Sprintf("http.Get(\"%s\")", url))
dbg.Println(getCurrentTask().Cmd)
resp, err := http.Get(url)
if err != nil {
return fmt.Errorf("%v: %v\n", getCurrentTask().Cmd, err)
}
defer resp.Body.Close()
if resp.StatusCode != 200 {
return fmt.Errorf("%v, got StatusCode: %d\n", getCurrentTask().Cmd, resp.StatusCode)
}
return nil
}
func hitCamliUi() error {
if *fast {
gofast(2)
return nil
}
urls := []string{
NameToCmd["hitCamliUi1"],
}
var err error
for _, v := range urls {
lerr := hitURL(v)
addRun(getCurrentTask(), lerr)
if lerr != nil {
err = lerr
}
}
return err
}
func camputOne(vivify bool) error {
if *fast {
gofast(3)
return nil
}
err := os.Chdir(camliRoot)
if err != nil {
log.Fatal(err)
}
defer func() {
err := os.Chdir(defaultDir)
if err != nil {
panic(err)
}
}()
// clean up camput caches
err = os.RemoveAll(camputCacheDir)
if err != nil {
log.Fatalf("Problem cleaning up camputCacheDir %v: %v", camputCacheDir, err)
}
// push the file to camli
tskString := NameToCmd["camput1"]
if vivify {
tskString = NameToCmd["camput2"]
}
setCurrentTask(tskString)
tsk := getCurrentTask()
out, err := runCmd(tsk.Cmd)
addRun(tsk, err)
if err != nil {
return err
}
// TODO(mpl): parsing camput output is kinda weak.
firstSHA1 := regexp.MustCompile(`.*(sha1-[a-zA-Z0-9]+)\nsha1-[a-zA-Z0-9]+\nsha1-[a-zA-Z0-9]+\n.*`)
if vivify {
firstSHA1 = regexp.MustCompile(`.*(sha1-[a-zA-Z0-9]+)\n.*`)
}
m := firstSHA1.FindStringSubmatch(out)
if m == nil {
return fmt.Errorf("%v: unexpected camput output\n", getCurrentTask().Cmd)
}
blobref := m[1]
// get the file's json to find out the file's blobref
setCurrentTask(NameToCmd["camget"] + blobref)
tsk = getCurrentTask()
out, err = runCmd(tsk.Cmd)
addRun(tsk, err)
if err != nil {
return err
}
blobrefPattern := regexp.MustCompile(`"blobRef": "(sha1-[a-zA-Z0-9]+)",\n.*`)
m = blobrefPattern.FindStringSubmatch(out)
if m == nil {
return fmt.Errorf("%v: unexpected camget output\n", getCurrentTask().Cmd)
}
blobref = m[1]
// finally, get the file back
setCurrentTask(NameToCmd["camget"] + blobref)
tsk = getCurrentTask()
out, err = runCmd(tsk.Cmd)
addRun(tsk, err)
if err != nil {
return err
}
// and compare it with the original
wantFile := testFile[0]
if vivify {
wantFile = testFile[1]
}
fileContents, err := ioutil.ReadFile(wantFile)
if err != nil {
log.Fatalf("Could not read %v: %v", wantFile, err)
}
if string(fileContents) != out {
return fmt.Errorf("%v: contents fetched with camget differ from %v contents", getCurrentTask().Cmd, wantFile)
}
return nil
}
func camputMany() error {
if *fast {
gofast(1)
return nil
}
err := os.Chdir(camliRoot)
if err != nil {
log.Fatal(err)
}
defer func() {
err := os.Chdir(defaultDir)
if err != nil {
panic(err)
}
}()
// upload the full camli pkg tree
setCurrentTask(NameToCmd["camput3"])
tsk := getCurrentTask()
_, err = runCmd(tsk.Cmd)
addRun(tsk, err)
if err != nil {
return err
}
return nil
}
func handleErr(err error, proc *os.Process) {
lastModified = time.Now()
lastErr = err
dbg.Printf("%v", err)
if proc != nil {
killCamli(proc)
}
addTestSuite()
}
func main() {
flag.Usage = usage
flag.Parse()
if *help {
usage()
}
setup()
defer cleanTempGopath()
go handleSignals()
http.HandleFunc(okPrefix, okHandler)
http.HandleFunc(failPrefix, failHandler)
http.HandleFunc(currentPrefix, progressHandler)
http.HandleFunc("/", statusHandler)
go http.ListenAndServe(*host, nil)
tryCount := 0
for {
if lastErr == nil || tryCount > 1 {
tryCount = 0
lastErr = nil
prepRepo()
}
if doBuildGo || doBuildCamli0 || doBuildCamli1 || lastErr != nil {
for _, isTip := range [2]bool{false, true} {
lastModified = time.Now()
restorePATH()
currentTestSuite = &testSuite{
Run: make([]*task, 0, 1),
GoHash: goTipHash,
CamliHash: camliHeadHash,
IsTip: isTip,
}
if isTip {
addToPATH(filepath.Join(goTipDir, "bin"))
if doBuildGo {
err := buildGoTip()
if err != nil {
handleErr(err, nil)
continue
}
}
}
err := buildCamli(isTip)
if err != nil {
handleErr(err, nil)
continue
}
proc, err := runCamli()
if err != nil {
handleErr(err, nil)
continue
}
err = hitCamliUi()
if err != nil {
handleErr(err, proc)
continue
}
doVivify := false
err = camputOne(doVivify)
if err != nil {
handleErr(err, proc)
continue
}
doVivify = true
err = camputOne(doVivify)
if err != nil {
handleErr(err, proc)
continue
}
err = camputMany()
if err != nil {
handleErr(err, proc)
continue
}
dbg.Println("All good.")
killCamli(proc)
addTestSuite()
lastModified = time.Now()
}
tryCount++
}
setCurrentTask(fmt.Sprintf("time.Sleep(%d)", interval))
dbg.Println(getCurrentTask().Cmd)
time.Sleep(interval)
}
}
var (
okPrefix = "/ok/"
failPrefix = "/fail/"
currentPrefix = "/current"
statusTpl = template.Must(template.New("status").Funcs(tmplFuncs).Parse(reportHTML))
taskTpl = template.Must(template.New("task").Parse(taskHTML))
testSuiteTpl = template.Must(template.New("ok").Parse(testSuiteHTML))
)
var tmplFuncs = template.FuncMap{
"camliRepoURL": camliRepoURL,
"goRepoURL": goRepoURL,
"shortHash": shortHash,
}
// unlocked; history needs to be protected from the caller.
func getPastTestSuite(key string) (*testSuite, error) {
idx := 0
date := ""
if strings.HasPrefix(key, "gotip-") {
date = strings.Replace(key, "gotip-", "", -1)
idx = 1
} else {
date = strings.Replace(key, "go1-", "", -1)
}
for _, v := range history {
if v[idx].Start.String() == date {
return v[idx], nil
}
}
return nil, fmt.Errorf("%v not found in history", date)
}
type progressData struct {
Ts *testSuite
Current string
}
// modtime is the modification time of the resource to be served, or IsZero().
// return value is whether this request is now complete.
func checkLastModified(w http.ResponseWriter, r *http.Request, modtime time.Time) bool {
if modtime.IsZero() {
return false
}
// The Date-Modified header truncates sub-second precision, so
// use mtime < t+1s instead of mtime <= t to check for unmodified.
if t, err := time.Parse(http.TimeFormat, r.Header.Get("If-Modified-Since")); err == nil && modtime.Before(t.Add(1*time.Second)) {
h := w.Header()
delete(h, "Content-Type")
delete(h, "Content-Length")
w.WriteHeader(http.StatusNotModified)
return true
}
w.Header().Set("Last-Modified", modtime.UTC().Format(http.TimeFormat))
return false
}
func okHandler(w http.ResponseWriter, r *http.Request) {
t := strings.Replace(r.URL.Path, okPrefix, "", -1)
historylk.Lock()
defer historylk.Unlock()
ts, err := getPastTestSuite(t)
if err != nil || len(ts.Run) == 0 {
http.NotFound(w, r)
return
}
lastTask := ts.Run[len(ts.Run)-1]
lastModTime := lastTask.start.Add(lastTask.Duration)
if checkLastModified(w, r, lastModTime) {
return
}
dat := &progressData{
Ts: ts,
}
err = testSuiteTpl.Execute(w, dat)
if err != nil {
log.Printf("ok template: %v\n", err)
}
}
func progressHandler(w http.ResponseWriter, r *http.Request) {
historylk.Lock()
defer historylk.Unlock()
currentTask := getCurrentTask()
if checkLastModified(w, r, currentTask.start) {
return
}
dat := &progressData{
Ts: currentTestSuite,
Current: currentTask.Cmd,
}
err := testSuiteTpl.Execute(w, dat)
if err != nil {
log.Printf("progress template: %v\n", err)
}
}
type TaskReport struct {
Cmd string
Err error
}
func failHandler(w http.ResponseWriter, r *http.Request) {
t := strings.Replace(r.URL.Path, failPrefix, "", -1)
historylk.Lock()
defer historylk.Unlock()
ts, err := getPastTestSuite(t)
if err != nil || len(ts.Run) == 0 || ts.failedTask == -1 {
http.NotFound(w, r)
return
}
failedTask := ts.Run[ts.failedTask]
lastModTime := failedTask.start.Add(failedTask.Duration)
if checkLastModified(w, r, lastModTime) {
return
}
taskReport := &TaskReport{Cmd: failedTask.Cmd, Err: ts.Err}
err = taskTpl.Execute(w, taskReport)
if err != nil {
log.Printf("fail template: %v\n", err)
}
}
type status struct {
Hs History
Ts *testSuite
}
func invertedHistory() (inverted History) {
inverted = make(History, len(history))
endpos := len(history) - 1
for k, v := range history {
inverted[endpos-k] = v
}
return inverted
}
func statusHandler(w http.ResponseWriter, r *http.Request) {
historylk.Lock()
defer historylk.Unlock()
stat := &status{
Hs: invertedHistory(),
Ts: currentTestSuite,
}
if checkLastModified(w, r, lastModified) {
return
}
err := statusTpl.Execute(w, stat)
if err != nil {
log.Printf("status template: %v\n", err)
}
}
// shortHash returns a short version of a hash.
func shortHash(hash string) string {
if len(hash) > 12 {
hash = hash[:12]
}
return hash
}
func goRepoURL(hash string) string {
return "https://code.google.com/p/go/source/detail?r=" + hash
}
func camliRepoURL(hash string) string {
return "https://camlistore.googlesource.com/camlistore/+/" + hash
}
// style inspired from $GOROOT/misc/dashboard/app/build/ui.html
var styleHTML = `
<style>
body {
font-family: sans-serif;
padding: 0; margin: 0;
}
h1, h2 {
margin: 0;
padding: 5px;
}
h1 {
background: #eee;
}
h2 {
margin-top: 20px;
}
.build, .packages {
margin: 5px;
border-collapse: collapse;
}
.build td, .build th, .packages td, .packages th {
vertical-align: top;
padding: 2px 4px;
font-size: 10pt;
}
.build tr.commit:nth-child(2n) {
background-color: #f0f0f0;
}
.build .hash {
font-family: monospace;
font-size: 9pt;
}
.build .result {
text-align: center;
width: 2em;
}
.col-hash, .col-result {
border-right: solid 1px #ccc;
}
.build .arch {
font-size: 66%;
font-weight: normal;
}
.build .time {
color: #666;
}
.build .ok {
font-size: 83%;
}
a.ok {
text-decoration:none;
}
.build .desc, .build .time, .build .user {
white-space: nowrap;
}
.paginate {
padding: 0.5em;
}
.paginate a {
padding: 0.5em;
background: #eee;
color: blue;
}
.paginate a.inactive {
color: #999;
}
.fail {
color: #C00;
}
</style>
`
var reportHTML = `
<!DOCTYPE HTML>
<html>
<head>
<title>Camlistore tests Dashboard</title>` +
styleHTML + `
</head>
<body>
<h1>Camlibot status</h1>
<table class="build">
<colgroup class="col-hash" span="1"></colgroup>
<colgroup class="build" span="1"></colgroup>
<colgroup class="build" span="1"></colgroup>
<colgroup class="user" span="1"></colgroup>
<colgroup class="user" span="1"></colgroup>
<tr>
<!-- extra row to make alternating colors use dark for first result -->
</tr>
<tr>
<th>&nbsp;</th>
<th colspan="1">Go tip hash</th>
<th colspan="1">Camli HEAD hash</th>
<th colspan="1">Go1</th>
<th colspan="1">Gotip</th>
</tr>
{{if .Ts}}
{{if .Ts.IsTip | not}}
<tr class="commit">
<td class="hash">{{.Ts.Start}}</td>
<td class="hash">
<a href="{{goRepoURL .Ts.GoHash}}">{{shortHash .Ts.GoHash}}</a>
</td>
<td class="hash">
<a href="{{camliRepoURL .Ts.CamliHash}}">{{shortHash .Ts.CamliHash}}</a>
</td>
<td class="result">
<a href="` + currentPrefix + `" class="ok">In progress</a>
</td>
</tr>
{{end}}
{{end}}
{{if .Hs}}
{{range $tss := .Hs}}
<tr class="commit">
{{range $k, $ts := $tss}}
{{if $k | not}}
<td class="hash">{{$ts.Start}}</td>
<td class="hash">
<a href="{{goRepoURL $ts.GoHash}}">{{shortHash $ts.GoHash}}</a>
</td>
<td class="hash">
<a href="{{camliRepoURL $ts.CamliHash}}">{{shortHash $ts.CamliHash}}</a>
</td>
<td class="result">
{{if $ts.Err}}
<a href="` + failPrefix + `go1-{{$ts.Start}}" class="fail">fail</a>
{{else}}
<a href="` + okPrefix + `go1-{{$ts.Start}}" class="ok">ok</a>
{{end}}
</td>
{{else}}
<td class="result">
{{if $ts}}
{{if $ts.Err}}
<a href="` + failPrefix + `gotip-{{$ts.Start}}" class="fail">fail</a>
{{else}}
<a href="` + okPrefix + `gotip-{{$ts.Start}}" class="ok">ok</a>
{{end}}
{{else}}
<a href="` + currentPrefix + `" class="ok">In progress</a>
{{end}}
</td>
{{end}}
{{end}}
</tr>
{{end}}
{{end}}
</table>
</body>
</html>
`
var testSuiteHTML = `
<!DOCTYPE HTML>
<html>
<head>
<title>Camlistore tests Dashboard</title>` +
styleHTML + `
</head>
<body>
{{if .Ts}}
<h2> Testsuite for {{if .Ts.IsTip}}Go tip{{else}}Go 1{{end}} at {{.Ts.Start}} </h2>
<table class="build">
<colgroup class="col-result" span="1"></colgroup>
<colgroup class="col-result" span="1"></colgroup>
<tr>
<!-- extra row to make alternating colors use dark for first result -->
</tr>
<tr>
<th colspan="1">Step</th>
<th colspan="1">Duration</th>
</tr>
{{range $k, $v := .Ts.Run}}
<tr>
<td>{{$v.Cmd}}</td>
<td>{{$v.Duration}}</td>
</tr>
{{end}}
{{if .Current}}
<tr>
<td>{{.Current}}</td>
<td>(running...)</td>
</tr>
{{end}}
</table>
{{end}}
</body>
</html>
`
var taskHTML = `
<!DOCTYPE HTML>
<html>
<head>
<title>Camlistore tests Dashboard</title>
</head>
<body>
{{if .Cmd}}
<h2>Command:</h2>
<p>{{.Cmd}}</p>
{{end}}
{{if .Err}}
<h2>Error:</h2>
<pre>
{{.Err}}
</pre>
{{end}}
</body>
</html>
`