blob: d79a3a3a9a468c99ac77ec7b771da3c5218a9723 [file] [log] [blame]
// 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.
package main
import (
const (
// publicGoRepoURL contains the Gerrit repository URL.
publicGoRepoURL = ""
// privateGoRepoURL contains the internal repository URL.
privateGoRepoURL = "sso://team/golang/go-private"
// gitCheckout sets up a fresh git checkout in which to work,
// in $HOME/go-releasebot-work/<release>/gitwork
// (where <release> is a string like go1.8.5).
// The first time it is run for a particular release,
// gitCheckout also creates a clean checkout in
// $HOME/go-releasebot-work/<release>/gitmirror,
// to use as an object cache to speed future checkouts.
func (w *Work) gitCheckout() {
w.Dir = filepath.Join(os.Getenv("HOME"), "go-releasebot-work/"+strings.ToLower(w.Version))
w.log.Printf("working in %s\n", w.Dir)
if err := os.MkdirAll(w.Dir, 0777); err != nil {
origin := publicGoRepoURL
if w.Security {
origin = privateGoRepoURL
// Check out a local mirror to work-mirror, to speed future checkouts for this point release.
mirror := filepath.Join(w.Dir, "gitmirror")
r := w.runner(mirror)
if _, err := os.Stat(mirror); err != nil {
w.runner(w.Dir).run("git", "clone", origin, mirror)"git", "config", "", "0") // don't throw away refs we fetch
} else {"git", "fetch", "origin", "master")
}"git", "fetch", "origin", w.ReleaseBranch)
// Clone real Gerrit, but using local mirror for most objects.
gitDir := filepath.Join(w.Dir, "gitwork")
if err := os.RemoveAll(gitDir); err != nil {
w.runner(w.Dir).run("git", "clone", "--reference", mirror, "-b", w.ReleaseBranch, origin, gitDir)
r = w.runner(gitDir)"git", "codereview", "change", "relwork")"git", "config", "", "0") // don't throw away refs we fetch
// gitTagExists returns whether a git tag is already present in the repository.
func (w *Work) gitTagExists() bool {
_, err := w.runner(filepath.Join(w.Dir, "gitwork")).runErr("git", "rev-parse", w.Version)
return err == nil
// gitTagVersion tags the release candidate or release in Git.
func (w *Work) gitTagVersion() {
r := w.runner(filepath.Join(w.Dir, "gitwork"))
if w.gitTagExists() {
out := r.runOut("git", "rev-parse", w.Version)
w.VersionCommit = strings.TrimSpace(string(out))
w.log.Printf("Git tag already exists (%s), resuming release.", w.VersionCommit)
out := r.runOut("git", "rev-parse", "HEAD")
w.VersionCommit = strings.TrimSpace(string(out))
out = r.runOut("git", "show", w.VersionCommit)
fmt.Printf("About to tag the following commit as %s:\n\n%s\n\nOk? (y/n) ", w.Version, out)
if dryRun {
var response string
_, err := fmt.Scanln(&response)
if err != nil {
if response != "y" {
out, err = r.runErr("git", "tag", w.Version, w.VersionCommit)
if err != nil {
w.logError("git tag failed: %s\n%s", err, out)
}"git", "push", "origin", w.Version)
// gitHeadCommit returns the hash of the HEAD commit.
func (w *Work) gitHeadCommit() string {
r := w.runner(filepath.Join(w.Dir, "gitwork"))
out := r.runOut("git", "rev-parse", "HEAD")
return strings.TrimSpace(string(out))
// gitRemoteBranchCommit returns the hash of the HEAD commit on the branch located
// on a remote repository. It will return false when the branch does not exist.
// It panics if there is a problem communicating with the remote repository.
func (w *Work) gitRemoteBranchCommit(repositoryURL, branch string) (string, bool) {
out := w.runner(w.Dir).runOut("git", "ls-remote", "--heads", repositoryURL, "refs/heads/"+branch)
if len(out) == 0 {
return "", false
sha := strings.SplitN(string(out), "\t", 2)[0]
return sha, true
// gitCommitExistsInBranch reports whether the commit hash exists in the current branch.
func (w *Work) gitCommitExistsInBranch(commitSHA string) bool {
_, err := w.runner(filepath.Join(w.Dir, "gitwork")).runErr("git", "merge-base", "--is-ancestor", commitSHA, "HEAD")
return err == nil