2013-08-07 14:49:37 -06:00
|
|
|
// Copyright 2013 The Go Authors. All rights reserved.
|
|
|
|
// Use of this source code is governed by a BSD-style
|
|
|
|
// license that can be found in the LICENSE file.
|
|
|
|
|
2013-08-09 16:44:00 -06:00
|
|
|
// +build ignore
|
|
|
|
|
2013-08-07 14:49:37 -06:00
|
|
|
// The run program is invoked via "go run" from src/run.bash or
|
|
|
|
// src/run.bat conditionally builds and runs the cmd/api tool.
|
|
|
|
//
|
|
|
|
// TODO(bradfitz): the "conditional" condition is always true.
|
|
|
|
// We should only do this if the user has the hg codereview extension
|
|
|
|
// enabled and verifies that the go.tools subrepo is checked out with
|
|
|
|
// a suitably recently version. In prep for the cmd/api rewrite.
|
|
|
|
package main
|
|
|
|
|
|
|
|
import (
|
|
|
|
"fmt"
|
|
|
|
"log"
|
2013-08-12 20:18:47 -06:00
|
|
|
"net/http"
|
2013-08-07 14:49:37 -06:00
|
|
|
"os"
|
|
|
|
"os/exec"
|
|
|
|
"path/filepath"
|
|
|
|
"strconv"
|
2013-08-24 08:51:42 -06:00
|
|
|
"strings"
|
2013-08-07 14:49:37 -06:00
|
|
|
)
|
|
|
|
|
2013-08-08 12:06:38 -06:00
|
|
|
// goToolsVersion is the hg revision of the go.tools subrepo we need
|
|
|
|
// to build cmd/api. This only needs to be updated whenever a go/types
|
|
|
|
// bug fix is needed by the cmd/api tool.
|
|
|
|
const goToolsVersion = "6698ca2900e2"
|
|
|
|
|
2013-08-07 14:49:37 -06:00
|
|
|
var goroot string
|
|
|
|
|
|
|
|
func main() {
|
|
|
|
log.SetFlags(0)
|
|
|
|
goroot = os.Getenv("GOROOT") // should be set by run.{bash,bat}
|
|
|
|
if goroot == "" {
|
|
|
|
log.Fatal("No $GOROOT set.")
|
|
|
|
}
|
|
|
|
isGoDeveloper := exec.Command("hg", "pq").Run() == nil
|
|
|
|
if !isGoDeveloper && !forceAPICheck() {
|
|
|
|
fmt.Println("Skipping cmd/api checks; hg codereview extension not available and GO_FORCE_API_CHECK not set")
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2013-08-08 12:06:38 -06:00
|
|
|
gopath := prepGoPath()
|
|
|
|
|
|
|
|
cmd := exec.Command("go", "install", "--tags=api_tool", "cmd/api")
|
2013-08-24 08:51:42 -06:00
|
|
|
cmd.Env = append([]string{"GOPATH=" + gopath}, filterOut(os.Environ(), "GOARCH")...)
|
2013-08-08 12:06:38 -06:00
|
|
|
out, err := cmd.CombinedOutput()
|
2013-08-07 14:49:37 -06:00
|
|
|
if err != nil {
|
|
|
|
log.Fatalf("Error installing cmd/api: %v\n%s", err, out)
|
|
|
|
}
|
2013-08-08 12:06:38 -06:00
|
|
|
|
2013-08-07 14:49:37 -06:00
|
|
|
out, err = exec.Command("go", "tool", "api",
|
|
|
|
"-c", file("go1", "go1.1"),
|
|
|
|
"-next", file("next"),
|
|
|
|
"-except", file("except")).CombinedOutput()
|
|
|
|
if err != nil {
|
|
|
|
log.Fatalf("Error running API checker: %v\n%s", err, out)
|
|
|
|
}
|
2013-08-08 14:36:22 -06:00
|
|
|
fmt.Print(string(out))
|
2013-08-07 14:49:37 -06:00
|
|
|
}
|
|
|
|
|
2013-08-24 08:51:42 -06:00
|
|
|
// filterOut returns a copy of the src environment without environment
|
|
|
|
// variables from remove.
|
|
|
|
// TODO: delete when issue 6201 is fixed.
|
|
|
|
func filterOut(src []string, remove ...string) (out []string) {
|
|
|
|
S:
|
|
|
|
for _, s := range src {
|
|
|
|
for _, r := range remove {
|
|
|
|
if strings.HasPrefix(s, r) && strings.HasPrefix(s, r+"=") {
|
|
|
|
continue S
|
|
|
|
}
|
|
|
|
}
|
|
|
|
out = append(out, s)
|
|
|
|
}
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2013-08-07 14:49:37 -06:00
|
|
|
// file expands s to $GOROOT/api/s.txt.
|
|
|
|
// If there are more than 1, they're comma-separated.
|
|
|
|
func file(s ...string) string {
|
|
|
|
if len(s) > 1 {
|
|
|
|
return file(s[0]) + "," + file(s[1:]...)
|
|
|
|
}
|
|
|
|
return filepath.Join(goroot, "api", s[0]+".txt")
|
|
|
|
}
|
|
|
|
|
|
|
|
// GO_FORCE_API_CHECK is set by builders.
|
|
|
|
func forceAPICheck() bool {
|
|
|
|
v, _ := strconv.ParseBool(os.Getenv("GO_FORCE_API_CHECK"))
|
|
|
|
return v
|
|
|
|
}
|
2013-08-08 12:06:38 -06:00
|
|
|
|
|
|
|
// prepGoPath returns a GOPATH for the "go" tool to compile the API tool with.
|
|
|
|
// It tries to re-use a go.tools checkout from a previous run if possible,
|
|
|
|
// else it hg clones it.
|
|
|
|
func prepGoPath() string {
|
|
|
|
const tempBase = "go.tools.TMP"
|
|
|
|
|
|
|
|
// The GOPATH we'll return
|
|
|
|
gopath := filepath.Join(os.TempDir(), "gopath-api", goToolsVersion)
|
|
|
|
|
|
|
|
// cloneDir is where we run "hg clone".
|
|
|
|
cloneDir := filepath.Join(gopath, "src", "code.google.com", "p")
|
|
|
|
|
|
|
|
// The dir we clone into. We only atomically rename it to finalDir on
|
|
|
|
// clone success.
|
|
|
|
tmpDir := filepath.Join(cloneDir, tempBase)
|
|
|
|
|
|
|
|
// finalDir is where the checkout will live once it's complete.
|
|
|
|
// If this exists already, we're done.
|
|
|
|
finalDir := filepath.Join(cloneDir, "go.tools")
|
|
|
|
|
|
|
|
if fi, err := os.Stat(finalDir); err == nil && fi.IsDir() {
|
|
|
|
return gopath
|
|
|
|
}
|
|
|
|
|
|
|
|
if err := os.MkdirAll(cloneDir, 0700); err != nil {
|
|
|
|
log.Fatal(err)
|
|
|
|
}
|
|
|
|
cmd := exec.Command("hg",
|
|
|
|
"clone", "--rev="+goToolsVersion,
|
|
|
|
"https://code.google.com/p/go.tools",
|
|
|
|
tempBase)
|
|
|
|
cmd.Dir = cloneDir
|
|
|
|
out, err := cmd.CombinedOutput()
|
|
|
|
if err != nil {
|
2013-08-12 20:18:47 -06:00
|
|
|
if _, err := http.Head("http://ip.appspot.com/"); err != nil {
|
|
|
|
log.Printf("# Skipping API check; network appears to be unavailable")
|
|
|
|
os.Exit(0)
|
|
|
|
}
|
2013-08-08 12:06:38 -06:00
|
|
|
log.Fatalf("Error running hg clone on go.tools: %v\n%s", err, out)
|
|
|
|
}
|
|
|
|
if err := os.Rename(tmpDir, finalDir); err != nil {
|
|
|
|
log.Fatal(err)
|
|
|
|
}
|
|
|
|
return gopath
|
|
|
|
}
|