Commit a1cbbe0d authored by Bryan C. Mills's avatar Bryan C. Mills

cmd/go/internal/modload: report errors explicitly from Lookup

Previously, we reported errors directly in (*loader).load via base.Errorf.
Unfortunately, (*loader).load can be called from contexts in which such errors
should not be considered fatal, such as by load.PackagesAndErrors.

Instead, we save the errors in pkg.err and modify Lookup to return that error.

This change is a bit awkward: we end up suppressing a "no Go files" error for
packages at the root of newly-imported modules, even if they really do contain
source files. I believe that that's due to a special-case lookup for modules in
the build list, which allows us to "validate" imports for modules in the build
list even though we haven't actually downloaded their sources (or verified that
they actually contain the requested package). The fix for that issue is in the
change that follows this one.

Fixes #26602.

Change-Id: I16f00ceb143fbb797cfc3cb07fd08aeb6154575b
Reviewed-on: https://go-review.googlesource.com/127936
Run-TryBot: Bryan C. Mills <bcmills@google.com>
Reviewed-by: default avatarRuss Cox <rsc@golang.org>
TryBot-Result: Gobot Gobot <gobot@golang.org>
parent 79bf7955
...@@ -515,13 +515,28 @@ func runGet(cmd *base.Command, args []string) { ...@@ -515,13 +515,28 @@ func runGet(cmd *base.Command, args []string) {
} }
if len(install) > 0 { if len(install) > 0 {
// All requested versions were explicitly @none.
// Note that 'go get -u' without any arguments results in len(install) == 1:
// search.CleanImportPaths returns "." for empty args.
work.BuildInit() work.BuildInit()
var pkgs []string var pkgs []string
for _, p := range load.PackagesAndErrors(install) { for _, p := range load.PackagesAndErrors(install) {
if p.Error == nil || !strings.HasPrefix(p.Error.Err, "no Go files") { // Ignore "no Go source files" errors for 'go get' operations on modules.
pkgs = append(pkgs, p.ImportPath) if p.Error != nil {
if len(args) == 0 && getU != "" && strings.HasPrefix(p.Error.Err, "no Go files") {
// Upgrading modules: skip the implicitly-requested package at the
// current directory, even if it is not tho module root.
continue
}
if strings.HasPrefix(p.Error.Err, "no Go files") && modload.ModuleInfo(p.ImportPath) != nil {
// Explicitly-requested module, but it doesn't contain a package at the
// module root.
continue
}
} }
pkgs = append(pkgs, p.ImportPath)
} }
// If -d was specified, we're done after the download: no build. // If -d was specified, we're done after the download: no build.
// (The load.PackagesAndErrors is what did the download // (The load.PackagesAndErrors is what did the download
// of the named packages and their dependencies.) // of the named packages and their dependencies.)
...@@ -564,7 +579,9 @@ func getQuery(path, vers string, forceModulePath bool) (module.Version, error) { ...@@ -564,7 +579,9 @@ func getQuery(path, vers string, forceModulePath bool) (module.Version, error) {
// if found in the current source code. // if found in the current source code.
// Then apply the version to that module. // Then apply the version to that module.
m, _, err := modload.Import(path) m, _, err := modload.Import(path)
if err != nil { if e, ok := err.(*modload.ImportMissingError); ok && e.Module.Path != "" {
m = e.Module
} else if err != nil {
return module.Version{}, err return module.Version{}, err
} }
if m.Path == "" { if m.Path == "" {
......
...@@ -158,6 +158,9 @@ func ImportPaths(args []string) []string { ...@@ -158,6 +158,9 @@ func ImportPaths(args []string) []string {
have[path] = true have[path] = true
if path == "all" { if path == "all" {
for _, pkg := range loaded.pkgs { for _, pkg := range loaded.pkgs {
if e, ok := pkg.err.(*ImportMissingError); ok && e.Module.Path == "" {
continue // Package doesn't actually exist, so don't report it.
}
if !have[pkg.path] { if !have[pkg.path] {
have[pkg.path] = true have[pkg.path] = true
final = append(final, pkg.path) final = append(final, pkg.path)
...@@ -270,6 +273,9 @@ func loadAll(testAll bool) []string { ...@@ -270,6 +273,9 @@ func loadAll(testAll bool) []string {
var paths []string var paths []string
for _, pkg := range loaded.pkgs { for _, pkg := range loaded.pkgs {
if e, ok := pkg.err.(*ImportMissingError); ok && e.Module.Path == "" {
continue // Package doesn't actually exist.
}
paths = append(paths, pkg.path) paths = append(paths, pkg.path)
} }
return paths return paths
...@@ -337,21 +343,22 @@ func ModuleUsedDirectly(path string) bool { ...@@ -337,21 +343,22 @@ func ModuleUsedDirectly(path string) bool {
return loaded.direct[path] return loaded.direct[path]
} }
// Lookup returns the source directory and import path for the package at path. // Lookup returns the source directory, import path, and any loading error for
// the package at path.
// Lookup requires that one of the Load functions in this package has already // Lookup requires that one of the Load functions in this package has already
// been called. // been called.
func Lookup(path string) (dir, realPath string, err error) { func Lookup(path string) (dir, realPath string, err error) {
realPath = ImportMap(path) pkg, ok := loaded.pkgCache.Get(path).(*loadPkg)
if realPath == "" { if !ok {
if isStandardImportPath(path) { if isStandardImportPath(path) {
dir := filepath.Join(cfg.GOROOT, "src", path) dir := filepath.Join(cfg.GOROOT, "src", path)
if _, err := os.Stat(dir); err == nil { if _, err := os.Stat(dir); err == nil {
return dir, path, nil return dir, path, nil
} }
} }
return "", "", fmt.Errorf("no such package in module") return "", "", errMissing
} }
return PackageDir(realPath), realPath, nil return pkg.dir, pkg.path, pkg.err
} }
// A loader manages the process of loading information about // A loader manages the process of loading information about
...@@ -459,9 +466,7 @@ func (ld *loader) load(roots func() []string) { ...@@ -459,9 +466,7 @@ func (ld *loader) load(roots func() []string) {
} }
continue continue
} }
if pkg.err != nil { // Leave other errors for Import or load.Packages to report.
base.Errorf("go: %s: %s", pkg.stackText(), pkg.err)
}
} }
base.ExitIfErrors() base.ExitIfErrors()
if numAdded == 0 { if numAdded == 0 {
...@@ -560,11 +565,6 @@ func (ld *loader) doPkg(item interface{}) { ...@@ -560,11 +565,6 @@ func (ld *loader) doPkg(item interface{}) {
var err error var err error
imports, testImports, err = scanDir(pkg.dir, ld.tags) imports, testImports, err = scanDir(pkg.dir, ld.tags)
if err != nil { if err != nil {
if strings.HasPrefix(err.Error(), "no Go ") {
// Don't print about directories with no Go source files.
// Let the eventual real package load do that.
return
}
pkg.err = err pkg.err = err
return return
} }
......
...@@ -6,16 +6,24 @@ stderr 'cannot find module providing package appengine' ...@@ -6,16 +6,24 @@ stderr 'cannot find module providing package appengine'
! go get x/y.z ! go get x/y.z
stderr 'cannot find module providing package x/y.z' stderr 'cannot find module providing package x/y.z'
# build should skip over appengine imports # build should report all unsatisfied imports,
! go build # but should be more definitive about non-module import paths
! stderr appengine ! go build ./useappengine
stderr 'cannot find package'
! go build ./usenonexistent
stderr 'cannot find module providing package nonexistent.rsc.io' stderr 'cannot find module providing package nonexistent.rsc.io'
# go mod vendor and go mod tidy should ignore appengine imports.
rm usenonexistent/x.go
go mod tidy
go mod vendor
-- go.mod -- -- go.mod --
module x module x
-- x.go -- -- useappengine/x.go --
package x package useappengine
import _ "appengine" // package does not exist
import _ "appengine" -- usenonexistent/x.go --
package usenonexistent
import _ "nonexistent.rsc.io" // domain does not exist import _ "nonexistent.rsc.io" // domain does not exist
...@@ -2,12 +2,6 @@ env GO111MODULE=on ...@@ -2,12 +2,6 @@ env GO111MODULE=on
# @commit should resolve # @commit should resolve
# go get should skip build with no Go files in root
go get golang.org/x/text@14c0d48
# ... and go get should skip build with -m
go get -m golang.org/x/text@14c0d48
# golang.org/x/text/language@commit should not resolve with -m, # golang.org/x/text/language@commit should not resolve with -m,
# because that's not a module path. # because that's not a module path.
! go get -m golang.org/x/text/language@14c0d48 ! go get -m golang.org/x/text/language@14c0d48
...@@ -17,6 +11,12 @@ go get -m golang.org/x/text@14c0d48 ...@@ -17,6 +11,12 @@ go get -m golang.org/x/text@14c0d48
go get -d -x golang.org/x/text/language@14c0d48 go get -d -x golang.org/x/text/language@14c0d48
! stderr 'compile|cp|gccgo .*language\.a$' ! stderr 'compile|cp|gccgo .*language\.a$'
# go get should skip build with no Go files in root
go get golang.org/x/text@14c0d48
# ... and go get should skip build with -m
go get -m golang.org/x/text@14c0d48
# dropping -d, we should see a build. # dropping -d, we should see a build.
go get -x golang.org/x/text/language@14c0d48 go get -x golang.org/x/text/language@14c0d48
stderr 'compile|cp|gccgo .*language\.a$' stderr 'compile|cp|gccgo .*language\.a$'
......
...@@ -11,8 +11,14 @@ go list -m -f '{{.Path}} {{.Version}}{{if .Indirect}} // indirect{{end}}' all ...@@ -11,8 +11,14 @@ go list -m -f '{{.Path}} {{.Version}}{{if .Indirect}} // indirect{{end}}' all
stdout '^golang.org/x/text [v0-9a-f\.-]+ // indirect' stdout '^golang.org/x/text [v0-9a-f\.-]+ // indirect'
grep 'golang.org/x/text [v0-9a-f\.-]+ // indirect' go.mod grep 'golang.org/x/text [v0-9a-f\.-]+ // indirect' go.mod
# indirect tag should be removed upon seeing direct import # importing an empty module root as a package makes it direct.
# TODO(bcmills): This doesn't seem correct. Fix is in the next change.
cp $WORK/tmp/usetext.go x.go cp $WORK/tmp/usetext.go x.go
go list -e
grep 'golang.org/x/text [v0-9a-f\.-]+$' go.mod
# indirect tag should be removed upon seeing direct import.
cp $WORK/tmp/uselang.go x.go
go list go list
grep 'rsc.io/quote v1.5.2$' go.mod grep 'rsc.io/quote v1.5.2$' go.mod
grep 'golang.org/x/text [v0-9a-f\.-]+$' go.mod grep 'golang.org/x/text [v0-9a-f\.-]+$' go.mod
...@@ -24,7 +30,7 @@ grep 'rsc.io/quote v1.5.2$' go.mod ...@@ -24,7 +30,7 @@ grep 'rsc.io/quote v1.5.2$' go.mod
grep 'golang.org/x/text [v0-9a-f\.-]+ // indirect' go.mod grep 'golang.org/x/text [v0-9a-f\.-]+ // indirect' go.mod
# requirement should be dropped entirely if not needed # requirement should be dropped entirely if not needed
cp $WORK/tmp/usetext.go x.go cp $WORK/tmp/uselang.go x.go
go mod tidy go mod tidy
! grep rsc.io/quote go.mod ! grep rsc.io/quote go.mod
grep 'golang.org/x/text [v0-9a-f\.-]+$' go.mod grep 'golang.org/x/text [v0-9a-f\.-]+$' go.mod
...@@ -37,6 +43,9 @@ package x ...@@ -37,6 +43,9 @@ package x
-- $WORK/tmp/usetext.go -- -- $WORK/tmp/usetext.go --
package x package x
import _ "golang.org/x/text" import _ "golang.org/x/text"
-- $WORK/tmp/uselang.go --
package x
import _ "golang.org/x/text/language"
-- $WORK/tmp/usequote.go -- -- $WORK/tmp/usequote.go --
package x package x
import _ "rsc.io/quote" import _ "rsc.io/quote"
...@@ -4,25 +4,35 @@ ...@@ -4,25 +4,35 @@
env GO111MODULE=on env GO111MODULE=on
cd example.com cd example.com
# Listing an otherwise-valid package with an unsatisfied direct import should succeed, # Without -e, listing an otherwise-valid package with an unsatisfied direct import should fail.
# but name that package in DepsErrors. # BUG: Today it succeeds.
! go list -f '{{if .Error}}error{{end}} {{if .Incomplete}}incomplete{{end}} {{range .DepsErrors}}bad dep: {{.Err}}{{end}}' example.com/direct go list -f '{{if .Error}}error{{end}} {{if .Incomplete}}incomplete{{end}} {{range .DepsErrors}}bad dep: {{.Err}}{{end}}' example.com/direct
stderr example.com[/\\]notfound ! stdout ^error
stdout 'incomplete'
stdout 'bad dep: .*example.com/notfound'
# Listing with -deps should also fail. # Listing with -deps should also fail.
! go list -deps example.com/direct # BUG: Today, it does not.
stderr example.com[/\\]notfound # ! go list -deps example.com/direct
# stderr example.com/notfound
go list -deps example.com/direct
stdout example.com/notfound
# Listing an otherwise-valid package that imports some *other* package with an # Listing an otherwise-valid package that imports some *other* package with an
# unsatisfied import should also succeed. # unsatisfied import should also fail.
# NOTE: This behavior differs between GOPATH mode and module mode. # BUG: Today, it succeeds.
! go list -f '{{if .Error}}error{{end}} {{if .Incomplete}}incomplete{{end}} {{range .DepsErrors}}bad dep: {{.Err}}{{end}}' example.com/indirect go list -f '{{if .Error}}error{{end}} {{if .Incomplete}}incomplete{{end}} {{range .DepsErrors}}bad dep: {{.Err}}{{end}}' example.com/indirect
stderr example.com[/\\]notfound ! stdout ^error
stdout incomplete
stdout 'bad dep: .*example.com/notfound'
# Again, -deps should fail. # Again, -deps should fail.
! go list -deps example.com/indirect # BUG: Again, it does not.
stderr example.com[/\\]notfound # ! go list -deps example.com/indirect
# stderr example.com/notfound
go list -deps example.com/indirect
stdout example.com/notfound
# Listing the missing dependency directly should fail outright... # Listing the missing dependency directly should fail outright...
...@@ -32,16 +42,17 @@ stderr 'cannot find module providing package example.com/notfound' ...@@ -32,16 +42,17 @@ stderr 'cannot find module providing package example.com/notfound'
! stdout incomplete ! stdout incomplete
# ...but listing with -e should succeed. # ...but listing with -e should succeed.
# BUG: Today, it fails. go list -e -f '{{if .Error}}error{{end}} {{if .Incomplete}}incomplete{{end}}' example.com/notfound
! go list -e -f '{{if .Error}}error{{end}} {{if .Incomplete}}incomplete{{end}}' example.com/notfound stdout error
stderr example.com[/\\]notfound stdout incomplete
# The pattern "all" should match only packages that acutally exist, # The pattern "all" should match only packages that acutally exist,
# ignoring those whose existence is merely implied by imports. # ignoring those whose existence is merely implied by imports.
# BUG: Today, `go list -e` fails if there are any unresolved imports. go list -e -f '{{.ImportPath}}' all
! go list -e -f '{{.ImportPath}}' all stdout example.com/direct
stderr example.com[/\\]notfound stdout example.com/indirect
! stdout example.com/notfound
-- example.com/go.mod -- -- example.com/go.mod --
......
env GO111MODULE=on env GO111MODULE=on
# -mod=readonly must not resolve missing modules nor update go.mod # -mod=readonly must not resolve missing modules nor update go.mod
#
# TODO(bcmills): 'go list' should suffice, but today it does not fail due to
# unresolved imports. When that is fixed, use 'go list' instead of 'go list all'.
env GOFLAGS=-mod=readonly env GOFLAGS=-mod=readonly
go mod edit -fmt go mod edit -fmt
cp go.mod go.mod.empty cp go.mod go.mod.empty
! go list ! go list all
stderr 'import lookup disabled by -mod=readonly' stderr 'import lookup disabled by -mod=readonly'
cmp go.mod go.mod.empty cmp go.mod go.mod.empty
......
...@@ -155,6 +155,12 @@ package m ...@@ -155,6 +155,12 @@ package m
import _ "appengine" import _ "appengine"
import _ "appengine/datastore" import _ "appengine/datastore"
-- nonexistent.go --
// +build alternatereality
package m
import _ "nonexistent.rsc.io"
-- mypkg/go.mod -- -- mypkg/go.mod --
module me module me
-- mypkg/mydir/d.go -- -- mypkg/mydir/d.go --
......
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