| // Copyright 2014 The Go Authors. All rights reserved. |
| // Use of this source code is governed by the Apache 2.0 |
| // license that can be found in the LICENSE file. |
| |
| // Command tip is the tip.golang.org server, |
| // serving the latest HEAD straight from the Git oven. |
| package main |
| |
| import ( |
| "bufio" |
| "crypto/tls" |
| "encoding/json" |
| "errors" |
| "flag" |
| "fmt" |
| "io" |
| "io/ioutil" |
| "log" |
| "net/http" |
| "net/http/httputil" |
| "net/url" |
| "os" |
| "os/exec" |
| "path/filepath" |
| "sync" |
| "time" |
| |
| "golang.org/x/crypto/acme/autocert" |
| ) |
| |
| const ( |
| repoURL = "https://go.googlesource.com/" |
| metaURL = "https://go.googlesource.com/?b=master&format=JSON" |
| startTimeout = 10 * time.Minute |
| ) |
| |
| var startTime = time.Now() |
| |
| var ( |
| autoCertDomain = flag.String("autocert", "", "if non-empty, listen on port 443 and serve a LetsEncrypt cert for this hostname") |
| ) |
| |
| func main() { |
| flag.Parse() |
| |
| const k = "TIP_BUILDER" |
| var b Builder |
| switch os.Getenv(k) { |
| case "godoc": |
| b = godocBuilder{} |
| case "talks": |
| b = talksBuilder{} |
| default: |
| log.Fatalf("Unknown %v value: %q", k, os.Getenv(k)) |
| } |
| |
| p := &Proxy{builder: b} |
| go p.run() |
| mux := newServeMux(p) |
| |
| log.Printf("Starting up tip server for builder %q", os.Getenv(k)) |
| |
| errc := make(chan error) |
| |
| go func() { |
| errc <- http.ListenAndServe(":8080", mux) |
| }() |
| if *autoCertDomain != "" { |
| log.Printf("Listening on port 443 with LetsEncrypt support on domain %q", *autoCertDomain) |
| m := autocert.Manager{ |
| Prompt: autocert.AcceptTOS, |
| HostPolicy: autocert.HostWhitelist(*autoCertDomain), |
| } |
| s := &http.Server{ |
| Addr: ":https", |
| Handler: mux, |
| TLSConfig: &tls.Config{GetCertificate: m.GetCertificate}, |
| } |
| go func() { |
| errc <- s.ListenAndServeTLS("", "") |
| }() |
| } |
| if err := <-errc; err != nil { |
| p.stop() |
| log.Fatal(err) |
| } |
| } |
| |
| // Proxy implements the tip.golang.org server: a reverse-proxy |
| // that builds and runs godoc instances showing the latest docs. |
| type Proxy struct { |
| builder Builder |
| |
| mu sync.Mutex // protects the followin' |
| proxy http.Handler |
| cur string // signature of gorepo+toolsrepo |
| cmd *exec.Cmd // live godoc instance, or nil for none |
| side string |
| hostport string // host and port of the live instance |
| err error |
| } |
| |
| type Builder interface { |
| Signature(heads map[string]string) string |
| Init(dir, hostport string, heads map[string]string) (*exec.Cmd, error) |
| HealthCheck(hostport string) error |
| } |
| |
| func (p *Proxy) ServeHTTP(w http.ResponseWriter, r *http.Request) { |
| if r.URL.Path == "/_tipstatus" { |
| p.serveStatus(w, r) |
| return |
| } |
| p.mu.Lock() |
| proxy := p.proxy |
| err := p.err |
| p.mu.Unlock() |
| if proxy == nil { |
| s := "starting up" |
| if err != nil { |
| s = err.Error() |
| } |
| http.Error(w, s, http.StatusInternalServerError) |
| return |
| } |
| proxy.ServeHTTP(w, r) |
| } |
| |
| func (p *Proxy) serveStatus(w http.ResponseWriter, r *http.Request) { |
| p.mu.Lock() |
| defer p.mu.Unlock() |
| fmt.Fprintf(w, "side=%v\ncurrent=%v\nerror=%v\nuptime=%v\n", p.side, p.cur, p.err, int(time.Since(startTime).Seconds())) |
| } |
| |
| func (p *Proxy) serveHealthCheck(w http.ResponseWriter, r *http.Request) { |
| p.mu.Lock() |
| defer p.mu.Unlock() |
| |
| // NOTE: (App Engine only; not GKE) Status 502, 503, 504 are |
| // the only status codes that signify an unhealthy app. So |
| // long as this handler returns one of those codes, this |
| // instance will not be sent any requests. |
| if p.proxy == nil { |
| log.Printf("Health check: not ready") |
| http.Error(w, "Not ready", http.StatusServiceUnavailable) |
| return |
| } |
| |
| if err := p.builder.HealthCheck(p.hostport); err != nil { |
| log.Printf("Health check failed: %v", err) |
| http.Error(w, "Health check failed", http.StatusServiceUnavailable) |
| return |
| } |
| io.WriteString(w, "ok") |
| } |
| |
| // run runs in its own goroutine. |
| func (p *Proxy) run() { |
| p.side = "a" |
| for { |
| p.poll() |
| time.Sleep(30 * time.Second) |
| } |
| } |
| |
| func (p *Proxy) stop() { |
| p.mu.Lock() |
| defer p.mu.Unlock() |
| if p.cmd != nil { |
| p.cmd.Process.Kill() |
| } |
| } |
| |
| // poll runs from the run loop goroutine. |
| func (p *Proxy) poll() { |
| heads := gerritMetaMap() |
| if heads == nil { |
| return |
| } |
| |
| sig := p.builder.Signature(heads) |
| |
| p.mu.Lock() |
| changes := sig != p.cur |
| curSide := p.side |
| p.cur = sig |
| p.mu.Unlock() |
| |
| if !changes { |
| return |
| } |
| |
| newSide := "b" |
| if curSide == "b" { |
| newSide = "a" |
| } |
| |
| dir := filepath.Join(os.TempDir(), "tip", newSide) |
| if err := os.MkdirAll(dir, 0755); err != nil { |
| p.err = err |
| return |
| } |
| hostport := "localhost:8081" |
| if newSide == "b" { |
| hostport = "localhost:8082" |
| } |
| cmd, err := p.builder.Init(dir, hostport, heads) |
| if err != nil { |
| err = fmt.Errorf("builder.Init: %v", err) |
| } else { |
| go func() { |
| // TODO(adg,bradfitz): be smarter about dead processes |
| if err := cmd.Wait(); err != nil { |
| log.Printf("process in %v exited: %v", dir, err) |
| } |
| }() |
| err = waitReady(p.builder, hostport) |
| if err != nil { |
| cmd.Process.Kill() |
| err = fmt.Errorf("waitReady: %v", err) |
| } |
| } |
| |
| p.mu.Lock() |
| defer p.mu.Unlock() |
| if err != nil { |
| log.Println(err) |
| p.err = err |
| return |
| } |
| |
| u, err := url.Parse(fmt.Sprintf("http://%v/", hostport)) |
| if err != nil { |
| err = fmt.Errorf("parsing hostport: %v", err) |
| log.Println(err) |
| p.err = err |
| return |
| } |
| p.proxy = httputil.NewSingleHostReverseProxy(u) |
| p.side = newSide |
| p.hostport = hostport |
| if p.cmd != nil { |
| p.cmd.Process.Kill() |
| } |
| p.cmd = cmd |
| } |
| |
| func newServeMux(p *Proxy) http.Handler { |
| mux := http.NewServeMux() |
| mux.Handle("/", httpsOnlyHandler{p}) |
| mux.HandleFunc("/_ah/health", p.serveHealthCheck) |
| return mux |
| } |
| |
| func waitReady(b Builder, hostport string) error { |
| var err error |
| deadline := time.Now().Add(startTimeout) |
| for time.Now().Before(deadline) { |
| if err = b.HealthCheck(hostport); err == nil { |
| return nil |
| } |
| time.Sleep(time.Second) |
| } |
| return fmt.Errorf("timed out waiting for process at %v: %v", hostport, err) |
| } |
| |
| func runErr(cmd *exec.Cmd) error { |
| out, err := cmd.CombinedOutput() |
| if err != nil { |
| if len(out) == 0 { |
| return err |
| } |
| return fmt.Errorf("%s\n%v", out, err) |
| } |
| return nil |
| } |
| |
| func checkout(repo, hash, path string) error { |
| // Clone git repo if it doesn't exist. |
| if _, err := os.Stat(filepath.Join(path, ".git")); os.IsNotExist(err) { |
| if err := os.MkdirAll(filepath.Dir(path), 0755); err != nil { |
| return fmt.Errorf("mkdir: %v", err) |
| } |
| if err := runErr(exec.Command("git", "clone", repo, path)); err != nil { |
| return fmt.Errorf("clone: %v", err) |
| } |
| } else if err != nil { |
| return fmt.Errorf("stat .git: %v", err) |
| } |
| |
| // Pull down changes and update to hash. |
| cmd := exec.Command("git", "fetch") |
| cmd.Dir = path |
| if err := runErr(cmd); err != nil { |
| return fmt.Errorf("fetch: %v", err) |
| } |
| cmd = exec.Command("git", "reset", "--hard", hash) |
| cmd.Dir = path |
| if err := runErr(cmd); err != nil { |
| return fmt.Errorf("reset: %v", err) |
| } |
| cmd = exec.Command("git", "clean", "-d", "-f", "-x") |
| cmd.Dir = path |
| if err := runErr(cmd); err != nil { |
| return fmt.Errorf("clean: %v", err) |
| } |
| return nil |
| } |
| |
| var timeoutClient = &http.Client{Timeout: 10 * time.Second} |
| |
| // gerritMetaMap returns the map from repo name (e.g. "go") to its |
| // latest master hash. |
| // The returned map is nil on any transient error. |
| func gerritMetaMap() map[string]string { |
| res, err := timeoutClient.Get(metaURL) |
| if err != nil { |
| log.Printf("Error getting Gerrit meta map: %v", err) |
| return nil |
| } |
| defer res.Body.Close() |
| defer io.Copy(ioutil.Discard, res.Body) // ensure EOF for keep-alive |
| if res.StatusCode != 200 { |
| return nil |
| } |
| var meta map[string]struct { |
| Branches map[string]string |
| } |
| br := bufio.NewReader(res.Body) |
| // For security reasons or something, this URL starts with ")]}'\n" before |
| // the JSON object. So ignore that. |
| // Shawn Pearce says it's guaranteed to always be just one line, ending in '\n'. |
| for { |
| b, err := br.ReadByte() |
| if err != nil { |
| return nil |
| } |
| if b == '\n' { |
| break |
| } |
| } |
| if err := json.NewDecoder(br).Decode(&meta); err != nil { |
| log.Printf("JSON decoding error from %v: %s", metaURL, err) |
| return nil |
| } |
| m := map[string]string{} |
| for repo, v := range meta { |
| if master, ok := v.Branches["master"]; ok { |
| m[repo] = master |
| } |
| } |
| return m |
| } |
| |
| func getOK(url string) (body []byte, err error) { |
| res, err := timeoutClient.Get(url) |
| if err != nil { |
| return nil, err |
| } |
| body, err = ioutil.ReadAll(res.Body) |
| res.Body.Close() |
| if err != nil { |
| return nil, err |
| } |
| if res.StatusCode != http.StatusOK { |
| return nil, errors.New(res.Status) |
| } |
| return body, nil |
| } |
| |
| // httpsOnlyHandler redirects requests to "http://example.com/foo?bar" to |
| // "https://example.com/foo?bar". It should be used when the server is listening |
| // for HTTP traffic behind a proxy that terminates TLS traffic, not when the Go |
| // server is terminating TLS directly. |
| type httpsOnlyHandler struct { |
| h http.Handler |
| } |
| |
| // isProxiedReq checks whether the server is running behind a proxy that may be |
| // terminating TLS. |
| func isProxiedReq(r *http.Request) bool { |
| if _, ok := r.Header["X-Appengine-Https"]; ok { |
| return true |
| } |
| if _, ok := r.Header["X-Forwarded-Proto"]; ok { |
| return true |
| } |
| return false |
| } |
| |
| func (h httpsOnlyHandler) ServeHTTP(w http.ResponseWriter, r *http.Request) { |
| if r.Header.Get("X-Appengine-Https") == "off" || r.Header.Get("X-Forwarded-Proto") == "http" || |
| (!isProxiedReq(r) && r.TLS == nil) { |
| r.URL.Scheme = "https" |
| r.URL.Host = r.Host |
| http.Redirect(w, r, r.URL.String(), http.StatusFound) |
| return |
| } |
| if r.Header.Get("X-Appengine-Https") == "on" || r.Header.Get("X-Forwarded-Proto") == "https" || |
| (!isProxiedReq(r) && r.TLS != nil) { |
| // Only set this header when we're actually in production. |
| w.Header().Set("Strict-Transport-Security", "max-age=31536000; preload") |
| } |
| h.h.ServeHTTP(w, r) |
| } |