Commit 30a84b38 authored by Russ Cox's avatar Russ Cox

cmd/go: replace -getmode with -mod, $GOPROXY

The old -getmode flag had two settings:
-getmode=local meant don't download from the network.
-getmode=vendor meant only use the vendor directory.

The new -mod flag has two settings:
-mod=readonly means refuse to automatically update go.mod (mainly for CI testing).
-mod=vendor means only use the vendor directory.

The old GOPROXY variable had two settings:
a proxy URL or else the empty string (direct connect).

The new GOPROXY variable has three settings:
a proxy URL, the string "off" (no network use allowed),
or else the empty string or the explicit string "direct" (direct connection).
We anticipate allow a comma-separated sequence in a future release,
so commas are disallowed entirely right now.

Fixes #24666.
Fixes #26586.
Fixes #26370.
Fixes #26361.

Change-Id: If2601a16b09f04800f666938c071fc053b4c3f9c
Reviewed-on: https://go-review.googlesource.com/126696
Run-TryBot: Russ Cox <rsc@golang.org>
TryBot-Result: Gobot Gobot <gobot@golang.org>
Reviewed-by: default avatarBryan C. Mills <bcmills@google.com>
parent c1a4fc3b
...@@ -21,7 +21,7 @@ var ( ...@@ -21,7 +21,7 @@ var (
BuildA bool // -a flag BuildA bool // -a flag
BuildBuildmode string // -buildmode flag BuildBuildmode string // -buildmode flag
BuildContext = defaultContext() BuildContext = defaultContext()
BuildGetmode string // -getmode flag BuildMod string // -mod flag
BuildI bool // -i flag BuildI bool // -i flag
BuildLinkshared bool // -linkshared flag BuildLinkshared bool // -linkshared flag
BuildMSan bool // -msan flag BuildMSan bool // -msan flag
......
...@@ -25,11 +25,14 @@ var HelpGoproxy = &base.Command{ ...@@ -25,11 +25,14 @@ var HelpGoproxy = &base.Command{
Short: "module proxy protocol", Short: "module proxy protocol",
Long: ` Long: `
The go command by default downloads modules from version control systems The go command by default downloads modules from version control systems
directly, just as 'go get' always has. If the GOPROXY environment variable directly, just as 'go get' always has. The GOPROXY environment variable allows
is set to the URL of a module proxy, the go command will instead fetch further control over the download source. If GOPROXY is unset, is the empty string,
all modules from that proxy. No matter the source of the modules, downloaded or is the string "direct", downloads use the default direct connection to version
modules must match existing entries in go.sum (see 'go help modules' for control systems. Setting GOPROXY to "off" disallows downloading modules from
discussion of verification). any source. Otherwise, GOPROXY is expected to be the URL of a module proxy,
in which case the go command will fetch all modules from that proxy.
No matter the source of the modules, downloaded modules must match existing
entries in go.sum (see 'go help modules' for discussion of verification).
A Go module proxy is any web server that can respond to GET requests for A Go module proxy is any web server that can respond to GET requests for
URLs of a specified form. The requests have no query parameters, so even URLs of a specified form. The requests have no query parameters, so even
......
...@@ -203,10 +203,13 @@ func Lookup(path string) (Repo, error) { ...@@ -203,10 +203,13 @@ func Lookup(path string) (Repo, error) {
// lookup returns the module with the given module path. // lookup returns the module with the given module path.
func lookup(path string) (r Repo, err error) { func lookup(path string) (r Repo, err error) {
if cfg.BuildGetmode != "" { if cfg.BuildMod == "vendor" {
return nil, fmt.Errorf("module lookup disabled by -getmode=%s", cfg.BuildGetmode) return nil, fmt.Errorf("module lookup disabled by -mod=%s", cfg.BuildMod)
} }
if proxyURL != "" { if proxyURL == "off" {
return nil, fmt.Errorf("module lookup disabled by GOPROXY=%s", proxyURL)
}
if proxyURL != "" && proxyURL != "direct" {
return lookupProxy(path) return lookupProxy(path)
} }
...@@ -241,8 +244,8 @@ func lookupCodeRepo(rr *get.RepoRoot) (codehost.Repo, error) { ...@@ -241,8 +244,8 @@ func lookupCodeRepo(rr *get.RepoRoot) (codehost.Repo, error) {
// the original "go get" would have used, at the specific repository revision // the original "go get" would have used, at the specific repository revision
// (typically a commit hash, but possibly also a source control tag). // (typically a commit hash, but possibly also a source control tag).
func ImportRepoRev(path, rev string) (Repo, *RevInfo, error) { func ImportRepoRev(path, rev string) (Repo, *RevInfo, error) {
if cfg.BuildGetmode != "" { if cfg.BuildMod == "vendor" || cfg.BuildMod == "readonly" {
return nil, nil, fmt.Errorf("repo version lookup disabled by -getmode=%s", cfg.BuildGetmode) return nil, nil, fmt.Errorf("repo version lookup disabled by -mod=%s", cfg.BuildMod)
} }
// Note: Because we are converting a code reference from a legacy // Note: Because we are converting a code reference from a legacy
......
...@@ -189,6 +189,11 @@ type task struct { ...@@ -189,6 +189,11 @@ type task struct {
} }
func runGet(cmd *base.Command, args []string) { func runGet(cmd *base.Command, args []string) {
// -mod=readonly has no effect on "go get".
if cfg.BuildMod == "readonly" {
cfg.BuildMod = ""
}
switch getU { switch getU {
case "", "patch", "true": case "", "patch", "true":
// ok // ok
...@@ -205,8 +210,8 @@ func runGet(cmd *base.Command, args []string) { ...@@ -205,8 +210,8 @@ func runGet(cmd *base.Command, args []string) {
fmt.Fprintf(os.Stderr, "go get: -t flag is a no-op when using modules\n") fmt.Fprintf(os.Stderr, "go get: -t flag is a no-op when using modules\n")
} }
if cfg.BuildGetmode == "vendor" { if cfg.BuildMod == "vendor" {
base.Fatalf("go get: disabled by -getmode=vendor") base.Fatalf("go get: disabled by -mod=%s", cfg.BuildMod)
} }
modload.LoadBuildList() modload.LoadBuildList()
......
...@@ -108,7 +108,7 @@ func moduleInfo(m module.Version, fromBuildList bool) *modinfo.ModulePublic { ...@@ -108,7 +108,7 @@ func moduleInfo(m module.Version, fromBuildList bool) *modinfo.ModulePublic {
info.GoVersion = loaded.goVersion[m.Path] info.GoVersion = loaded.goVersion[m.Path]
} }
if cfg.BuildGetmode == "vendor" { if cfg.BuildMod == "vendor" {
info.Dir = filepath.Join(ModRoot, "vendor", m.Path) info.Dir = filepath.Join(ModRoot, "vendor", m.Path)
return info return info
} }
...@@ -137,7 +137,7 @@ func moduleInfo(m module.Version, fromBuildList bool) *modinfo.ModulePublic { ...@@ -137,7 +137,7 @@ func moduleInfo(m module.Version, fromBuildList bool) *modinfo.ModulePublic {
} }
} }
} }
if cfg.BuildGetmode == "vendor" { if cfg.BuildMod == "vendor" {
m.Dir = filepath.Join(ModRoot, "vendor", m.Path) m.Dir = filepath.Join(ModRoot, "vendor", m.Path)
} }
} }
......
...@@ -20,12 +20,12 @@ including recording and resolving dependencies on other modules. ...@@ -20,12 +20,12 @@ including recording and resolving dependencies on other modules.
Modules replace the old GOPATH-based approach to specifying Modules replace the old GOPATH-based approach to specifying
which source files are used in a given build. which source files are used in a given build.
Experimental module support Preliminary module support
Go 1.11 includes experimental support for Go modules, Go 1.11 includes preliminary support for Go modules,
including a new module-aware 'go get' command. including a new module-aware 'go get' command.
We intend to keep revising this support, while preserving compatibility, We intend to keep revising this support, while preserving compatibility,
until it can be declared official (no longer experimental), until it can be declared official (no longer preliminary),
and then at a later point we may remove support for work and then at a later point we may remove support for work
in GOPATH and the old 'go get' command. in GOPATH and the old 'go get' command.
...@@ -173,6 +173,19 @@ automatically make any implied upgrades and update go.mod to reflect them. ...@@ -173,6 +173,19 @@ automatically make any implied upgrades and update go.mod to reflect them.
The 'go mod' command provides other functionality for use in maintaining The 'go mod' command provides other functionality for use in maintaining
and understanding modules and go.mod files. See 'go help mod'. and understanding modules and go.mod files. See 'go help mod'.
The -mod build flag provides additional control over updating and use of go.mod.
If invoked with -mod=readonly, the go command is disallowed from the implicit
automatic updating of go.mod described above. Instead, it fails when any changes
to go.mod are needed. This setting is most useful to check that go.mod does
not need updates, such as in a continuous integration and testing system.
The "go get" command remains permitted to update go.mod even with -mod=readonly,
and the "go mod" commands do not take the -mod flag (or any other build flags).
If invoked with -mod=vendor, the go command assumes that the vendor
directory holds the correct copies of dependencies and ignores
the dependency descriptions in go.mod.
Pseudo-versions Pseudo-versions
The go.mod file and the go command more generally use semantic versions as The go.mod file and the go command more generally use semantic versions as
...@@ -363,7 +376,7 @@ tests of packages in the main module. ...@@ -363,7 +376,7 @@ tests of packages in the main module.
To build using the main module's top-level vendor directory to satisfy To build using the main module's top-level vendor directory to satisfy
dependencies (disabling use of the usual network sources and local dependencies (disabling use of the usual network sources and local
caches), use 'go build -getmode=vendor'. Note that only the main module's caches), use 'go build -mod=vendor'. Note that only the main module's
top-level vendor directory is used; vendor directories in other locations top-level vendor directory is used; vendor directories in other locations
are still ignored. are still ignored.
`, `,
......
...@@ -66,9 +66,9 @@ func Import(path string) (m module.Version, dir string, err error) { ...@@ -66,9 +66,9 @@ func Import(path string) (m module.Version, dir string, err error) {
} }
} }
// -getmode=vendor is special. // -mod=vendor is special.
// Everything must be in the main module or the main module's vendor directory. // Everything must be in the main module or the main module's vendor directory.
if cfg.BuildGetmode == "vendor" { if cfg.BuildMod == "vendor" {
mainDir, mainOK := dirInModule(path, Target.Path, ModRoot, true) mainDir, mainOK := dirInModule(path, Target.Path, ModRoot, true)
vendorDir, vendorOK := dirInModule(path, "", filepath.Join(ModRoot, "vendor"), false) vendorDir, vendorOK := dirInModule(path, "", filepath.Join(ModRoot, "vendor"), false)
if mainOK && vendorOK { if mainOK && vendorOK {
...@@ -146,8 +146,8 @@ func Import(path string) (m module.Version, dir string, err error) { ...@@ -146,8 +146,8 @@ func Import(path string) (m module.Version, dir string, err error) {
// Look up module containing the package, for addition to the build list. // Look up module containing the package, for addition to the build list.
// Goal is to determine the module, download it to dir, and return m, dir, ErrMissing. // Goal is to determine the module, download it to dir, and return m, dir, ErrMissing.
if cfg.BuildGetmode == "local" { if cfg.BuildMod == "readonly" {
return module.Version{}, "", fmt.Errorf("import lookup disabled by -getmode=local") return module.Version{}, "", fmt.Errorf("import lookup disabled by -mod=%s", cfg.BuildMod)
} }
for p := path; p != "."; p = pathpkg.Dir(p) { for p := path; p != "."; p = pathpkg.Dir(p) {
......
...@@ -525,8 +525,6 @@ func WriteGoMod() { ...@@ -525,8 +525,6 @@ func WriteGoMod() {
return return
} }
modfetch.WriteGoSum()
if loaded != nil { if loaded != nil {
reqs := MinReqs() reqs := MinReqs()
min, err := reqs.Required(Target) min, err := reqs.Required(Target)
...@@ -550,12 +548,15 @@ func WriteGoMod() { ...@@ -550,12 +548,15 @@ func WriteGoMod() {
if err != nil { if err != nil {
base.Fatalf("go: %v", err) base.Fatalf("go: %v", err)
} }
if bytes.Equal(old, new) { if !bytes.Equal(old, new) {
return if cfg.BuildMod == "readonly" {
base.Fatalf("go: updates to go.mod needed, disabled by -mod=readonly")
} }
if err := ioutil.WriteFile(file, new, 0666); err != nil { if err := ioutil.WriteFile(file, new, 0666); err != nil {
base.Fatalf("go: %v", err) base.Fatalf("go: %v", err)
} }
}
modfetch.WriteGoSum()
} }
func fixVersion(path, vers string) (string, error) { func fixVersion(path, vers string) (string, error) {
......
...@@ -651,6 +651,7 @@ func Replacement(mod module.Version) module.Version { ...@@ -651,6 +651,7 @@ func Replacement(mod module.Version) module.Version {
// Happens during testing. // Happens during testing.
return module.Version{} return module.Version{}
} }
var found *modfile.Replace var found *modfile.Replace
for _, r := range modFile.Replace { for _, r := range modFile.Replace {
if r.Old.Path == mod.Path && (r.Old.Version == "" || r.Old.Version == mod.Version) { if r.Old.Path == mod.Path && (r.Old.Version == "" || r.Old.Version == mod.Version) {
...@@ -761,7 +762,7 @@ func (r *mvsReqs) required(mod module.Version) ([]module.Version, error) { ...@@ -761,7 +762,7 @@ func (r *mvsReqs) required(mod module.Version) ([]module.Version, error) {
return append(list, r.buildList[1:]...), nil return append(list, r.buildList[1:]...), nil
} }
if cfg.BuildGetmode == "vendor" { if cfg.BuildMod == "vendor" {
// For every module other than the target, // For every module other than the target,
// return the full list of modules from modules.txt. // return the full list of modules from modules.txt.
readVendorList() readVendorList()
......
...@@ -27,33 +27,33 @@ var vetFlagDefn = []*cmdflag.Defn{ ...@@ -27,33 +27,33 @@ var vetFlagDefn = []*cmdflag.Defn{
// to vet. We handle them in vetFlags. // to vet. We handle them in vetFlags.
// local. // local.
{Name: "all", BoolVar: new(bool)}, {Name: "all", BoolVar: new(bool), PassToTest: true},
{Name: "asmdecl", BoolVar: new(bool)}, {Name: "asmdecl", BoolVar: new(bool), PassToTest: true},
{Name: "assign", BoolVar: new(bool)}, {Name: "assign", BoolVar: new(bool), PassToTest: true},
{Name: "atomic", BoolVar: new(bool)}, {Name: "atomic", BoolVar: new(bool), PassToTest: true},
{Name: "bool", BoolVar: new(bool)}, {Name: "bool", BoolVar: new(bool), PassToTest: true},
{Name: "buildtags", BoolVar: new(bool)}, {Name: "buildtags", BoolVar: new(bool), PassToTest: true},
{Name: "cgocall", BoolVar: new(bool)}, {Name: "cgocall", BoolVar: new(bool), PassToTest: true},
{Name: "composites", BoolVar: new(bool)}, {Name: "composites", BoolVar: new(bool), PassToTest: true},
{Name: "copylocks", BoolVar: new(bool)}, {Name: "copylocks", BoolVar: new(bool), PassToTest: true},
{Name: "httpresponse", BoolVar: new(bool)}, {Name: "httpresponse", BoolVar: new(bool), PassToTest: true},
{Name: "lostcancel", BoolVar: new(bool)}, {Name: "lostcancel", BoolVar: new(bool), PassToTest: true},
{Name: "methods", BoolVar: new(bool)}, {Name: "methods", BoolVar: new(bool), PassToTest: true},
{Name: "nilfunc", BoolVar: new(bool)}, {Name: "nilfunc", BoolVar: new(bool), PassToTest: true},
{Name: "printf", BoolVar: new(bool)}, {Name: "printf", BoolVar: new(bool), PassToTest: true},
{Name: "printfuncs"}, {Name: "printfuncs", PassToTest: true},
{Name: "rangeloops", BoolVar: new(bool)}, {Name: "rangeloops", BoolVar: new(bool), PassToTest: true},
{Name: "shadow", BoolVar: new(bool)}, {Name: "shadow", BoolVar: new(bool), PassToTest: true},
{Name: "shadowstrict", BoolVar: new(bool)}, {Name: "shadowstrict", BoolVar: new(bool), PassToTest: true},
{Name: "shift", BoolVar: new(bool)}, {Name: "shift", BoolVar: new(bool), PassToTest: true},
{Name: "source", BoolVar: new(bool)}, {Name: "source", BoolVar: new(bool), PassToTest: true},
{Name: "structtags", BoolVar: new(bool)}, {Name: "structtags", BoolVar: new(bool), PassToTest: true},
{Name: "tests", BoolVar: new(bool)}, {Name: "tests", BoolVar: new(bool), PassToTest: true},
{Name: "unreachable", BoolVar: new(bool)}, {Name: "unreachable", BoolVar: new(bool), PassToTest: true},
{Name: "unsafeptr", BoolVar: new(bool)}, {Name: "unsafeptr", BoolVar: new(bool), PassToTest: true},
{Name: "unusedfuncs"}, {Name: "unusedfuncs", PassToTest: true},
{Name: "unusedresult", BoolVar: new(bool)}, {Name: "unusedresult", BoolVar: new(bool), PassToTest: true},
{Name: "unusedstringmethods"}, {Name: "unusedstringmethods", PassToTest: true},
} }
var vetTool string var vetTool string
...@@ -91,9 +91,17 @@ func vetFlags(args []string) (passToVet, packageNames []string) { ...@@ -91,9 +91,17 @@ func vetFlags(args []string) (passToVet, packageNames []string) {
if err := f.Value.Set(value); err != nil { if err := f.Value.Set(value); err != nil {
base.Fatalf("invalid flag argument for -%s: %v", f.Name, err) base.Fatalf("invalid flag argument for -%s: %v", f.Name, err)
} }
keep := f.PassToTest
if !keep {
// A build flag, probably one we don't want to pass to vet.
// Can whitelist.
switch f.Name { switch f.Name {
case "tags", "v":
keep = true
}
}
if !keep {
// Flags known to the build but not to vet, so must be dropped. // Flags known to the build but not to vet, so must be dropped.
case "a", "x", "n", "vettool", "compiler":
if extraWord { if extraWord {
args = append(args[:i], args[i+2:]...) args = append(args[:i], args[i+2:]...)
extraWord = false extraWord = false
......
...@@ -86,8 +86,6 @@ and test commands: ...@@ -86,8 +86,6 @@ and test commands:
arguments to pass on each gccgo compiler/linker invocation. arguments to pass on each gccgo compiler/linker invocation.
-gcflags '[pattern=]arg list' -gcflags '[pattern=]arg list'
arguments to pass on each go tool compile invocation. arguments to pass on each go tool compile invocation.
-getmode mode
module download mode to use. See 'go help modules' for more.
-installsuffix suffix -installsuffix suffix
a suffix to use in the name of the package installation directory, a suffix to use in the name of the package installation directory,
in order to keep output separate from default builds. in order to keep output separate from default builds.
...@@ -100,6 +98,9 @@ and test commands: ...@@ -100,6 +98,9 @@ and test commands:
-linkshared -linkshared
link against shared libraries previously created with link against shared libraries previously created with
-buildmode=shared. -buildmode=shared.
-mod mode
module download mode to use: readonly, release, or vendor.
See 'go help modules' for more.
-pkgdir dir -pkgdir dir
install and load all packages from dir instead of the usual locations. install and load all packages from dir instead of the usual locations.
For example, when building with a non-standard configuration, For example, when building with a non-standard configuration,
...@@ -220,7 +221,7 @@ func AddBuildFlags(cmd *base.Command) { ...@@ -220,7 +221,7 @@ func AddBuildFlags(cmd *base.Command) {
cmd.Flag.StringVar(&cfg.BuildBuildmode, "buildmode", "default", "") cmd.Flag.StringVar(&cfg.BuildBuildmode, "buildmode", "default", "")
cmd.Flag.Var(&load.BuildGcflags, "gcflags", "") cmd.Flag.Var(&load.BuildGcflags, "gcflags", "")
cmd.Flag.Var(&load.BuildGccgoflags, "gccgoflags", "") cmd.Flag.Var(&load.BuildGccgoflags, "gccgoflags", "")
cmd.Flag.StringVar(&cfg.BuildGetmode, "getmode", "", "") cmd.Flag.StringVar(&cfg.BuildMod, "mod", "", "")
cmd.Flag.StringVar(&cfg.BuildContext.InstallSuffix, "installsuffix", "", "") cmd.Flag.StringVar(&cfg.BuildContext.InstallSuffix, "installsuffix", "", "")
cmd.Flag.Var(&load.BuildLdflags, "ldflags", "") cmd.Flag.Var(&load.BuildLdflags, "ldflags", "")
cmd.Flag.BoolVar(&cfg.BuildLinkshared, "linkshared", false, "") cmd.Flag.BoolVar(&cfg.BuildLinkshared, "linkshared", false, "")
......
...@@ -14,6 +14,7 @@ import ( ...@@ -14,6 +14,7 @@ import (
"fmt" "fmt"
"os" "os"
"path/filepath" "path/filepath"
"strings"
) )
func BuildInit() { func BuildInit() {
...@@ -227,15 +228,30 @@ func buildModeInit() { ...@@ -227,15 +228,30 @@ func buildModeInit() {
} }
} }
switch cfg.BuildGetmode { switch cfg.BuildMod {
case "": case "":
// ok // ok
case "local", "vendor": case "readonly", "vendor":
// ok but check for modules if load.ModLookup == nil && !inGOFLAGS("-mod") {
if load.ModLookup == nil { base.Fatalf("build flag -mod=%s only valid when using modules", cfg.BuildMod)
base.Fatalf("build flag -getmode=%s only valid when using modules", cfg.BuildGetmode)
} }
default: default:
base.Fatalf("-getmode=%s not supported (can be '', 'local', or 'vendor')", cfg.BuildGetmode) base.Fatalf("-mod=%s not supported (can be '', 'readonly', or 'vendor')", cfg.BuildMod)
} }
} }
func inGOFLAGS(flag string) bool {
for _, goflag := range base.GOFLAGS() {
name := goflag
if strings.HasPrefix(name, "--") {
name = name[1:]
}
if i := strings.Index(name, "="); i >= 0 {
name = name[:i]
}
if name == flag {
return true
}
}
return false
}
# Use download cache for -getmode=local. # Allow (cached) downloads for -mod=readonly.
env GO111MODULE=on env GO111MODULE=on
env GOPATH=$WORK/gopath1 env GOPATH=$WORK/gopath1
cd $WORK/x cd $WORK/x
! go list -getmode=local go mod edit -fmt
go list go list -mod=readonly
go list -getmode=local
env GOPROXY=file:///nonexist env GOPROXY=file:///nonexist
go list -getmode=local go list
grep v1.5.1 $GOPATH/src/mod/cache/download/rsc.io/quote/@v/list grep v1.5.1 $GOPATH/src/mod/cache/download/rsc.io/quote/@v/list
# Use download cache as file:/// proxy. # Use download cache as file:/// proxy.
[windows] stop # TODO: file://$WORK puts backslashes in the URL [windows] stop # TODO: file://$WORK puts backslashes in the URL
env GOPATH=$WORK/gopath2 env GOPATH=$WORK/gopath2
env GOPROXY=file:///nonexist env GOPROXY=file:///nonexist
! go list -getmode=local
! go list ! go list
env GOPROXY=file://$WORK/gopath1/src/mod/cache/download env GOPROXY=file://$WORK/gopath1/src/mod/cache/download
! go list -getmode=local
go list go list
grep v1.5.1 $GOPATH/src/mod/cache/download/rsc.io/quote/@v/list grep v1.5.1 $GOPATH/src/mod/cache/download/rsc.io/quote/@v/list
......
...@@ -5,15 +5,15 @@ go mod vendor ...@@ -5,15 +5,15 @@ go mod vendor
env GOPATH=$WORK/empty env GOPATH=$WORK/empty
env GOPROXY=file:///nonexist env GOPROXY=file:///nonexist
go list -getmode=vendor go list -mod=vendor
go list -getmode=vendor -m -f '{{.Path}} {{.Version}} {{.Dir}}' all go list -mod=vendor -m -f '{{.Path}} {{.Version}} {{.Dir}}' all
stdout '^rsc.io/quote v1.5.1 .*vendor[\\/]rsc.io[\\/]quote$' stdout '^rsc.io/quote v1.5.1 .*vendor[\\/]rsc.io[\\/]quote$'
stdout '^golang.org/x/text v0.0.0.* .*vendor[\\/]golang.org[\\/]x[\\/]text$' stdout '^golang.org/x/text v0.0.0.* .*vendor[\\/]golang.org[\\/]x[\\/]text$'
! go list -getmode=vendor -m rsc.io/quote@latest ! go list -mod=vendor -m rsc.io/quote@latest
stderr 'module lookup disabled by -getmode=vendor' stderr 'module lookup disabled by -mod=vendor'
! go get -getmode=vendor -u ! go get -mod=vendor -u
stderr 'go get: disabled by -getmode=vendor' stderr 'go get: disabled by -mod=vendor'
-- go.mod -- -- go.mod --
module x module x
......
env GO111MODULE=on
# -mod=readonly must not resolve missing modules nor update go.mod
env GOFLAGS=-mod=readonly
go mod edit -fmt
cp go.mod go.mod.empty
! go list
stderr 'import lookup disabled by -mod=readonly'
cmp go.mod go.mod.empty
# update go.mod - go get allowed
go get rsc.io/quote
grep rsc.io/quote go.mod
# update go.mod - go mod tidy allowed
cp go.mod.empty go.mod
go mod tidy
# -mod=readonly must succeed once go.mod is up-to-date...
go list
# ... even if it needs downloads
go clean -modcache
go list
# -mod=readonly should reject inconsistent go.mod files
# (ones that would be rewritten).
go mod edit -require rsc.io/sampler@v1.2.0
cp go.mod go.mod.inconsistent
! go list
stderr 'go: updates to go.mod needed, disabled by -mod=readonly'
cmp go.mod go.mod.inconsistent
-- go.mod --
module m
-- x.go --
package x
import _ "rsc.io/quote"
...@@ -5,7 +5,7 @@ stdout '^x v1.0.0 => ./x' ...@@ -5,7 +5,7 @@ stdout '^x v1.0.0 => ./x'
stdout '^w' stdout '^w'
[!short] go build [!short] go build
[!short] ! go build -getmode=vendor [!short] ! go build -mod=vendor
go list -f {{.Dir}} x go list -f {{.Dir}} x
stdout 'src[\\/]x' stdout 'src[\\/]x'
...@@ -25,16 +25,16 @@ stdout 'src[\\/]x' ...@@ -25,16 +25,16 @@ stdout 'src[\\/]x'
go list -f {{.Dir}} -m x go list -f {{.Dir}} -m x
stdout 'src[\\/]x' stdout 'src[\\/]x'
go list -getmode=vendor -f {{.Dir}} x go list -mod=vendor -f {{.Dir}} x
stdout 'src[\\/]vendor[\\/]x' stdout 'src[\\/]vendor[\\/]x'
go list -getmode=vendor -f {{.Dir}} -m x go list -mod=vendor -f {{.Dir}} -m x
stdout 'src[\\/]vendor[\\/]x' stdout 'src[\\/]vendor[\\/]x'
go list -f {{.Dir}} -m w go list -f {{.Dir}} -m w
stdout 'src[\\/]w' stdout 'src[\\/]w'
! go list -getmode=vendor -f {{.Dir}} w ! go list -mod=vendor -f {{.Dir}} w
stderr 'src[\\/]vendor[\\/]w' stderr 'src[\\/]vendor[\\/]w'
! exists vendor/x/testdata ! exists vendor/x/testdata
...@@ -58,9 +58,9 @@ exists vendor/mysite/myname/mypkg/LICENSE.txt ...@@ -58,9 +58,9 @@ exists vendor/mysite/myname/mypkg/LICENSE.txt
[short] stop [short] stop
go build go build
go build -getmode=vendor go build -mod=vendor
go test -getmode=vendor . ./subdir go test -mod=vendor . ./subdir
go test -getmode=vendor ./... go test -mod=vendor ./...
-- go.mod -- -- go.mod --
module m module m
......
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