perkeep/misc/buildbot/master/master.go

1300 lines
32 KiB
Go
Raw Normal View History

/*
Copyright 2013 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 is Camlistore's continuous builder.
// This master program monitors changes to the Go and Camlistore trees,
// then rebuilds and restarts a builder when a change dictates as much.
// It receives a report from a builder when it has finished running
// a test suite, but it can also poll a builder before completion
// to get a progress report.
// It also serves the web requests.
package main
import (
"bytes"
buildbot/master: add Basic Auth support. Moved BasicAuth parsing and localhost detection code from pkg/auth -> pkg/httputil for use by buildbot master. Added user config file for remote access. The file's name is "masterbot-config.json" and is located in osutil.CamliConfigDir(), which on Unix will resolve to $XDG_CONFIG_HOME/camlistore/, if XDG_CONFIG_HOME set, or ~/.config/camlistore/. On Windows it will be under %APPDATA%\Camlistore\. The expected format is a json object with usernames as the keys and sha1 sums of the password as the values, i.e.: { "user1": "1234567890abcdef12341234567890abcdef1234", "user2": "1234abcdef12345678901234abcdef1234567890" } This file is polled at a 1 minute interval and reparsed if the file's modification time is more recent then the previous parse attempt. It is ok for the file to go missing, it will zero out the remote user list. A malformed file will result in the master exiting. New commandline flags, -tlsCertFile & -tlsKeyFile, added. Specifying both will enable TLS on the listener specified by -host. The go source contains generate_cert.go in crypto/tls that can be used to generate self-signed cert.pem and key.pem for testing. Added -skiptlscheck commandline option to builder. This allows the builder to report to https:// addresses with self-signed certs as we don't currently have a way to specify the cert chains to be used for TLS verification. This is a stop-gap solution. When launching a master that listens for secure connections, we currently need tell the builders to skip certificate validation. Add '-builderopts="-skiptlscheck"' to the master's commandline to skip cerfication verification. Change-Id: I0750b5c9fa8f4def67fc05a841087b50abded2f7
2013-10-31 05:00:17 +00:00
"crypto/sha1"
"encoding/hex"
"encoding/json"
"flag"
"fmt"
"html/template"
"io"
"io/ioutil"
"log"
"net"
"net/http"
"os"
"os/exec"
"os/signal"
"path/filepath"
"regexp"
"runtime"
"strings"
"sync"
"syscall"
"time"
buildbot/master: add Basic Auth support. Moved BasicAuth parsing and localhost detection code from pkg/auth -> pkg/httputil for use by buildbot master. Added user config file for remote access. The file's name is "masterbot-config.json" and is located in osutil.CamliConfigDir(), which on Unix will resolve to $XDG_CONFIG_HOME/camlistore/, if XDG_CONFIG_HOME set, or ~/.config/camlistore/. On Windows it will be under %APPDATA%\Camlistore\. The expected format is a json object with usernames as the keys and sha1 sums of the password as the values, i.e.: { "user1": "1234567890abcdef12341234567890abcdef1234", "user2": "1234abcdef12345678901234abcdef1234567890" } This file is polled at a 1 minute interval and reparsed if the file's modification time is more recent then the previous parse attempt. It is ok for the file to go missing, it will zero out the remote user list. A malformed file will result in the master exiting. New commandline flags, -tlsCertFile & -tlsKeyFile, added. Specifying both will enable TLS on the listener specified by -host. The go source contains generate_cert.go in crypto/tls that can be used to generate self-signed cert.pem and key.pem for testing. Added -skiptlscheck commandline option to builder. This allows the builder to report to https:// addresses with self-signed certs as we don't currently have a way to specify the cert chains to be used for TLS verification. This is a stop-gap solution. When launching a master that listens for secure connections, we currently need tell the builders to skip certificate validation. Add '-builderopts="-skiptlscheck"' to the master's commandline to skip cerfication verification. Change-Id: I0750b5c9fa8f4def67fc05a841087b50abded2f7
2013-10-31 05:00:17 +00:00
"camlistore.org/pkg/httputil"
"camlistore.org/pkg/osutil"
)
const (
interval = 60 * time.Second // polling frequency
historySize = 30
maxStderrSize = 1 << 20 // Keep last 1 MB of logging.
)
var (
altCamliRevURL = flag.String("camlirevurl", "", "alternative URL to query about the latest camlistore revision hash (e.g camlistore.org/latesthash), to alleviate hitting too often the Camlistore git repo.")
builderOpts = flag.String("builderopts", "", "list of comma separated options that will be passed to the builders (ex: '-verbose=true,-faketests=true,-skipgo1build=true'). Mainly for debugging.")
builderPort = flag.String("builderport", "8081", "listening port for the builder bot")
builderSrc = flag.String("buildersrc", "", "Go source file for the builder bot. For testing changes to the builder bot that haven't been committed yet.")
getGo = flag.Bool("getgo", false, "Do not use the system's Go to build the builder, use the downloaded gotip instead.")
help = flag.Bool("h", false, "show this help")
host = flag.String("host", "0.0.0.0:8080", "listening hostname and port")
peers = flag.String("peers", "", "comma separated list of host:port masters (besides this one) our builders will report to.")
verbose = flag.Bool("verbose", false, "print what's going on")
buildbot/master: add Basic Auth support. Moved BasicAuth parsing and localhost detection code from pkg/auth -> pkg/httputil for use by buildbot master. Added user config file for remote access. The file's name is "masterbot-config.json" and is located in osutil.CamliConfigDir(), which on Unix will resolve to $XDG_CONFIG_HOME/camlistore/, if XDG_CONFIG_HOME set, or ~/.config/camlistore/. On Windows it will be under %APPDATA%\Camlistore\. The expected format is a json object with usernames as the keys and sha1 sums of the password as the values, i.e.: { "user1": "1234567890abcdef12341234567890abcdef1234", "user2": "1234abcdef12345678901234abcdef1234567890" } This file is polled at a 1 minute interval and reparsed if the file's modification time is more recent then the previous parse attempt. It is ok for the file to go missing, it will zero out the remote user list. A malformed file will result in the master exiting. New commandline flags, -tlsCertFile & -tlsKeyFile, added. Specifying both will enable TLS on the listener specified by -host. The go source contains generate_cert.go in crypto/tls that can be used to generate self-signed cert.pem and key.pem for testing. Added -skiptlscheck commandline option to builder. This allows the builder to report to https:// addresses with self-signed certs as we don't currently have a way to specify the cert chains to be used for TLS verification. This is a stop-gap solution. When launching a master that listens for secure connections, we currently need tell the builders to skip certificate validation. Add '-builderopts="-skiptlscheck"' to the master's commandline to skip cerfication verification. Change-Id: I0750b5c9fa8f4def67fc05a841087b50abded2f7
2013-10-31 05:00:17 +00:00
certFile = flag.String("tlsCertFile", "", "TLS public key in PEM format. Must be used with -tlsKeyFile")
keyFile = flag.String("tlsKeyFile", "", "TLS private key in PEM format. Must be used with -tlsCertFile")
)
var (
camliHeadHash string
camliRoot string
dbg *debugger
defaultDir string
doBuildGo, doBuildCamli bool
goTipDir string
goTipHash string
historylk sync.Mutex
history = make(map[string][]*biTestSuite) // key is the OS_Arch on which the tests were run
inProgresslk sync.Mutex
inProgress *testSuite
// Process of the local builder bot, so we can kill it
// when we get killed.
builderProc *os.Process
// For "If-Modified-Since" requests on the status page.
// Updated every time a new test suite starts or ends.
lastModified time.Time
// Override the os.Stderr used by the default logger so we can provide
// more debug info on status page.
logStderr = newLockedBuffer()
multiWriter io.Writer
buildbot/master: add Basic Auth support. Moved BasicAuth parsing and localhost detection code from pkg/auth -> pkg/httputil for use by buildbot master. Added user config file for remote access. The file's name is "masterbot-config.json" and is located in osutil.CamliConfigDir(), which on Unix will resolve to $XDG_CONFIG_HOME/camlistore/, if XDG_CONFIG_HOME set, or ~/.config/camlistore/. On Windows it will be under %APPDATA%\Camlistore\. The expected format is a json object with usernames as the keys and sha1 sums of the password as the values, i.e.: { "user1": "1234567890abcdef12341234567890abcdef1234", "user2": "1234abcdef12345678901234abcdef1234567890" } This file is polled at a 1 minute interval and reparsed if the file's modification time is more recent then the previous parse attempt. It is ok for the file to go missing, it will zero out the remote user list. A malformed file will result in the master exiting. New commandline flags, -tlsCertFile & -tlsKeyFile, added. Specifying both will enable TLS on the listener specified by -host. The go source contains generate_cert.go in crypto/tls that can be used to generate self-signed cert.pem and key.pem for testing. Added -skiptlscheck commandline option to builder. This allows the builder to report to https:// addresses with self-signed certs as we don't currently have a way to specify the cert chains to be used for TLS verification. This is a stop-gap solution. When launching a master that listens for secure connections, we currently need tell the builders to skip certificate validation. Add '-builderopts="-skiptlscheck"' to the master's commandline to skip cerfication verification. Change-Id: I0750b5c9fa8f4def67fc05a841087b50abded2f7
2013-10-31 05:00:17 +00:00
// Set after flag parsing based on certFile & keyFile.
useTLS bool
)
// lockedBuffer protects all Write calls with a mutex. Users of lockedBuffer
// must wrap any calls to Bytes, and use of the resulting slice with calls to
// Lock/Unlock.
type lockedBuffer struct {
sync.Mutex // guards ringBuffer
*ringBuffer
}
func newLockedBuffer() *lockedBuffer {
return &lockedBuffer{ringBuffer: newRingBuffer(maxStderrSize)}
}
func (lb *lockedBuffer) Write(b []byte) (int, error) {
lb.Lock()
defer lb.Unlock()
return lb.ringBuffer.Write(b)
}
type ringBuffer struct {
buf []byte
off int // End of ring buffer.
l int // Length of ring buffer filled.
}
func newRingBuffer(maxSize int) *ringBuffer {
return &ringBuffer{
buf: make([]byte, maxSize),
}
}
func (rb *ringBuffer) Bytes() []byte {
if (rb.off - rb.l) >= 0 {
// Partially full buffer with no wrap.
return rb.buf[rb.off-rb.l : rb.off]
}
// Buffer has been wrapped, copy second half then first half.
start := rb.off - rb.l
if start < 0 {
start = rb.off
}
b := make([]byte, 0, cap(rb.buf))
b = append(b, rb.buf[start:]...)
b = append(b, rb.buf[:start]...)
return b
}
func (rb *ringBuffer) Write(buf []byte) (int, error) {
ringLen := cap(rb.buf)
for i, b := range buf {
rb.buf[(rb.off+i)%ringLen] = b
}
rb.off = (rb.off + len(buf)) % ringLen
rb.l = rb.l + len(buf)
if rb.l > ringLen {
rb.l = ringLen
}
return len(buf), nil
}
buildbot/master: add Basic Auth support. Moved BasicAuth parsing and localhost detection code from pkg/auth -> pkg/httputil for use by buildbot master. Added user config file for remote access. The file's name is "masterbot-config.json" and is located in osutil.CamliConfigDir(), which on Unix will resolve to $XDG_CONFIG_HOME/camlistore/, if XDG_CONFIG_HOME set, or ~/.config/camlistore/. On Windows it will be under %APPDATA%\Camlistore\. The expected format is a json object with usernames as the keys and sha1 sums of the password as the values, i.e.: { "user1": "1234567890abcdef12341234567890abcdef1234", "user2": "1234abcdef12345678901234abcdef1234567890" } This file is polled at a 1 minute interval and reparsed if the file's modification time is more recent then the previous parse attempt. It is ok for the file to go missing, it will zero out the remote user list. A malformed file will result in the master exiting. New commandline flags, -tlsCertFile & -tlsKeyFile, added. Specifying both will enable TLS on the listener specified by -host. The go source contains generate_cert.go in crypto/tls that can be used to generate self-signed cert.pem and key.pem for testing. Added -skiptlscheck commandline option to builder. This allows the builder to report to https:// addresses with self-signed certs as we don't currently have a way to specify the cert chains to be used for TLS verification. This is a stop-gap solution. When launching a master that listens for secure connections, we currently need tell the builders to skip certificate validation. Add '-builderopts="-skiptlscheck"' to the master's commandline to skip cerfication verification. Change-Id: I0750b5c9fa8f4def67fc05a841087b50abded2f7
2013-10-31 05:00:17 +00:00
var userAuthFile = filepath.Join(osutil.CamliConfigDir(), "masterbot-config.json")
type userAuth struct {
sync.Mutex // guards userPass map.
userPass map[string]string
configFile string
pollInterval time.Duration
lastModTime time.Time
}
func newUserAuth(configFile string) (*userAuth, error) {
ua := &userAuth{
configFile: configFile,
pollInterval: time.Minute,
}
if _, err := os.Stat(configFile); err != nil {
if !os.IsNotExist(err) {
return nil, err
}
// It is okay to have no remote users configured.
log.Printf("no user config file found %q, remote reporting disabled",
configFile)
}
go ua.pollUsers()
return ua, nil
}
func (ua *userAuth) resetMissing(err error) error {
if os.IsNotExist(err) {
ua.Lock()
if ua.userPass != nil {
log.Printf("%q disappeared, remote reporting disabled",
ua.configFile)
}
ua.userPass = nil
ua.Unlock()
return nil
}
return err
}
func (ua *userAuth) loadUsers() error {
s, err := os.Stat(ua.configFile)
if err != nil {
return ua.resetMissing(err)
}
defer func() {
ua.lastModTime = s.ModTime()
}()
if ua.lastModTime.Before(s.ModTime()) {
r, err := os.Open(ua.configFile)
if err != nil {
return ua.resetMissing(err)
}
defer r.Close()
dec := json.NewDecoder(r)
// Use tmp map so failed parsing doesn't accidentally wipe out user
// list.
tmp := make(map[string]string)
err = dec.Decode(&tmp)
if err != nil {
return err
}
ua.Lock()
ua.userPass = tmp
ua.Unlock()
log.Println("Found", len(ua.userPass), "remote users in config",
ua.configFile)
}
return nil
}
func (ua *userAuth) pollUsers() {
for {
if err := ua.loadUsers(); err != nil {
log.Fatalf("Error loading user file %q: %v", ua.configFile, err)
}
time.Sleep(ua.pollInterval)
}
}
func hashPassword(pw string) string {
h := sha1.New()
fmt.Fprint(h, pw)
return hex.EncodeToString(h.Sum(nil))
}
func (ua *userAuth) auth(r *http.Request) bool {
user, pass, err := httputil.BasicAuth(r)
if user == "" || pass == "" || err != nil {
return false
}
ua.Lock()
defer ua.Unlock()
passHash, ok := ua.userPass[user]
if !ok {
return false
}
return passHash == hashPassword(pass)
}
var devcamBin = filepath.Join("bin", "devcam")
var (
hgCloneGoTipCmd = newTask("hg", "clone", "-u", "tip", "https://code.google.com/p/go")
hgPullCmd = newTask("hg", "pull")
hgLogCmd = newTask("hg", "log", "-r", "tip", "--template", "{node}")
gitCloneCmd = newTask("git", "clone", "https://camlistore.googlesource.com/camlistore")
gitPullCmd = newTask("git", "pull")
gitRevCmd = newTask("git", "rev-parse", "HEAD")
buildGoCmd = newTask("./make.bash")
)
func usage() {
fmt.Fprintf(os.Stderr, "\t masterBot \n")
flag.PrintDefaults()
os.Exit(2)
}
type debugger struct {
lg *log.Logger
}
func (dbg *debugger) Printf(format string, v ...interface{}) {
if dbg != nil && *verbose {
dbg.lg.Printf(format, v...)
}
}
func (dbg *debugger) Println(v ...interface{}) {
if v == nil {
return
}
if dbg != nil && *verbose {
dbg.lg.Println(v...)
}
}
type task struct {
Program string
Args []string
Start time.Time
Duration time.Duration
Err string
}
func newTask(program string, args ...string) *task {
return &task{Program: program, Args: args}
}
func (t *task) String() string {
return fmt.Sprintf("%v %v", t.Program, t.Args)
}
func (t *task) run() (string, error) {
var err error
dbg.Println(t.String())
var stdout, stderr bytes.Buffer
cmd := exec.Command(t.Program, t.Args...)
cmd.Stdout = &stdout
cmd.Stderr = &stderr
if err = cmd.Run(); err != nil {
return "", fmt.Errorf("%v: %v", err, stderr.String())
}
return stdout.String(), nil
}
type testSuite struct {
Run []*task
CamliHash string
GoHash string
Err string
Start time.Time
IsTip bool
}
type biTestSuite struct {
Local bool
Go1 *testSuite
GoTip *testSuite
}
func addTestSuites(OSArch string, ts *biTestSuite) {
if ts == nil {
return
}
historylk.Lock()
if ts.Local {
inProgresslk.Lock()
defer inProgresslk.Unlock()
}
defer historylk.Unlock()
historyOSArch := history[OSArch]
if len(historyOSArch) > historySize {
historyOSArch = append(historyOSArch[1:historySize], ts)
} else {
historyOSArch = append(historyOSArch, ts)
}
history[OSArch] = historyOSArch
if ts.Local {
inProgress = nil
}
lastModified = time.Now()
}
func main() {
flag.Usage = usage
flag.Parse()
if *help {
usage()
}
buildbot/master: add Basic Auth support. Moved BasicAuth parsing and localhost detection code from pkg/auth -> pkg/httputil for use by buildbot master. Added user config file for remote access. The file's name is "masterbot-config.json" and is located in osutil.CamliConfigDir(), which on Unix will resolve to $XDG_CONFIG_HOME/camlistore/, if XDG_CONFIG_HOME set, or ~/.config/camlistore/. On Windows it will be under %APPDATA%\Camlistore\. The expected format is a json object with usernames as the keys and sha1 sums of the password as the values, i.e.: { "user1": "1234567890abcdef12341234567890abcdef1234", "user2": "1234abcdef12345678901234abcdef1234567890" } This file is polled at a 1 minute interval and reparsed if the file's modification time is more recent then the previous parse attempt. It is ok for the file to go missing, it will zero out the remote user list. A malformed file will result in the master exiting. New commandline flags, -tlsCertFile & -tlsKeyFile, added. Specifying both will enable TLS on the listener specified by -host. The go source contains generate_cert.go in crypto/tls that can be used to generate self-signed cert.pem and key.pem for testing. Added -skiptlscheck commandline option to builder. This allows the builder to report to https:// addresses with self-signed certs as we don't currently have a way to specify the cert chains to be used for TLS verification. This is a stop-gap solution. When launching a master that listens for secure connections, we currently need tell the builders to skip certificate validation. Add '-builderopts="-skiptlscheck"' to the master's commandline to skip cerfication verification. Change-Id: I0750b5c9fa8f4def67fc05a841087b50abded2f7
2013-10-31 05:00:17 +00:00
useTLS = *certFile != "" && *keyFile != ""
go handleSignals()
buildbot/master: add Basic Auth support. Moved BasicAuth parsing and localhost detection code from pkg/auth -> pkg/httputil for use by buildbot master. Added user config file for remote access. The file's name is "masterbot-config.json" and is located in osutil.CamliConfigDir(), which on Unix will resolve to $XDG_CONFIG_HOME/camlistore/, if XDG_CONFIG_HOME set, or ~/.config/camlistore/. On Windows it will be under %APPDATA%\Camlistore\. The expected format is a json object with usernames as the keys and sha1 sums of the password as the values, i.e.: { "user1": "1234567890abcdef12341234567890abcdef1234", "user2": "1234abcdef12345678901234abcdef1234567890" } This file is polled at a 1 minute interval and reparsed if the file's modification time is more recent then the previous parse attempt. It is ok for the file to go missing, it will zero out the remote user list. A malformed file will result in the master exiting. New commandline flags, -tlsCertFile & -tlsKeyFile, added. Specifying both will enable TLS on the listener specified by -host. The go source contains generate_cert.go in crypto/tls that can be used to generate self-signed cert.pem and key.pem for testing. Added -skiptlscheck commandline option to builder. This allows the builder to report to https:// addresses with self-signed certs as we don't currently have a way to specify the cert chains to be used for TLS verification. This is a stop-gap solution. When launching a master that listens for secure connections, we currently need tell the builders to skip certificate validation. Add '-builderopts="-skiptlscheck"' to the master's commandline to skip cerfication verification. Change-Id: I0750b5c9fa8f4def67fc05a841087b50abded2f7
2013-10-31 05:00:17 +00:00
ua, err := newUserAuth(userAuthFile)
if err != nil {
log.Fatalf("Error creating user auth wrapper: %v", err)
}
authWrapper := func(f http.HandlerFunc) http.HandlerFunc {
return func(w http.ResponseWriter, r *http.Request) {
if !(httputil.IsLocalhost(r) || ua.auth(r)) {
w.Header().Set("WWW-Authenticate", `Basic realm="buildbot master"`)
http.Error(w, "Unauthorized access", http.StatusUnauthorized)
return
}
f(w, r)
}
}
http.HandleFunc(okPrefix, okHandler)
http.HandleFunc(failPrefix, failHandler)
http.HandleFunc(progressPrefix, progressHandler)
http.HandleFunc(stderrPrefix, logHandler)
http.HandleFunc("/", statusHandler)
buildbot/master: add Basic Auth support. Moved BasicAuth parsing and localhost detection code from pkg/auth -> pkg/httputil for use by buildbot master. Added user config file for remote access. The file's name is "masterbot-config.json" and is located in osutil.CamliConfigDir(), which on Unix will resolve to $XDG_CONFIG_HOME/camlistore/, if XDG_CONFIG_HOME set, or ~/.config/camlistore/. On Windows it will be under %APPDATA%\Camlistore\. The expected format is a json object with usernames as the keys and sha1 sums of the password as the values, i.e.: { "user1": "1234567890abcdef12341234567890abcdef1234", "user2": "1234abcdef12345678901234abcdef1234567890" } This file is polled at a 1 minute interval and reparsed if the file's modification time is more recent then the previous parse attempt. It is ok for the file to go missing, it will zero out the remote user list. A malformed file will result in the master exiting. New commandline flags, -tlsCertFile & -tlsKeyFile, added. Specifying both will enable TLS on the listener specified by -host. The go source contains generate_cert.go in crypto/tls that can be used to generate self-signed cert.pem and key.pem for testing. Added -skiptlscheck commandline option to builder. This allows the builder to report to https:// addresses with self-signed certs as we don't currently have a way to specify the cert chains to be used for TLS verification. This is a stop-gap solution. When launching a master that listens for secure connections, we currently need tell the builders to skip certificate validation. Add '-builderopts="-skiptlscheck"' to the master's commandline to skip cerfication verification. Change-Id: I0750b5c9fa8f4def67fc05a841087b50abded2f7
2013-10-31 05:00:17 +00:00
http.HandleFunc(reportPrefix, authWrapper(reportHandler))
go func() {
log.Printf("Now starting to listen on %v", *host)
buildbot/master: add Basic Auth support. Moved BasicAuth parsing and localhost detection code from pkg/auth -> pkg/httputil for use by buildbot master. Added user config file for remote access. The file's name is "masterbot-config.json" and is located in osutil.CamliConfigDir(), which on Unix will resolve to $XDG_CONFIG_HOME/camlistore/, if XDG_CONFIG_HOME set, or ~/.config/camlistore/. On Windows it will be under %APPDATA%\Camlistore\. The expected format is a json object with usernames as the keys and sha1 sums of the password as the values, i.e.: { "user1": "1234567890abcdef12341234567890abcdef1234", "user2": "1234abcdef12345678901234abcdef1234567890" } This file is polled at a 1 minute interval and reparsed if the file's modification time is more recent then the previous parse attempt. It is ok for the file to go missing, it will zero out the remote user list. A malformed file will result in the master exiting. New commandline flags, -tlsCertFile & -tlsKeyFile, added. Specifying both will enable TLS on the listener specified by -host. The go source contains generate_cert.go in crypto/tls that can be used to generate self-signed cert.pem and key.pem for testing. Added -skiptlscheck commandline option to builder. This allows the builder to report to https:// addresses with self-signed certs as we don't currently have a way to specify the cert chains to be used for TLS verification. This is a stop-gap solution. When launching a master that listens for secure connections, we currently need tell the builders to skip certificate validation. Add '-builderopts="-skiptlscheck"' to the master's commandline to skip cerfication verification. Change-Id: I0750b5c9fa8f4def67fc05a841087b50abded2f7
2013-10-31 05:00:17 +00:00
if useTLS {
if err := http.ListenAndServeTLS(*host, *certFile, *keyFile, nil); err != nil {
log.Fatalf("Could not start listening (TLS) on %v: %v", *host, err)
}
} else {
if err := http.ListenAndServe(*host, nil); err != nil {
log.Fatalf("Could not start listening on %v: %v", *host, err)
}
}
}()
setup()
for {
if err := pollGoChange(); err != nil {
log.Print(err)
goto Sleep
}
if err := pollCamliChange(); err != nil {
log.Print(err)
goto Sleep
}
if doBuildGo || doBuildCamli {
if err := buildBuilder(); err != nil {
log.Printf("Could not build builder bot: %v", err)
goto Sleep
}
cmd, err := startBuilder(goTipHash, camliHeadHash)
if err != nil {
log.Printf("Could not start builder bot: %v", err)
goto Sleep
}
dbg.Println("Waiting for builder to finish")
if err := cmd.Wait(); err != nil {
log.Printf("builder finished with error: %v", err)
}
resetBuilderState()
}
Sleep:
tsk := newTask("time.Sleep", interval.String())
dbg.Println(tsk.String())
time.Sleep(interval)
}
}
func resetBuilderState() {
inProgresslk.Lock()
defer inProgresslk.Unlock()
builderProc = nil
inProgress = nil
}
func setup() {
// Install custom stderr for display in status webpage.
multiWriter = io.MultiWriter(logStderr, os.Stderr)
log.SetOutput(multiWriter)
var err error
defaultDir, err = os.Getwd()
if err != nil {
log.Fatalf("Could not get current dir: %v", err)
}
dbg = &debugger{log.New(multiWriter, "", log.LstdFlags)}
goTipDir, err = filepath.Abs("gotip")
if err != nil {
log.Fatal(err)
}
// if gotip dir exist, just reuse it
if _, err := os.Stat(goTipDir); err != nil {
if !os.IsNotExist(err) {
log.Fatalf("Could not stat %v: %v", goTipDir, err)
}
if _, err := hgCloneGoTipCmd.run(); err != nil {
log.Fatalf("Could not hg clone %v: %v", goTipDir, err)
}
if err := os.Rename("go", goTipDir); err != nil {
log.Fatalf("Could not rename go dir into %v: %v", goTipDir, err)
}
}
if _, err := exec.LookPath("go"); err != nil {
// Go was not found on this machine, but we've already
// downloaded gotip anyway, so let's install it and
// use it to build the builder bot.
*getGo = true
}
if *getGo {
// set PATH
splitter := ":"
switch runtime.GOOS {
case "windows":
splitter = ";"
case "plan9":
panic("unsupported OS")
}
p := os.Getenv("PATH")
if p == "" {
log.Fatal("PATH not set")
}
p = filepath.Join(goTipDir, "bin") + splitter + p
if err := os.Setenv("PATH", p); err != nil {
log.Fatalf("Could not set PATH to %v: %v", p, err)
}
// and check if we already have a gotip binary
if _, err := exec.LookPath("go"); err != nil {
// if not, build gotip
if err := buildGo(); err != nil {
log.Fatal(err)
}
}
}
// get camlistore source
if err := os.Chdir(defaultDir); err != nil {
log.Fatalf("Could not cd to %v: %v", defaultDir, err)
}
camliRoot, err = filepath.Abs("src/camlistore.org")
if err != nil {
log.Fatal(err)
}
// if camlistore dir already exists, reuse it
if _, err := os.Stat(camliRoot); err != nil {
if !os.IsNotExist(err) {
log.Fatalf("Could not stat %v: %v", camliRoot, err)
}
cloneCmd := newTask(gitCloneCmd.Program, append(gitCloneCmd.Args, camliRoot)...)
if _, err := cloneCmd.run(); err != nil {
log.Fatalf("Could not git clone into %v: %v", camliRoot, err)
}
}
// override GOPATH to only point to our freshly updated camlistore source.
if err := os.Setenv("GOPATH", defaultDir); err != nil {
log.Fatalf("Could not set GOPATH to %v: %v", defaultDir, err)
}
}
func buildGo() error {
if err := os.Chdir(filepath.Join(goTipDir, "src")); err != nil {
log.Fatalf("Could not cd to %v: %v", goTipDir, err)
}
if _, err := buildGoCmd.run(); err != nil {
return err
}
return nil
}
func handleSignals() {
c := make(chan os.Signal)
sigs := []os.Signal{syscall.SIGINT, syscall.SIGTERM, syscall.SIGQUIT}
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, syscall.SIGQUIT:
log.Printf("Received %v signal; cleaning up and terminating.", sig)
if builderProc != nil {
if err := builderProc.Kill(); err != nil {
log.Fatalf("Failed to kill our builder bot with pid %v: %v.", builderProc.Pid, err)
}
}
os.Exit(0)
default:
panic("should not get other signals here")
}
}
}
func pollGoChange() error {
doBuildGo = false
if err := os.Chdir(goTipDir); err != nil {
log.Fatalf("Could not cd to %v: %v", goTipDir, err)
}
tasks := []*task{
hgPullCmd,
hgLogCmd,
}
hash := ""
for _, t := range tasks {
out, err := t.run()
if err != nil {
if t.String() == hgPullCmd.String() {
log.Printf("Could not pull from Go repo with %v: %v", t.String(), err)
continue
}
return fmt.Errorf("Could not prepare the Go tree with %v: %v", t.String(), err)
}
hash = strings.TrimRight(out, "\n")
}
dbg.Println("previous head in go tree: " + goTipHash)
dbg.Println("current head in go tree: " + hash)
if hash != goTipHash {
if !plausibleHashRx.MatchString(hash) {
log.Printf("Go rev %q does not look like an hg hash.", hash)
} else {
goTipHash = hash
doBuildGo = true
dbg.Println("Changes in go tree detected; a builder will be started.")
}
}
// Should never happen, but be paranoid.
if !plausibleHashRx.MatchString(goTipHash) {
return fmt.Errorf("goTipHash %q does not look like an hg hash.", goTipHash)
}
return nil
}
var plausibleHashRx = regexp.MustCompile(`^[a-f0-9]{40}$`)
func altCamliPolling() (string, error) {
resp, err := http.Get(*altCamliRevURL)
if err != nil {
return "", fmt.Errorf("Could not get camliHash from %v: %v", *altCamliRevURL, err)
}
defer resp.Body.Close()
body, err := ioutil.ReadAll(resp.Body)
if err != nil {
return "", fmt.Errorf("Could not read camliHash from %v's response: %v", *altCamliRevURL, err)
}
hash := strings.TrimSpace(string(body))
if !plausibleHashRx.MatchString(hash) {
return "", fmt.Errorf("%v's response does not look like a git hash.", *altCamliRevURL)
}
return hash, nil
}
func pollCamliChange() error {
doBuildCamli = false
altDone := false
var err error
rev := ""
if *altCamliRevURL != "" {
rev, err = altCamliPolling()
if err != nil {
log.Print(err)
dbg.Println("Defaulting to the camli repo instead")
} else {
dbg.Printf("Got camli rev %v from %v\n", rev, *altCamliRevURL)
altDone = true
}
}
if !altDone {
if err := os.Chdir(camliRoot); err != nil {
log.Fatalf("Could not cd to %v: %v", camliRoot, err)
}
tasks := []*task{
gitPullCmd,
gitRevCmd,
}
for _, t := range tasks {
out, err := t.run()
if err != nil {
if t.String() == gitPullCmd.String() {
log.Printf("Could not pull from Camli repo with %v: %v", t.String(), err)
continue
}
return fmt.Errorf("Could not prepare the Camli tree with %v: %v\n", t.String(), err)
}
rev = strings.TrimRight(out, "\n")
}
}
dbg.Println("previous head in camli tree: " + camliHeadHash)
dbg.Println("current head in camli tree: " + rev)
if rev != camliHeadHash {
if !plausibleHashRx.MatchString(rev) {
return fmt.Errorf("Camlistore rev %q does not look like a git hash.", rev)
} else {
camliHeadHash = rev
doBuildCamli = true
dbg.Println("Changes in camli tree detected; a builder will be started.")
}
}
if !plausibleHashRx.MatchString(camliHeadHash) {
return fmt.Errorf("camliHeadHash %q does not look like a git hash.", camliHeadHash)
}
return nil
}
const builderBotBin = "builderBot"
func buildBuilder() error {
source := *builderSrc
if source == "" {
if *altCamliRevURL != "" {
// since we used altCamliRevURL (and not git pull), our camli tree
// and hence our buildbot source code, might not be up to date.
if err := os.Chdir(camliRoot); err != nil {
log.Fatalf("Could not cd to %v: %v", camliRoot, err)
}
out, err := gitRevCmd.run()
if err != nil {
return fmt.Errorf("Could not get camli tree revision with %v: %v\n", gitRevCmd.String(), err)
}
rev := strings.TrimRight(out, "\n")
if rev != camliHeadHash {
// camli tree needs to be updated
_, err := gitPullCmd.run()
if err != nil {
log.Printf("Could not update the Camli repo with %v: %v\n", gitPullCmd.String(), err)
}
}
}
source = filepath.Join(camliRoot, filepath.FromSlash("misc/buildbot/builder/builder.go"))
}
if err := os.Chdir(defaultDir); err != nil {
log.Fatalf("Could not cd to %v: %v", defaultDir, err)
}
tsk := newTask(
"go",
"build",
"-o",
builderBotBin,
source,
)
if _, err := tsk.run(); err != nil {
return err
}
return nil
}
func startBuilder(goHash, camliHash string) (*exec.Cmd, error) {
if err := os.Chdir(defaultDir); err != nil {
log.Fatalf("Could not cd to %v: %v", defaultDir, err)
}
dbg.Println("Starting builder bot")
builderHost := "localhost:" + *builderPort
ourHost, ourPort, err := net.SplitHostPort(*host)
if err != nil {
return nil, fmt.Errorf("Could not find out our host/port: %v", err)
}
if ourHost == "0.0.0.0" {
ourHost = "localhost"
}
masterHosts := ourHost + ":" + ourPort
buildbot/master: add Basic Auth support. Moved BasicAuth parsing and localhost detection code from pkg/auth -> pkg/httputil for use by buildbot master. Added user config file for remote access. The file's name is "masterbot-config.json" and is located in osutil.CamliConfigDir(), which on Unix will resolve to $XDG_CONFIG_HOME/camlistore/, if XDG_CONFIG_HOME set, or ~/.config/camlistore/. On Windows it will be under %APPDATA%\Camlistore\. The expected format is a json object with usernames as the keys and sha1 sums of the password as the values, i.e.: { "user1": "1234567890abcdef12341234567890abcdef1234", "user2": "1234abcdef12345678901234abcdef1234567890" } This file is polled at a 1 minute interval and reparsed if the file's modification time is more recent then the previous parse attempt. It is ok for the file to go missing, it will zero out the remote user list. A malformed file will result in the master exiting. New commandline flags, -tlsCertFile & -tlsKeyFile, added. Specifying both will enable TLS on the listener specified by -host. The go source contains generate_cert.go in crypto/tls that can be used to generate self-signed cert.pem and key.pem for testing. Added -skiptlscheck commandline option to builder. This allows the builder to report to https:// addresses with self-signed certs as we don't currently have a way to specify the cert chains to be used for TLS verification. This is a stop-gap solution. When launching a master that listens for secure connections, we currently need tell the builders to skip certificate validation. Add '-builderopts="-skiptlscheck"' to the master's commandline to skip cerfication verification. Change-Id: I0750b5c9fa8f4def67fc05a841087b50abded2f7
2013-10-31 05:00:17 +00:00
if useTLS {
masterHosts = "https://" + masterHosts
}
if *peers != "" {
masterHosts += "," + *peers
}
args := []string{
"-host",
builderHost,
"-masterhosts",
masterHosts,
}
if *builderOpts != "" {
moreOpts := strings.Split(*builderOpts, ",")
args = append(args, moreOpts...)
}
cmd := exec.Command("./"+builderBotBin, args...)
cmd.Stdout = multiWriter
cmd.Stderr = multiWriter
if err := cmd.Start(); err != nil {
return nil, err
}
inProgresslk.Lock()
defer inProgresslk.Unlock()
builderProc = cmd.Process
inProgress = &testSuite{
Start: time.Now(),
GoHash: goHash,
CamliHash: camliHash,
}
return cmd, nil
}
var (
okPrefix = "/ok/"
failPrefix = "/fail/"
progressPrefix = "/progress"
currentPrefix = "/current"
stderrPrefix = "/stderr"
reportPrefix = "/report"
statusTpl = template.Must(template.New("status").Funcs(tmplFuncs).Parse(statusHTML))
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,
}
var OSArchVersionTime = regexp.MustCompile(`(.*_.*)/(gotip|go1)/(.*)`)
// unlocked; history needs to be protected from the caller.
func getPastTestSuite(key string) (*testSuite, error) {
parts := OSArchVersionTime.FindStringSubmatch(key)
if parts == nil || len(parts) != 4 {
return nil, fmt.Errorf("bogus osArch/goversion/time url path: %v", key)
}
isGoTip := false
switch parts[2] {
case "gotip":
isGoTip = true
case "go1":
default:
return nil, fmt.Errorf("bogus go version in url path: %v", parts[2])
}
historyOSArch, ok := history[parts[1]]
if !ok {
return nil, fmt.Errorf("os %v not found in history", parts[1])
}
for _, v := range historyOSArch {
ts := v.Go1
if isGoTip {
ts = v.GoTip
}
if ts.Start.String() == parts[3] {
return ts, nil
}
}
return nil, fmt.Errorf("date %v not found in history for osArch %v", parts[3], parts[1])
}
// 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 reportHandler(w http.ResponseWriter, r *http.Request) {
if r.Method != "POST" {
log.Println("Invalid method for report handler")
http.Error(w, "Invalid method", http.StatusMethodNotAllowed)
return
}
body, err := ioutil.ReadAll(r.Body)
if err != nil {
log.Println("Invalid request for report handler")
http.Error(w, "Invalid method", http.StatusBadRequest)
return
}
defer r.Body.Close()
var report struct {
OSArch string
Ts *biTestSuite
}
err = json.Unmarshal(body, &report)
if err != nil {
log.Printf("Could not decode builder report: %v", err)
http.Error(w, "internal error", http.StatusInternalServerError)
return
}
addTestSuites(report.OSArch, report.Ts)
fmt.Fprintf(w, "Report ok")
}
func logHandler(w http.ResponseWriter, r *http.Request) {
fmt.Fprintln(w, `<!doctype html>
<html>
<body><pre>`)
switch r.URL.Path {
case stderrPrefix:
logStderr.Lock()
_, err := w.Write(logStderr.Bytes())
logStderr.Unlock()
if err != nil {
log.Println("Error serving logStderr:", err)
}
default:
fmt.Fprintln(w, "Unknown log file path passed to logHandler:", r.URL.Path)
log.Println("Unknown log file path passed to logHandler:", r.URL.Path)
}
}
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
}
var dat struct {
BiTs [2]*testSuite
}
if ts.IsTip {
dat.BiTs[1] = ts
} else {
dat.BiTs[0] = ts
}
err = testSuiteTpl.Execute(w, &dat)
if err != nil {
log.Printf("ok template: %v\n", err)
}
}
func progressHandler(w http.ResponseWriter, r *http.Request) {
if inProgress == nil {
http.Redirect(w, r, "/", http.StatusTemporaryRedirect)
return
}
// We only display the progress link and ask for progress for
// our local builder. The remote ones simply send their full report
// when they're done.
resp, err := http.Get("http://localhost:" + *builderPort + "/progress")
if err != nil {
log.Printf("Could not get a progress response from builder: %v", err)
http.Error(w, "internal error", http.StatusInternalServerError)
return
}
defer resp.Body.Close()
body, err := ioutil.ReadAll(resp.Body)
if err != nil {
log.Printf("Could not read progress response from builder: %v", err)
http.Error(w, "internal error", http.StatusInternalServerError)
return
}
var ts biTestSuite
err = json.Unmarshal(body, &ts)
if err != nil {
log.Printf("Could not decode builder progress report: %v", err)
http.Error(w, "internal error", http.StatusInternalServerError)
return
}
lastModified = time.Now()
var dat struct {
BiTs [2]*testSuite
}
dat.BiTs[0] = ts.Go1
if ts.GoTip != nil && !ts.GoTip.Start.IsZero() {
dat.BiTs[1] = ts.GoTip
}
err = testSuiteTpl.Execute(w, &dat)
if err != nil {
log.Printf("progress template: %v\n", err)
}
}
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 {
http.NotFound(w, r)
return
}
var failedTask *task
for _, v := range ts.Run {
if v.Err != "" {
failedTask = v
break
}
}
if failedTask == nil {
http.NotFound(w, r)
return
}
lastModTime := failedTask.Start.Add(failedTask.Duration)
if checkLastModified(w, r, lastModTime) {
return
}
failReport := struct {
TaskErr string
TsErr string
}{
TaskErr: failedTask.String() + "\n" + failedTask.Err,
TsErr: ts.Err,
}
err = taskTpl.Execute(w, &failReport)
if err != nil {
log.Printf("fail template: %v\n", err)
}
}
// unprotected read to history, caller needs to lock.
func invertedHistory(OSArch string) (inverted []*biTestSuite) {
historyOSArch, ok := history[OSArch]
if !ok {
return nil
}
inverted = make([]*biTestSuite, len(historyOSArch))
endpos := len(historyOSArch) - 1
for k, v := range historyOSArch {
inverted[endpos-k] = v
}
return inverted
}
type statusReport struct {
OSArch string
Hs []*biTestSuite
Progress *testSuite
}
func statusHandler(w http.ResponseWriter, r *http.Request) {
historylk.Lock()
inProgresslk.Lock()
defer inProgresslk.Unlock()
defer historylk.Unlock()
var localOne *statusReport
if inProgress != nil {
localOne = &statusReport{
Progress: &testSuite{
Start: inProgress.Start,
CamliHash: inProgress.CamliHash,
GoHash: inProgress.GoHash,
},
}
}
var reports []*statusReport
for OSArch, historyOSArch := range history {
if len(historyOSArch) == 0 {
continue
}
hs := invertedHistory(OSArch)
if historyOSArch[0].Local {
if localOne == nil {
localOne = &statusReport{}
}
localOne.OSArch = OSArch
localOne.Hs = hs
continue
}
reports = append(reports, &statusReport{
OSArch: OSArch,
Hs: hs,
})
}
if localOne != nil {
reports = append([]*statusReport{localOne}, reports...)
}
if checkLastModified(w, r, lastModified) {
return
}
err := statusTpl.Execute(w, reports)
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;
}
.pull-right {
float: right;
}
.fail {
color: #C00;
}
</style>
`
var statusHTML = `
<!DOCTYPE HTML>
<html>
<head>
<title>Camlistore tests Dashboard</title>` +
styleHTML + `
</head>
<body>
<h1>Camlibot status<span class="pull-right"><a href="` + stderrPrefix + `">stderr</a></span></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>
{{range $report := .}}
<tr>
<th>{{$report.OSArch}}</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 $report.Progress}}
<tr class="commit">
<td class="hash">{{$report.Progress.Start}}</td>
<td class="hash">
<a href="{{goRepoURL $report.Progress.GoHash}}">{{shortHash $report.Progress.GoHash}}</a>
</td>
<td class="hash">
<a href="{{camliRepoURL $report.Progress.CamliHash}}">{{shortHash $report.Progress.CamliHash}}</a>
</td>
<td class="result" colspan="2">
<a href="` + progressPrefix + `" class="ok">In progress</a>
</td>
</tr>
{{end}}
{{if $report.Hs}}
{{range $bits := $report.Hs}}
<tr class="commit">
<td class="hash">{{$bits.Go1.Start}}</td>
<td class="hash">
<a href="{{goRepoURL $bits.Go1.GoHash}}">{{shortHash $bits.Go1.GoHash}}</a>
</td>
<td class="hash">
<a href="{{camliRepoURL $bits.Go1.CamliHash}}">{{shortHash $bits.Go1.CamliHash}}</a>
</td>
<td class="result">
{{if $bits.Go1.Err}}
<a href="` + failPrefix + `{{$report.OSArch}}/go1/{{$bits.Go1.Start}}" class="fail">fail</a>
{{else}}
<a href="` + okPrefix + `{{$report.OSArch}}/go1/{{$bits.Go1.Start}}" class="ok">ok</a>
{{end}}
</td>
<td class="result">
{{if $bits.GoTip}}
{{if $bits.GoTip.Err}}
<a href="` + failPrefix + `{{$report.OSArch}}/gotip/{{$bits.GoTip.Start}}" class="fail">fail</a>
{{else}}
<a href="` + okPrefix + `{{$report.OSArch}}/gotip/{{$bits.GoTip.Start}}" class="ok">ok</a>
{{end}}
{{else}}
<a href="` + currentPrefix + `" class="ok">In progress</a>
{{end}}
</td>
</tr>
{{end}}
{{end}}
<tr>
<td colspan="5">&nbsp;</td>
</tr>
{{end}}
</table>
</body>
</html>
`
var testSuiteHTML = `
<!DOCTYPE HTML>
<html>
<head>
<title>Camlistore tests Dashboard</title>` +
styleHTML + `
</head>
<body>
{{range $ts := .BiTs}}
{{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, $tsk := $ts.Run}}
<tr>
<td>{{printf "%v" $tsk}}</td>
<td>{{$tsk.Duration}}</td>
</tr>
{{end}}
</table>
{{end}}
{{end}}
</body>
</html>
`
var taskHTML = `
<!DOCTYPE HTML>
<html>
<head>
<title>Camlistore tests Dashboard</title>
</head>
<body>
{{if .TaskErr}}
<h2>Task:</h2>
<pre>{{.TaskErr}}</pre>
{{end}}
{{if .TsErr}}
<h2>Error:</h2>
<pre>
{{.TsErr}}
</pre>
{{end}}
</body>
</html>
`