debian-mirror-gitlab/workhorse-vendor/gocloud.dev/internal/releasehelper/releasehelper.go
2023-01-13 15:02:22 +05:30

262 lines
6.6 KiB
Go

// Copyright 2019 The Go Cloud Development Kit Authors
//
// Licensed under the Apache License, Version 2.0 (the "License");
// you may not use this file except in compliance with the License.
// You may obtain a copy of the License at
//
// https://www.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
// See the License for the specific language governing permissions and
// limitations under the License.
// Helper tool for creating new releases of the Go CDK. Run without arguments
// or with 'help' for details.
package main
import (
"bufio"
"encoding/json"
"fmt"
"log"
"os"
"os/exec"
"path/filepath"
"regexp"
"strings"
)
var helpText string = `
Helper tool for creating new releases of the Go CDK.
Automates the modifications required in the project's
go.mod files to create an test new releases.
The tool processes all modules listed in the 'allmodules'
file. For each module it handles all dependencies on
other gocloud.dev modules.
Run it from the root directory of the repository, as follows:
$ %s <command>
Where command is one of the following:
addreplace adds 'replace' directives to point to local versions
for testing.
dropreplace removes these directives.
setversion <version>
sets 'required' version of modules to a given version formatted
as vX.Y.Z
tag <version>
runs 'git tag <module>/<version>' for all CDK modules
help prints this usage message
`
func printHelp() {
_, binName := filepath.Split(os.Args[0])
fmt.Fprintf(os.Stderr, helpText, binName)
fmt.Fprintln(os.Stderr)
}
// cmdCheck invokes the command given in s, echoing the invocation to stdout.
// It checks that the command was successful and returns its standard output.
// If the command returned a non-0 status, log.Fatal is invoked.
func cmdCheck(s string) []byte {
fmt.Printf(" -> %s\n", s)
fields := strings.Fields(s)
if len(fields) < 1 {
log.Fatal(`Expected "command <arguments>"`)
}
b, err := exec.Command(fields[0], fields[1:]...).Output()
if exiterr, ok := err.(*exec.ExitError); ok {
log.Fatalf("%s; stderr: %s\n", err, string(exiterr.Stderr))
} else if err != nil {
log.Fatal("exec.Command", err)
}
return b
}
// GoMod holds "go mod" parameters, taken from "go mod help edit", in order to parse the JSON
// output of `go mod edit -json`.
type GoMod struct {
Module Module
Go string
Require []Require
Exclude []Module
Replace []Replace
}
// Module represents a Go module.
type Module struct {
Path string
Version string
}
// Require represents a required module.
type Require struct {
Path string
Version string
Indirect bool
}
// Replace represents a replace directive.
type Replace struct {
Old Module
New Module
}
// parseModuleInfo parses module information from a go.mod file at path.
func parseModuleInfo(path string) GoMod {
rawJson := cmdCheck("go mod edit -json " + path)
var modInfo GoMod
err := json.Unmarshal(rawJson, &modInfo)
if err != nil {
log.Fatal(err)
}
return modInfo
}
// runOnGomod processes a single go.mod file (located in directory 'path').
// Each require in the go.mod file is processed with reqHandler, a callback
// function. It's called with these arguments:
//
// gomodPath - path to the go.mod file where this 'require' was found
// mod - name of the module being 'require'd
// modPath - mod's location in the filesystem relative to
// the go.mod 'require'ing it
func runOnGomod(path string, reqHandler func(gomodPath, mod, modPath string)) {
gomodPath := filepath.Join(path, "go.mod")
fmt.Println("Processing", gomodPath)
modInfo := parseModuleInfo(gomodPath)
base := "gocloud.dev"
for _, r := range modInfo.Require {
// Find requirements on modules within the gocloud.dev tree.
if strings.HasPrefix(r.Path, base) {
// Find the relative path from 'path' and the module required here.
var reqPath string
if r.Path == base {
reqPath = "."
} else {
reqPath = strings.TrimPrefix(r.Path, base+"/")
}
rel, err := filepath.Rel(path, reqPath)
if err != nil {
log.Fatal(err)
}
// When path is '.', filepath.Rel will append a /. to the result and we
// may get paths like ../../.
if strings.HasSuffix(rel, "/.") {
rel, _ = filepath.Split(rel)
}
reqHandler(gomodPath, r.Path, rel)
}
}
}
func gomodAddReplace(path string) {
runOnGomod(path, func(gomodPath, mod, modPath string) {
cmdCheck(fmt.Sprintf("go mod edit -replace=%s=%s %s", mod, modPath, gomodPath))
})
}
func gomodDropReplace(path string) {
runOnGomod(path, func(gomodPath, mod, modPath string) {
cmdCheck(fmt.Sprintf("go mod edit -dropreplace=%s %s", mod, gomodPath))
})
}
func gomodSetVersion(path string, v string) {
runOnGomod(path, func(gomodPath, mod, modPath string) {
cmdCheck(fmt.Sprintf("go mod edit -require=%s@%s %s", mod, v, gomodPath))
})
}
func gomodTag(path string, v string) {
var tagName string
if path == "." {
tagName = v
} else {
tagName = filepath.Join(path, v)
}
cmdCheck(fmt.Sprintf("git tag %s", tagName))
}
func validSemanticVersion(v string) bool {
match, err := regexp.MatchString(`v\d+\.\d+\.\d+`, v)
if err != nil {
return false
}
return match
}
func main() {
if len(os.Args) < 2 {
printHelp()
os.Exit(0)
}
var gomodHandler func(path string)
switch os.Args[1] {
case "help":
printHelp()
os.Exit(0)
case "addreplace":
gomodHandler = gomodAddReplace
case "dropreplace":
gomodHandler = gomodDropReplace
case "setversion":
if len(os.Args) < 3 || !validSemanticVersion(os.Args[2]) {
printHelp()
os.Exit(1)
}
gomodHandler = func(path string) {
gomodSetVersion(path, os.Args[2])
}
case "tag":
if len(os.Args) < 3 || !validSemanticVersion(os.Args[2]) {
printHelp()
os.Exit(1)
}
gomodHandler = func(path string) {
gomodTag(path, os.Args[2])
}
default:
printHelp()
os.Exit(1)
}
f, err := os.Open("allmodules")
if err != nil {
log.Fatal(err)
}
input := bufio.NewScanner(f)
input.Split(bufio.ScanLines)
for input.Scan() {
if len(input.Text()) > 0 && !strings.HasPrefix(input.Text(), "#") {
fields := strings.Fields(input.Text())
if len(fields) != 2 {
log.Fatalf("want 2 fields, got '%s'\n", input.Text())
}
// "tag" only runs if the released field is "yes". Other commands run
// for every line.
if os.Args[1] != "tag" || fields[1] == "yes" {
gomodHandler(fields[0])
}
}
}
if input.Err() != nil {
log.Fatal(input.Err())
}
}