whatcanGOwrong
This commit is contained in:
@@ -0,0 +1,106 @@
|
||||
// Copyright 2020 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 release checks that the a given version of gopls is ready for
|
||||
// release. It can also tag and publish the release.
|
||||
//
|
||||
// To run:
|
||||
//
|
||||
// $ cd $GOPATH/src/golang.org/x/tools/gopls
|
||||
// $ go run release/release.go -version=<version>
|
||||
package main
|
||||
|
||||
import (
|
||||
"flag"
|
||||
"fmt"
|
||||
"log"
|
||||
"os"
|
||||
"os/exec"
|
||||
"path/filepath"
|
||||
"strings"
|
||||
|
||||
"golang.org/x/mod/modfile"
|
||||
"golang.org/x/mod/semver"
|
||||
)
|
||||
|
||||
var versionFlag = flag.String("version", "", "version to tag")
|
||||
|
||||
func main() {
|
||||
flag.Parse()
|
||||
|
||||
if *versionFlag == "" {
|
||||
log.Fatalf("must provide -version flag")
|
||||
}
|
||||
if !semver.IsValid(*versionFlag) {
|
||||
log.Fatalf("invalid version %s", *versionFlag)
|
||||
}
|
||||
if semver.Major(*versionFlag) != "v0" {
|
||||
log.Fatalf("expected major version v0, got %s", semver.Major(*versionFlag))
|
||||
}
|
||||
if semver.Build(*versionFlag) != "" {
|
||||
log.Fatalf("unexpected build suffix: %s", *versionFlag)
|
||||
}
|
||||
// Validate that the user is running the program from the gopls module.
|
||||
wd, err := os.Getwd()
|
||||
if err != nil {
|
||||
log.Fatal(err)
|
||||
}
|
||||
if filepath.Base(wd) != "gopls" {
|
||||
log.Fatalf("must run from the gopls module")
|
||||
}
|
||||
// Confirm that the versions in the go.mod file are correct.
|
||||
if err := validateGoModFile(wd); err != nil {
|
||||
log.Fatal(err)
|
||||
}
|
||||
fmt.Println("Validated that the release is ready.")
|
||||
os.Exit(0)
|
||||
}
|
||||
|
||||
func validateGoModFile(goplsDir string) error {
|
||||
filename := filepath.Join(goplsDir, "go.mod")
|
||||
data, err := os.ReadFile(filename)
|
||||
if err != nil {
|
||||
return err
|
||||
}
|
||||
gomod, err := modfile.Parse(filename, data, nil)
|
||||
if err != nil {
|
||||
return err
|
||||
}
|
||||
// Confirm that there is no replace directive in the go.mod file.
|
||||
if len(gomod.Replace) > 0 {
|
||||
return fmt.Errorf("expected no replace directives, got %v", len(gomod.Replace))
|
||||
}
|
||||
// Confirm that the version of x/tools in the gopls/go.mod file points to
|
||||
// the second-to-last commit. (The last commit will be the one to update the
|
||||
// go.mod file.)
|
||||
cmd := exec.Command("git", "rev-parse", "@~")
|
||||
stdout, err := cmd.Output()
|
||||
if err != nil {
|
||||
return err
|
||||
}
|
||||
hash := string(stdout)
|
||||
// Find the golang.org/x/tools require line and compare the versions.
|
||||
var version string
|
||||
for _, req := range gomod.Require {
|
||||
if req.Mod.Path == "golang.org/x/tools" {
|
||||
version = req.Mod.Version
|
||||
break
|
||||
}
|
||||
}
|
||||
if version == "" {
|
||||
return fmt.Errorf("no require for golang.org/x/tools")
|
||||
}
|
||||
split := strings.Split(version, "-")
|
||||
if len(split) != 3 {
|
||||
return fmt.Errorf("unexpected pseudoversion format %s", version)
|
||||
}
|
||||
last := split[len(split)-1]
|
||||
if last == "" {
|
||||
return fmt.Errorf("unexpected pseudoversion format %s", version)
|
||||
}
|
||||
if !strings.HasPrefix(hash, last) {
|
||||
return fmt.Errorf("golang.org/x/tools pseudoversion should be at commit %s, instead got %s", hash, last)
|
||||
}
|
||||
return nil
|
||||
}
|
||||
Reference in New Issue
Block a user