mirror of
https://github.com/golang/go
synced 2024-11-05 18:56:10 -07:00
807424b52b
During the first run of `go-contrib-init` it tries to install the golang.org/x/review/git-codereview package using `go get`. If this command were to fail, we would check for the error and log that the command failed to succeed. However, when failure occurred we would only log the error and not interrupt the flow of the program. This would cause the program to continue with the assumption that git-codereview had been installed correctly. This change enhances the `go-contrib-init` command to exit with a bad status code, after logging the failure, if installing git-codereview fails. Fixes golang/go#21040 Change-Id: Ie01d78557d54285001db61faafbb409888b2893c Reviewed-on: https://go-review.googlesource.com/49151 Reviewed-by: Brad Fitzpatrick <bradfitz@golang.org>
290 lines
7.7 KiB
Go
290 lines
7.7 KiB
Go
// Copyright 2017 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.
|
|
|
|
// The go-contrib-init command helps new Go contributors get their development
|
|
// environment set up for the Go contribution process.
|
|
//
|
|
// It aims to be a complement or alternative to https://golang.org/doc/contribute.html.
|
|
package main
|
|
|
|
import (
|
|
"bytes"
|
|
"flag"
|
|
"fmt"
|
|
"go/build"
|
|
"io/ioutil"
|
|
"log"
|
|
"os"
|
|
"os/exec"
|
|
"path/filepath"
|
|
"regexp"
|
|
"runtime"
|
|
"strings"
|
|
)
|
|
|
|
var (
|
|
repo = flag.String("repo", detectrepo(), "Which go repo you want to contribute to. Use \"go\" for the core, or e.g. \"net\" for golang.org/x/net/*")
|
|
dry = flag.Bool("dry-run", false, "Fail with problems instead of trying to fix things.")
|
|
)
|
|
|
|
func main() {
|
|
log.SetFlags(0)
|
|
flag.Parse()
|
|
|
|
checkCLA()
|
|
checkGoroot()
|
|
checkWorkingDir()
|
|
checkGitOrigin()
|
|
checkGitCodeReview()
|
|
fmt.Print("All good. Happy hacking!\n" +
|
|
"Remember to squash your revised commits and preserve the magic Change-Id lines.\n" +
|
|
"Next steps: https://golang.org/doc/contribute.html#commit_changes\n")
|
|
}
|
|
|
|
func detectrepo() string {
|
|
wd, err := os.Getwd()
|
|
if err != nil {
|
|
return "go"
|
|
}
|
|
|
|
for _, path := range filepath.SplitList(build.Default.GOPATH) {
|
|
rightdir := filepath.Join(path, "src", "golang.org", "x") + string(os.PathSeparator)
|
|
if strings.HasPrefix(wd, rightdir) {
|
|
tail := wd[len(rightdir):]
|
|
end := strings.Index(tail, string(os.PathSeparator))
|
|
if end > 0 {
|
|
repo := tail[:end]
|
|
return repo
|
|
}
|
|
}
|
|
}
|
|
|
|
return "go"
|
|
}
|
|
|
|
var googleSourceRx = regexp.MustCompile(`(?m)^(go|go-review)?\.googlesource.com\b`)
|
|
|
|
func checkCLA() {
|
|
slurp, err := ioutil.ReadFile(cookiesFile())
|
|
if err != nil && !os.IsNotExist(err) {
|
|
log.Fatal(err)
|
|
}
|
|
if googleSourceRx.Match(slurp) {
|
|
// Probably good.
|
|
return
|
|
}
|
|
log.Fatal("Your .gitcookies file isn't configured.\n" +
|
|
"Next steps:\n" +
|
|
" * Submit a CLA (https://golang.org/doc/contribute.html#cla) if not done\n" +
|
|
" * Go to https://go.googlesource.com/ and click \"Generate Password\" at the top,\n" +
|
|
" then follow instructions.\n" +
|
|
" * Run go-contrib-init again.\n")
|
|
}
|
|
|
|
func expandUser(s string) string {
|
|
env := "HOME"
|
|
if runtime.GOOS == "windows" {
|
|
env = "USERPROFILE"
|
|
} else if runtime.GOOS == "plan9" {
|
|
env = "home"
|
|
}
|
|
home := os.Getenv(env)
|
|
if home == "" {
|
|
return s
|
|
}
|
|
|
|
if len(s) >= 2 && s[0] == '~' && os.IsPathSeparator(s[1]) {
|
|
if runtime.GOOS == "windows" {
|
|
s = filepath.ToSlash(filepath.Join(home, s[2:]))
|
|
} else {
|
|
s = filepath.Join(home, s[2:])
|
|
}
|
|
}
|
|
return os.Expand(s, func(env string) string {
|
|
if env == "HOME" {
|
|
return home
|
|
}
|
|
return os.Getenv(env)
|
|
})
|
|
}
|
|
|
|
func cookiesFile() string {
|
|
out, _ := exec.Command("git", "config", "http.cookiefile").Output()
|
|
if s := strings.TrimSpace(string(out)); s != "" {
|
|
if strings.HasPrefix(s, "~") {
|
|
s = expandUser(s)
|
|
}
|
|
return s
|
|
}
|
|
if runtime.GOOS == "windows" {
|
|
return filepath.Join(os.Getenv("USERPROFILE"), ".gitcookies")
|
|
}
|
|
return filepath.Join(os.Getenv("HOME"), ".gitcookies")
|
|
}
|
|
|
|
func checkGoroot() {
|
|
v := os.Getenv("GOROOT")
|
|
if v == "" {
|
|
return
|
|
}
|
|
if *repo == "go" {
|
|
if strings.HasPrefix(v, "/usr/") {
|
|
log.Fatalf("Your GOROOT environment variable is set to %q\n"+
|
|
"This is almost certainly not what you want. Either unset\n"+
|
|
"your GOROOT or set it to the path of your development version\n"+
|
|
"of Go.", v)
|
|
}
|
|
slurp, err := ioutil.ReadFile(filepath.Join(v, "VERSION"))
|
|
if err == nil {
|
|
slurp = bytes.TrimSpace(slurp)
|
|
log.Fatalf("Your GOROOT environment variable is set to %q\n"+
|
|
"But that path is to a binary release of Go, with VERSION file %q.\n"+
|
|
"You should hack on Go in a fresh checkout of Go. Fix or unset your GOROOT.\n",
|
|
v, slurp)
|
|
}
|
|
}
|
|
}
|
|
|
|
func checkWorkingDir() {
|
|
wd, err := os.Getwd()
|
|
if err != nil {
|
|
log.Fatal(err)
|
|
}
|
|
if *repo == "go" {
|
|
if inGoPath(wd) {
|
|
log.Fatalf(`You can't work on Go from within your GOPATH. Please checkout Go outside of your GOPATH
|
|
|
|
Current directory: %s
|
|
GOPATH: %s
|
|
`, wd, os.Getenv("GOPATH"))
|
|
}
|
|
return
|
|
}
|
|
|
|
gopath := firstGoPath()
|
|
if gopath == "" {
|
|
log.Fatal("Your GOPATH is not set, please set it")
|
|
}
|
|
|
|
rightdir := filepath.Join(gopath, "src", "golang.org", "x", *repo)
|
|
if !strings.HasPrefix(wd, rightdir) {
|
|
dirExists, err := exists(rightdir)
|
|
if err != nil {
|
|
log.Fatal(err)
|
|
}
|
|
if !dirExists {
|
|
log.Fatalf("The repo you want to work on is currently not on your system.\n"+
|
|
"Run %q to obtain this repo\n"+
|
|
"then go to the directory %q\n",
|
|
"go get -d golang.org/x/"+*repo, rightdir)
|
|
}
|
|
log.Fatalf("Your current directory is:%q\n"+
|
|
"Working on golang/x/%v requires you be in %q\n",
|
|
wd, *repo, rightdir)
|
|
}
|
|
}
|
|
|
|
func firstGoPath() string {
|
|
list := filepath.SplitList(build.Default.GOPATH)
|
|
if len(list) < 1 {
|
|
return ""
|
|
}
|
|
return list[0]
|
|
}
|
|
|
|
func exists(path string) (bool, error) {
|
|
_, err := os.Stat(path)
|
|
if os.IsNotExist(err) {
|
|
return false, nil
|
|
}
|
|
return true, err
|
|
}
|
|
|
|
func inGoPath(wd string) bool {
|
|
if os.Getenv("GOPATH") == "" {
|
|
return false
|
|
}
|
|
|
|
for _, path := range filepath.SplitList(os.Getenv("GOPATH")) {
|
|
if strings.HasPrefix(wd, filepath.Join(path, "src")) {
|
|
return true
|
|
}
|
|
}
|
|
|
|
return false
|
|
}
|
|
|
|
// mostly check that they didn't clone from github
|
|
func checkGitOrigin() {
|
|
if _, err := exec.LookPath("git"); err != nil {
|
|
log.Fatalf("You don't appear to have git installed. Do that.")
|
|
}
|
|
wantRemote := "https://go.googlesource.com/" + *repo
|
|
remotes, err := exec.Command("git", "remote", "-v").Output()
|
|
if err != nil {
|
|
msg := cmdErr(err)
|
|
if strings.Contains(msg, "Not a git repository") {
|
|
log.Fatalf("Your current directory is not in a git checkout of %s", wantRemote)
|
|
}
|
|
log.Fatalf("Error running git remote -v: %v", msg)
|
|
}
|
|
matches := 0
|
|
for _, line := range strings.Split(string(remotes), "\n") {
|
|
line = strings.TrimSpace(line)
|
|
if !strings.HasPrefix(line, "origin") {
|
|
continue
|
|
}
|
|
if !strings.Contains(line, wantRemote) {
|
|
curRemote := strings.Fields(strings.TrimPrefix(line, "origin"))[0]
|
|
// TODO: if not in dryRun mode, just fix it?
|
|
log.Fatalf("Current directory's git was cloned from %q; origin should be %q", curRemote, wantRemote)
|
|
}
|
|
matches++
|
|
}
|
|
if matches == 0 {
|
|
log.Fatalf("git remote -v output didn't contain expected %q. Got:\n%s", wantRemote, remotes)
|
|
}
|
|
}
|
|
|
|
func cmdErr(err error) string {
|
|
if ee, ok := err.(*exec.ExitError); ok && len(ee.Stderr) > 0 {
|
|
return fmt.Sprintf("%s: %s", err, ee.Stderr)
|
|
}
|
|
return fmt.Sprint(err)
|
|
}
|
|
|
|
func checkGitCodeReview() {
|
|
if _, err := exec.LookPath("git-codereview"); err != nil {
|
|
if *dry {
|
|
log.Fatalf("You don't appear to have git-codereview tool. While this is technically optional,\n" +
|
|
"almost all Go contributors use it. Our documentation and this tool assume it is used.\n" +
|
|
"To install it, run:\n\n\t$ go get golang.org/x/review/git-codereview\n\n(Then run go-contrib-init again)")
|
|
}
|
|
err := exec.Command("go", "get", "golang.org/x/review/git-codereview").Run()
|
|
if err != nil {
|
|
log.Fatalf("Error running go get golang.org/x/review/git-codereview: %v", cmdErr(err))
|
|
}
|
|
log.Printf("Installed git-codereview (ran `go get golang.org/x/review/git-codereview`)")
|
|
}
|
|
missing := false
|
|
for _, cmd := range []string{"change", "gofmt", "mail", "pending", "submit", "sync"} {
|
|
v, _ := exec.Command("git", "config", "alias."+cmd).Output()
|
|
if strings.Contains(string(v), "codereview") {
|
|
continue
|
|
}
|
|
if *dry {
|
|
log.Printf("Missing alias. Run:\n\t$ git config alias.%s \"codereview %s\"", cmd, cmd)
|
|
missing = true
|
|
} else {
|
|
err := exec.Command("git", "config", "alias."+cmd, "codereview "+cmd).Run()
|
|
if err != nil {
|
|
log.Fatalf("Error setting alias.%s: %v", cmd, cmdErr(err))
|
|
}
|
|
}
|
|
}
|
|
if missing {
|
|
log.Fatalf("Missing aliases. (While optional, this tool assumes you use them.)")
|
|
}
|
|
}
|