cmd/tip: refactor tipgodoc into general-purpose tip server

This will allow us to serve blog.golang.org and talks.golang.org from
the latest sources without re-deploying.

Change-Id: I2399a8a7eb60a0c6648916052f5f129cb826d546
Reviewed-on: https://go-review.googlesource.com/14662
Reviewed-by: Andrew Gerrand <adg@golang.org>
This commit is contained in:
Andrew Gerrand 2015-09-17 11:19:51 +10:00
parent 6c387a1d07
commit 25693e10e1
6 changed files with 128 additions and 85 deletions

View File

@ -1,13 +1,13 @@
FROM golang:1.4.2 FROM golang:1.5
RUN apt-get update && apt-get install --no-install-recommends -y -q build-essential git RUN apt-get update && apt-get install --no-install-recommends -y -q build-essential git
# golang puts its go install here (weird but true) # golang puts its go install here (weird but true)
ENV GOROOT_BOOTSTRAP /usr/src/go ENV GOROOT_BOOTSTRAP /usr/local/go
# golang sets GOPATH=/go # golang sets GOPATH=/go
ADD . /go/src/tipgodoc ADD . /go/src/tip
RUN go install tipgodoc RUN go install tip
ENTRYPOINT ["/go/bin/tipgodoc"] ENTRYPOINT ["/go/bin/tip"]
# Kubernetes expects us to listen on port 8080 # Kubernetes expects us to listen on port 8080
EXPOSE 8080 EXPOSE 8080

3
cmd/tip/README Normal file
View File

@ -0,0 +1,3 @@
To deploy tip.golang.org:
$ gcloud --project golang-org preview app deploy --set-default godoc.yaml

88
cmd/tip/godoc.go Normal file
View File

@ -0,0 +1,88 @@
// Copyright 2015 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.
package main
import (
"bytes"
"fmt"
"io/ioutil"
"log"
"net/http"
"os"
"os/exec"
"path/filepath"
"time"
)
type godocBuilder struct {
}
func (b godocBuilder) Signature(heads map[string]string) string {
return heads["go"] + "-" + heads["tools"]
}
var indexingMsg = []byte("Indexing in progress: result may be inaccurate")
func (b godocBuilder) Init(dir, hostport string, heads map[string]string) (*exec.Cmd, error) {
goDir := filepath.Join(dir, "go")
toolsDir := filepath.Join(dir, "gopath/src/golang.org/x/tools")
if err := checkout(repoURL+"go", heads["go"], goDir); err != nil {
return nil, err
}
if err := checkout(repoURL+"tools", heads["tools"], toolsDir); err != nil {
return nil, err
}
make := exec.Command(filepath.Join(goDir, "src/make.bash"))
make.Dir = filepath.Join(goDir, "src")
if err := runErr(make); err != nil {
return nil, err
}
goBin := filepath.Join(goDir, "bin/go")
install := exec.Command(goBin, "install", "golang.org/x/tools/cmd/godoc")
install.Env = []string{
"GOROOT=" + goDir,
"GOPATH=" + filepath.Join(dir, "gopath"),
"GOROOT_BOOTSTRAP=" + os.Getenv("GOROOT_BOOTSTRAP"),
}
if err := runErr(install); err != nil {
return nil, err
}
godocBin := filepath.Join(goDir, "bin/godoc")
godoc := exec.Command(godocBin, "-http="+hostport, "-index", "-index_interval=-1s")
godoc.Env = []string{"GOROOT=" + goDir}
// TODO(adg): log this somewhere useful
godoc.Stdout = os.Stdout
godoc.Stderr = os.Stderr
if err := godoc.Start(); err != nil {
return nil, err
}
go func() {
// TODO(bradfitz): tell the proxy that this side is dead
if err := godoc.Wait(); err != nil {
log.Printf("process in %v exited: %v", dir, err)
}
}()
var err error
deadline := time.Now().Add(startTimeout)
for time.Now().Before(deadline) {
time.Sleep(time.Second)
var res *http.Response
res, err = http.Get(fmt.Sprintf("http://%v/search?q=FALLTHROUGH", hostport))
if err != nil {
continue
}
rbody, err := ioutil.ReadAll(res.Body)
res.Body.Close()
if err == nil && res.StatusCode == http.StatusOK &&
!bytes.Contains(rbody, indexingMsg) {
return godoc, nil
}
}
godoc.Process.Kill()
return nil, fmt.Errorf("timed out waiting for process in %v at %v (%v)", dir, hostport, err)
}

View File

@ -10,3 +10,6 @@ automatic_scaling:
handlers: handlers:
- url: /.* - url: /.*
script: _go_app script: _go_app
env_variables:
TIP_BUILDER: 'godoc'

View File

@ -8,7 +8,6 @@ package main
import ( import (
"bufio" "bufio"
"bytes"
"encoding/json" "encoding/json"
"fmt" "fmt"
"io" "io"
@ -30,10 +29,17 @@ const (
startTimeout = 5 * time.Minute startTimeout = 5 * time.Minute
) )
var indexingMsg = []byte("Indexing in progress: result may be inaccurate")
func main() { func main() {
p := new(Proxy) const k = "TIP_BUILDER"
var b Builder
switch os.Getenv(k) {
case "godoc":
b = godocBuilder{}
default:
log.Fatalf("Unknown %v value: %q", k, os.Getenv(k))
}
p := &Proxy{builder: b}
go p.run() go p.run()
http.Handle("/", p) http.Handle("/", p)
@ -46,6 +52,8 @@ func main() {
// Proxy implements the tip.golang.org server: a reverse-proxy // Proxy implements the tip.golang.org server: a reverse-proxy
// that builds and runs godoc instances showing the latest docs. // that builds and runs godoc instances showing the latest docs.
type Proxy struct { type Proxy struct {
builder Builder
mu sync.Mutex // protects the followin' mu sync.Mutex // protects the followin'
proxy http.Handler proxy http.Handler
cur string // signature of gorepo+toolsrepo cur string // signature of gorepo+toolsrepo
@ -54,6 +62,11 @@ type Proxy struct {
err error err error
} }
type Builder interface {
Signature(heads map[string]string) string
Init(dir, hostport string, heads map[string]string) (*exec.Cmd, error)
}
func (p *Proxy) ServeHTTP(w http.ResponseWriter, r *http.Request) { func (p *Proxy) ServeHTTP(w http.ResponseWriter, r *http.Request) {
if r.URL.Path == "/_tipstatus" { if r.URL.Path == "/_tipstatus" {
p.serveStatus(w, r) p.serveStatus(w, r)
@ -64,7 +77,7 @@ func (p *Proxy) ServeHTTP(w http.ResponseWriter, r *http.Request) {
err := p.err err := p.err
p.mu.Unlock() p.mu.Unlock()
if proxy == nil { if proxy == nil {
s := "tip.golang.org is starting up" s := "starting up"
if err != nil { if err != nil {
s = err.Error() s = err.Error()
} }
@ -108,7 +121,7 @@ func (p *Proxy) poll() {
return return
} }
sig := heads["go"] + "-" + heads["tools"] sig := p.builder.Signature(heads)
p.mu.Lock() p.mu.Lock()
changes := sig != p.cur changes := sig != p.cur
@ -125,7 +138,16 @@ func (p *Proxy) poll() {
newSide = "a" newSide = "a"
} }
cmd, hostport, err := initSide(newSide, heads["go"], heads["tools"]) 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)
p.mu.Lock() p.mu.Lock()
defer p.mu.Unlock() defer p.mu.Unlock()
@ -149,76 +171,6 @@ func (p *Proxy) poll() {
p.cmd = cmd p.cmd = cmd
} }
func initSide(side, goHash, toolsHash string) (godoc *exec.Cmd, hostport string, err error) {
dir := filepath.Join(os.TempDir(), "tipgodoc", side)
if err := os.MkdirAll(dir, 0755); err != nil {
return nil, "", err
}
goDir := filepath.Join(dir, "go")
toolsDir := filepath.Join(dir, "gopath/src/golang.org/x/tools")
if err := checkout(repoURL+"go", goHash, goDir); err != nil {
return nil, "", err
}
if err := checkout(repoURL+"tools", toolsHash, toolsDir); err != nil {
return nil, "", err
}
make := exec.Command(filepath.Join(goDir, "src/make.bash"))
make.Dir = filepath.Join(goDir, "src")
if err := runErr(make); err != nil {
return nil, "", err
}
goBin := filepath.Join(goDir, "bin/go")
install := exec.Command(goBin, "install", "golang.org/x/tools/cmd/godoc")
install.Env = []string{
"GOROOT=" + goDir,
"GOPATH=" + filepath.Join(dir, "gopath"),
"GOROOT_BOOTSTRAP=" + os.Getenv("GOROOT_BOOTSTRAP"),
}
if err := runErr(install); err != nil {
return nil, "", err
}
godocBin := filepath.Join(goDir, "bin/godoc")
hostport = "localhost:8081"
if side == "b" {
hostport = "localhost:8082"
}
godoc = exec.Command(godocBin, "-http="+hostport, "-index", "-index_interval=-1s")
godoc.Env = []string{"GOROOT=" + goDir}
// TODO(adg): log this somewhere useful
godoc.Stdout = os.Stdout
godoc.Stderr = os.Stderr
if err := godoc.Start(); err != nil {
return nil, "", err
}
go func() {
// TODO(bradfitz): tell the proxy that this side is dead
if err := godoc.Wait(); err != nil {
log.Printf("side %v exited: %v", side, err)
}
}()
deadline := time.Now().Add(startTimeout)
for time.Now().Before(deadline) {
time.Sleep(time.Second)
var res *http.Response
res, err = http.Get(fmt.Sprintf("http://%v/search?q=FALLTHROUGH", hostport))
if err != nil {
continue
}
rbody, err := ioutil.ReadAll(res.Body)
res.Body.Close()
if err == nil && res.StatusCode == http.StatusOK &&
!bytes.Contains(rbody, indexingMsg) {
return godoc, hostport, nil
}
}
godoc.Process.Kill()
return nil, "", fmt.Errorf("timed out waiting for side %v at %v (%v)", side, hostport, err)
}
func runErr(cmd *exec.Cmd) error { func runErr(cmd *exec.Cmd) error {
out, err := cmd.CombinedOutput() out, err := cmd.CombinedOutput()
if err != nil { if err != nil {

View File

@ -1,3 +0,0 @@
To deploy as an App Engine Manged VM, use gcloud:
$ gcloud --project golang-org preview app deploy --set-default app.yaml