Commit 94563de8 authored by Dmitri Shuralyov's avatar Dmitri Shuralyov

cmd/go: fix the default build output name for versioned binaries

This change is a re-apply of the reverted CL 140863 with changes to
address issue #30821. Specifically, path.Split continues to be used
to split the '/'-separated import path, rather than filepath.Split.

Document the algorithm for how the default executable name is determined
in DefaultExecName.

Rename a variable returned from os.Stat from bs to fi for consistency.

CL 140863 factored out the logic to determine the default executable
name from the Package.load method into a DefaultExecName function,
and started using it in more places to avoid having to re-implement
the logic everywhere it's needed. Most previous callers already computed
the default executable name based on the import path. The load.Package
method, before CL 140863, was the exception, in that it used the p.Dir
value in GOPATH mode instead. There was a NOTE(rsc) comment that it
should be equivalent to use import path, but it was too late in Go 1.11
cycle to risk implementing that change.

This is part 1, a more conservative change for backporting to Go 1.12.2,
and it keeps the original behavior of splitting on p.Dir in GOPATH mode.
Part 2 will address the NOTE(rsc) comment and modify behavior in
Package.load to always use DefaultExecName which splits the import path
rather than directory. It is intended to be included in Go 1.13.

Fixes #27283 (again)
Updates #26869
Fixes #30821

Change-Id: Ib1ebb95acba7c85c24e3a55c40cdf48405af34f3
Reviewed-on: https://go-review.googlesource.com/c/go/+/167503Reviewed-by: default avatarJay Conrod <jayconrod@google.com>
Reviewed-by: default avatarHyang-Ah Hana Kim <hyangah@gmail.com>
parent 1e83369c
...@@ -1186,6 +1186,36 @@ var cgoSyscallExclude = map[string]bool{ ...@@ -1186,6 +1186,36 @@ var cgoSyscallExclude = map[string]bool{
var foldPath = make(map[string]string) var foldPath = make(map[string]string)
// DefaultExecName returns the default executable name
// for a package with the import path importPath.
//
// The default executable name is the last element of the import path.
// In module-aware mode, an additional rule is used. If the last element
// is a vN path element specifying the major version, then the second last
// element of the import path is used instead.
func DefaultExecName(importPath string) string {
_, elem := pathpkg.Split(importPath)
if cfg.ModulesEnabled {
// If this is example.com/mycmd/v2, it's more useful to install it as mycmd than as v2.
// See golang.org/issue/24667.
isVersion := func(v string) bool {
if len(v) < 2 || v[0] != 'v' || v[1] < '1' || '9' < v[1] {
return false
}
for i := 2; i < len(v); i++ {
if c := v[i]; c < '0' || '9' < c {
return false
}
}
return true
}
if isVersion(elem) {
_, elem = pathpkg.Split(pathpkg.Dir(importPath))
}
}
return elem
}
// load populates p using information from bp, err, which should // load populates p using information from bp, err, which should
// be the result of calling build.Context.Import. // be the result of calling build.Context.Import.
func (p *Package) load(stk *ImportStack, bp *build.Package, err error) { func (p *Package) load(stk *ImportStack, bp *build.Package, err error) {
...@@ -1228,7 +1258,7 @@ func (p *Package) load(stk *ImportStack, bp *build.Package, err error) { ...@@ -1228,7 +1258,7 @@ func (p *Package) load(stk *ImportStack, bp *build.Package, err error) {
} }
_, elem := filepath.Split(p.Dir) _, elem := filepath.Split(p.Dir)
if cfg.ModulesEnabled { if cfg.ModulesEnabled {
// NOTE(rsc): Using p.ImportPath instead of p.Dir // NOTE(rsc,dmitshur): Using p.ImportPath instead of p.Dir
// makes sure we install a package in the root of a // makes sure we install a package in the root of a
// cached module directory as that package name // cached module directory as that package name
// not name@v1.2.3. // not name@v1.2.3.
...@@ -1237,26 +1267,9 @@ func (p *Package) load(stk *ImportStack, bp *build.Package, err error) { ...@@ -1237,26 +1267,9 @@ func (p *Package) load(stk *ImportStack, bp *build.Package, err error) {
// even for non-module-enabled code, // even for non-module-enabled code,
// but I'm not brave enough to change the // but I'm not brave enough to change the
// non-module behavior this late in the // non-module behavior this late in the
// release cycle. Maybe for Go 1.12. // release cycle. Can be done for Go 1.13.
// See golang.org/issue/26869. // See golang.org/issue/26869.
_, elem = pathpkg.Split(p.ImportPath) elem = DefaultExecName(p.ImportPath)
// If this is example.com/mycmd/v2, it's more useful to install it as mycmd than as v2.
// See golang.org/issue/24667.
isVersion := func(v string) bool {
if len(v) < 2 || v[0] != 'v' || v[1] < '1' || '9' < v[1] {
return false
}
for i := 2; i < len(v); i++ {
if c := v[i]; c < '0' || '9' < c {
return false
}
}
return true
}
if isVersion(elem) {
_, elem = pathpkg.Split(pathpkg.Dir(p.ImportPath))
}
} }
full := cfg.BuildContext.GOOS + "_" + cfg.BuildContext.GOARCH + "/" + elem full := cfg.BuildContext.GOOS + "_" + cfg.BuildContext.GOARCH + "/" + elem
if cfg.BuildContext.GOOS != base.ToolGOOS || cfg.BuildContext.GOARCH != base.ToolGOARCH { if cfg.BuildContext.GOOS != base.ToolGOOS || cfg.BuildContext.GOARCH != base.ToolGOARCH {
......
...@@ -811,7 +811,7 @@ func builderTest(b *work.Builder, p *load.Package) (buildAction, runAction, prin ...@@ -811,7 +811,7 @@ func builderTest(b *work.Builder, p *load.Package) (buildAction, runAction, prin
if p.ImportPath == "command-line-arguments" { if p.ImportPath == "command-line-arguments" {
elem = p.Name elem = p.Name
} else { } else {
_, elem = path.Split(p.ImportPath) elem = load.DefaultExecName(p.ImportPath)
} }
testBinary := elem + ".test" testBinary := elem + ".test"
......
...@@ -10,7 +10,6 @@ import ( ...@@ -10,7 +10,6 @@ import (
"go/build" "go/build"
"os" "os"
"os/exec" "os/exec"
"path"
"path/filepath" "path/filepath"
"runtime" "runtime"
"strings" "strings"
...@@ -285,7 +284,7 @@ func runBuild(cmd *base.Command, args []string) { ...@@ -285,7 +284,7 @@ func runBuild(cmd *base.Command, args []string) {
pkgs := load.PackagesForBuild(args) pkgs := load.PackagesForBuild(args)
if len(pkgs) == 1 && pkgs[0].Name == "main" && cfg.BuildO == "" { if len(pkgs) == 1 && pkgs[0].Name == "main" && cfg.BuildO == "" {
_, cfg.BuildO = path.Split(pkgs[0].ImportPath) cfg.BuildO = load.DefaultExecName(pkgs[0].ImportPath)
cfg.BuildO += cfg.ExeSuffix cfg.BuildO += cfg.ExeSuffix
} }
...@@ -320,14 +319,13 @@ func runBuild(cmd *base.Command, args []string) { ...@@ -320,14 +319,13 @@ func runBuild(cmd *base.Command, args []string) {
// If the -o name exists and is a directory, then // If the -o name exists and is a directory, then
// write all main packages to that directory. // write all main packages to that directory.
// Otherwise require only a single package be built. // Otherwise require only a single package be built.
if bs, err := os.Stat(cfg.BuildO); err == nil && bs.IsDir() { if fi, err := os.Stat(cfg.BuildO); err == nil && fi.IsDir() {
a := &Action{Mode: "go build"} a := &Action{Mode: "go build"}
for _, p := range pkgs { for _, p := range pkgs {
if p.Name != "main" { if p.Name != "main" {
continue continue
} }
_, elem := path.Split(p.ImportPath) p.Target = filepath.Join(cfg.BuildO, load.DefaultExecName(p.ImportPath))
p.Target = filepath.Join(cfg.BuildO, elem)
p.Target += cfg.ExeSuffix p.Target += cfg.ExeSuffix
p.Stale = true p.Stale = true
p.StaleReason = "build -o flag in use" p.StaleReason = "build -o flag in use"
...@@ -540,7 +538,7 @@ func InstallPackages(patterns []string, pkgs []*load.Package) { ...@@ -540,7 +538,7 @@ func InstallPackages(patterns []string, pkgs []*load.Package) {
if len(patterns) == 0 && len(pkgs) == 1 && pkgs[0].Name == "main" { if len(patterns) == 0 && len(pkgs) == 1 && pkgs[0].Name == "main" {
// Compute file 'go build' would have created. // Compute file 'go build' would have created.
// If it exists and is an executable file, remove it. // If it exists and is an executable file, remove it.
_, targ := filepath.Split(pkgs[0].ImportPath) targ := load.DefaultExecName(pkgs[0].ImportPath)
targ += cfg.ExeSuffix targ += cfg.ExeSuffix
if filepath.Join(pkgs[0].Dir, targ) != pkgs[0].Target { // maybe $GOBIN is the current directory if filepath.Join(pkgs[0].Dir, targ) != pkgs[0].Target { // maybe $GOBIN is the current directory
fi, err := os.Stat(targ) fi, err := os.Stat(targ)
......
...@@ -13,3 +13,9 @@ import "rsc.io/quote" ...@@ -13,3 +13,9 @@ import "rsc.io/quote"
func main() { func main() {
println(quote.Hello()) println(quote.Hello())
} }
-- fortune_test.go --
package main
import "testing"
func TestFortuneV2(t *testing.T) {}
env GO111MODULE=on
go get -m rsc.io/fortune/v2
# The default executable name shouldn't be v2$exe
go build rsc.io/fortune/v2
! exists v2$exe
exists fortune$exe
# The default test binary name shouldn't be v2.test$exe
go test -c rsc.io/fortune/v2
! exists v2.test$exe
exists fortune.test$exe
-- go.mod --
module scratch
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment