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:
parent
6c387a1d07
commit
25693e10e1
|
@ -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
|
|
@ -0,0 +1,3 @@
|
||||||
|
To deploy tip.golang.org:
|
||||||
|
|
||||||
|
$ gcloud --project golang-org preview app deploy --set-default godoc.yaml
|
|
@ -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)
|
||||||
|
}
|
|
@ -10,3 +10,6 @@ automatic_scaling:
|
||||||
handlers:
|
handlers:
|
||||||
- url: /.*
|
- url: /.*
|
||||||
script: _go_app
|
script: _go_app
|
||||||
|
|
||||||
|
env_variables:
|
||||||
|
TIP_BUILDER: 'godoc'
|
|
@ -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 {
|
|
@ -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
|
|
Loading…
Reference in New Issue