gopls/internal/lsp/source: references: restrict search to workspace packages
This change restricts the scope of the search to workspace
packages so that we don't report irrelevant references,
and more importantly, spend a lot of time type-checking them.
Conceptually, we simply call call snapshot.WorkspaceMetadata
and intersect the resulits of the ReverseDependency operations
with it.
However, the diff is somewhat trickier because it merges the
old call to snapshot.WorkspaceMetadata within expandMethodSearch
into the new one, and downgrades 'expansions' into a set of IDs
so that expandMethodSearch needn't trade in Metadatas.
It also factors the removal of intermediate test variants
that was previously done in two places, by applying it
to the workspace set.
Casual testing (searching for refs to os.File.Close in
kubernetes, using the CLI tool which performs an IWL)
suggests that it is >2x faster:
v0.11.0 = 35s; master = 20s; this CL = 14s.
Also, a test.
Fixes golang/go#59674
Change-Id: I58c072af7f878f4d64ceffd5b62a68bc3e5daae4
Reviewed-on: https://go-review.googlesource.com/c/tools/+/487017
TryBot-Result: Gopher Robot <gobot@golang.org>
Run-TryBot: Alan Donovan <adonovan@google.com>
gopls-CI: kokoro <noreply+kokoro@google.com>
Reviewed-by: Robert Findley <rfindley@google.com>
diff --git a/gopls/internal/lsp/cache/load.go b/gopls/internal/lsp/cache/load.go
index d24c792..5f796f0 100644
--- a/gopls/internal/lsp/cache/load.go
+++ b/gopls/internal/lsp/cache/load.go
@@ -712,8 +712,9 @@
return false
}
-// computeWorkspacePackagesLocked computes workspace packages in the snapshot s
-// for the given metadata graph.
+// computeWorkspacePackagesLocked computes workspace packages in the
+// snapshot s for the given metadata graph. The result does not
+// contain intermediate test variants.
//
// s.mu must be held while calling this function.
func computeWorkspacePackagesLocked(s *snapshot, meta *metadataGraph) map[PackageID]PackagePath {
@@ -747,6 +748,7 @@
//
// Notably, this excludes intermediate test variants from workspace
// packages.
+ assert(!m.IsIntermediateTestVariant(), "unexpected ITV")
workspacePackages[m.ID] = m.ForTest
}
}
diff --git a/gopls/internal/lsp/cache/snapshot.go b/gopls/internal/lsp/cache/snapshot.go
index 6557a0c..cf2b930 100644
--- a/gopls/internal/lsp/cache/snapshot.go
+++ b/gopls/internal/lsp/cache/snapshot.go
@@ -126,7 +126,7 @@
analyses *persistent.Map // from analysisKey to analysisPromise
// workspacePackages contains the workspace's packages, which are loaded
- // when the view is created.
+ // when the view is created. It contains no intermediate test variants.
workspacePackages map[PackageID]PackagePath
// shouldLoad tracks packages that need to be reloaded, mapping a PackageID
diff --git a/gopls/internal/lsp/diagnostics.go b/gopls/internal/lsp/diagnostics.go
index 4c595a9..0ba8399 100644
--- a/gopls/internal/lsp/diagnostics.go
+++ b/gopls/internal/lsp/diagnostics.go
@@ -323,8 +323,8 @@
}
store(modVulncheckSource, "diagnosing vulnerabilities", vulnReports, vulnErr, false)
- activeMetas, activeErr := snapshot.WorkspaceMetadata(ctx)
- if s.shouldIgnoreError(ctx, snapshot, activeErr) {
+ workspace, err := snapshot.WorkspaceMetadata(ctx)
+ if s.shouldIgnoreError(ctx, snapshot, err) {
return
}
criticalErr := snapshot.CriticalError(ctx)
@@ -346,7 +346,7 @@
// If there are no workspace packages, there is nothing to diagnose and
// there are no orphaned files.
- if len(activeMetas) == 0 {
+ if len(workspace) == 0 {
return
}
@@ -368,7 +368,7 @@
toDiagnose = make(map[source.PackageID]*source.Metadata)
toAnalyze = make(map[source.PackageID]*source.Metadata)
)
- for _, m := range activeMetas {
+ for _, m := range workspace {
var hasNonIgnored, hasOpenFile bool
for _, uri := range m.CompiledGoFiles {
seen[uri] = struct{}{}
diff --git a/gopls/internal/lsp/source/references.go b/gopls/internal/lsp/source/references.go
index 3b439c7..75e3a61 100644
--- a/gopls/internal/lsp/source/references.go
+++ b/gopls/internal/lsp/source/references.go
@@ -142,7 +142,21 @@
if err != nil {
return nil, err
}
+
+ // Restrict search to workspace packages.
+ workspace, err := snapshot.WorkspaceMetadata(ctx)
+ if err != nil {
+ return nil, err
+ }
+ workspaceMap := make(map[PackageID]*Metadata, len(workspace))
+ for _, m := range workspace {
+ workspaceMap[m.ID] = m
+ }
+
for _, rdep := range rdeps {
+ if _, ok := workspaceMap[rdep.ID]; !ok {
+ continue
+ }
for _, uri := range rdep.CompiledGoFiles {
fh, err := snapshot.ReadFile(ctx, uri)
if err != nil {
@@ -257,9 +271,9 @@
// Is object exported?
// If so, compute scope and targets of the global search.
var (
- globalScope = make(map[PackageID]*Metadata)
+ globalScope = make(map[PackageID]*Metadata) // (excludes ITVs)
globalTargets map[PackagePath]map[objectpath.Path]unit
- expansions = make(map[*Metadata]unit) // packages that caused search expansion
+ expansions = make(map[PackageID]unit) // packages that caused search expansion
)
// TODO(adonovan): what about generic functions? Need to consider both
// uninstantiated and instantiated. The latter have no objectpath. Use Origin?
@@ -269,6 +283,47 @@
pkgPath: {path: {}}, // primary target
}
+ // Compute set of (non-ITV) workspace packages.
+ // We restrict references to this subset.
+ workspace, err := snapshot.WorkspaceMetadata(ctx)
+ if err != nil {
+ return nil, err
+ }
+ workspaceMap := make(map[PackageID]*Metadata, len(workspace))
+ workspaceIDs := make([]PackageID, 0, len(workspace))
+ for _, m := range workspace {
+ workspaceMap[m.ID] = m
+ workspaceIDs = append(workspaceIDs, m.ID)
+ }
+
+ // addRdeps expands the global scope to include the
+ // reverse dependencies of the specified package.
+ addRdeps := func(id PackageID, transitive bool) error {
+ rdeps, err := snapshot.ReverseDependencies(ctx, id, transitive)
+ if err != nil {
+ return err
+ }
+ for rdepID, rdep := range rdeps {
+ // Skip non-workspace packages.
+ //
+ // This means we also skip any expansion of the
+ // search that might be caused by a non-workspace
+ // package, possibly causing us to miss references
+ // to the expanded target set from workspace packages.
+ //
+ // TODO(adonovan): don't skip those expansions.
+ // The challenge is how to so without type-checking
+ // a lot of non-workspace packages not covered by
+ // the initial workspace load.
+ if _, ok := workspaceMap[rdepID]; !ok {
+ continue
+ }
+
+ globalScope[rdepID] = rdep
+ }
+ return nil
+ }
+
// How far need we search?
// For package-level objects, we need only search the direct importers.
// For fields and methods, we must search transitively.
@@ -278,13 +333,9 @@
// (Each set is disjoint so there's no benefit to
// to combining the metadata graph traversals.)
for _, m := range variants {
- rdeps, err := snapshot.ReverseDependencies(ctx, m.ID, transitive)
- if err != nil {
+ if err := addRdeps(m.ID, transitive); err != nil {
return nil, err
}
- for id, rdep := range rdeps {
- globalScope[id] = rdep
- }
}
// Is object a method?
@@ -297,7 +348,7 @@
// 'expansions' records the packages that declared
// such types.
if recv := effectiveReceiver(obj); recv != nil {
- if err := expandMethodSearch(ctx, snapshot, obj.(*types.Func), recv, globalScope, globalTargets, expansions); err != nil {
+ if err := expandMethodSearch(ctx, snapshot, workspaceIDs, obj.(*types.Func), recv, addRdeps, globalTargets, expansions); err != nil {
return nil, err
}
}
@@ -388,18 +439,18 @@
// Also compute local references within packages that declare
// corresponding methods (see above), which expand the global search.
// The target objects are identified by (PkgPath, objectpath).
- for m := range expansions {
- m := m
+ for id := range expansions {
+ id := id
group.Go(func() error {
// TODO(adonovan): opt: batch these TypeChecks.
- pkgs, err := snapshot.TypeCheck(ctx, m.ID)
+ pkgs, err := snapshot.TypeCheck(ctx, id)
if err != nil {
return err
}
pkg := pkgs[0]
targets := make(map[types.Object]bool)
- for objpath := range globalTargets[m.PkgPath] {
+ for objpath := range globalTargets[pkg.Metadata().PkgPath] {
obj, err := objectpath.Object(pkg.GetTypes(), objpath)
if err != nil {
return err // can't happen?
@@ -413,16 +464,7 @@
// Compute global references for selected reverse dependencies.
group.Go(func() error {
var globalIDs []PackageID
- for id, m := range globalScope {
- // Skip intermediate test variants.
- //
- // Strictly, an ITV's cross-reference index
- // may have different objectpaths from the
- // ordinary variant, but we ignore that. See
- // explanation at IsIntermediateTestVariant.
- if m.IsIntermediateTestVariant() {
- continue
- }
+ for id := range globalScope {
globalIDs = append(globalIDs, id)
}
indexes, err := snapshot.References(ctx, globalIDs...)
@@ -444,37 +486,28 @@
}
// expandMethodSearch expands the scope and targets of a global search
-// for an exported method to include all methods that correspond to
-// it through interface satisfaction.
+// for an exported method to include all methods in the workspace
+// that correspond to it through interface satisfaction.
//
// Each package that declares a corresponding type is added to
// expansions so that we can also find local references to the type
// within the package, which of course requires type checking.
//
+// The scope is expanded by a sequence of calls (not concurrent) to addRdeps.
+//
// recv is the method's effective receiver type, for method-set computations.
-func expandMethodSearch(ctx context.Context, snapshot Snapshot, method *types.Func, recv types.Type, scope map[PackageID]*Metadata, targets map[PackagePath]map[objectpath.Path]unit, expansions map[*Metadata]unit) error {
+func expandMethodSearch(ctx context.Context, snapshot Snapshot, workspaceIDs []PackageID, method *types.Func, recv types.Type, addRdeps func(id PackageID, transitive bool) error, targets map[PackagePath]map[objectpath.Path]unit, expansions map[PackageID]unit) error {
// Compute the method-set fingerprint used as a key to the global search.
key, hasMethods := methodsets.KeyOf(recv)
if !hasMethods {
return bug.Errorf("KeyOf(%s)={} yet %s is a method", recv, method)
}
- metas, err := snapshot.AllMetadata(ctx)
- if err != nil {
- return err
- }
- // Discard ITVs to avoid redundant type-checking.
- // (See explanation at IsIntermediateTestVariant.)
- RemoveIntermediateTestVariants(&metas)
- allIDs := make([]PackageID, 0, len(metas))
- for _, m := range metas {
- allIDs = append(allIDs, m.ID)
- }
// Search the methodset index of each package in the workspace.
- indexes, err := snapshot.MethodSets(ctx, allIDs...)
+ indexes, err := snapshot.MethodSets(ctx, workspaceIDs...)
if err != nil {
return err
}
- var mu sync.Mutex // guards scope, targets, expansions
+ var mu sync.Mutex // guards addRdeps, targets, expansions
var group errgroup.Group
for i, index := range indexes {
i := i
@@ -486,21 +519,21 @@
return nil
}
- // Expand global search scope to include rdeps of this pkg.
- rdeps, err := snapshot.ReverseDependencies(ctx, allIDs[i], true)
- if err != nil {
- return err
- }
+ // We have discovered one or more corresponding types.
+ id := workspaceIDs[i]
mu.Lock()
defer mu.Unlock()
- expansions[metas[i]] = unit{}
-
- for _, rdep := range rdeps {
- scope[rdep.ID] = rdep
+ // Expand global search scope to include rdeps of this pkg.
+ if err := addRdeps(id, true); err != nil {
+ return err
}
+ // Mark this package so that we search within it for
+ // local references to the additional types/methods.
+ expansions[id] = unit{}
+
// Add each corresponding method the to set of global search targets.
for _, res := range results {
methodPkg := PackagePath(res.PkgPath)
diff --git a/gopls/internal/lsp/source/view.go b/gopls/internal/lsp/source/view.go
index 9fe6599..de69852 100644
--- a/gopls/internal/lsp/source/view.go
+++ b/gopls/internal/lsp/source/view.go
@@ -170,7 +170,8 @@
ReverseDependencies(ctx context.Context, id PackageID, transitive bool) (map[PackageID]*Metadata, error)
// WorkspaceMetadata returns a new, unordered slice containing
- // metadata for all packages in the workspace.
+ // metadata for all ordinary and test packages (but not
+ // intermediate test variants) in the workspace.
WorkspaceMetadata(ctx context.Context) ([]*Metadata, error)
// AllMetadata returns a new unordered array of metadata for all packages in the workspace.
diff --git a/gopls/internal/regtest/misc/references_test.go b/gopls/internal/regtest/misc/references_test.go
index e1f5d8e..a207333 100644
--- a/gopls/internal/regtest/misc/references_test.go
+++ b/gopls/internal/regtest/misc/references_test.go
@@ -13,7 +13,9 @@
"github.com/google/go-cmp/cmp"
"golang.org/x/tools/gopls/internal/lsp/protocol"
+ "golang.org/x/tools/gopls/internal/lsp/regtest"
. "golang.org/x/tools/gopls/internal/lsp/regtest"
+ "golang.org/x/tools/internal/testenv"
)
func TestStdlibReferences(t *testing.T) {
@@ -250,16 +252,6 @@
Run(t, files, func(t *testing.T, env *Env) {
env.OpenFile("foo/foo.go")
- // Helper to map locations relative file paths.
- fileLocations := func(locs []protocol.Location) []string {
- var got []string
- for _, loc := range locs {
- got = append(got, env.Sandbox.Workdir.URIToPath(loc.URI))
- }
- sort.Strings(got)
- return got
- }
-
refTests := []struct {
re string
wantRefs []string
@@ -280,7 +272,7 @@
loc := env.RegexpSearch("foo/foo.go", test.re)
refs := env.References(loc)
- got := fileLocations(refs)
+ got := fileLocations(env, refs)
if diff := cmp.Diff(test.wantRefs, got); diff != "" {
t.Errorf("References(%q) returned unexpected diff (-want +got):\n%s", test.re, diff)
}
@@ -304,7 +296,7 @@
loc := env.RegexpSearch("foo/foo.go", test.re)
impls := env.Implementations(loc)
- got := fileLocations(impls)
+ got := fileLocations(env, impls)
if diff := cmp.Diff(test.wantImpls, got); diff != "" {
t.Errorf("Implementations(%q) returned unexpected diff (-want +got):\n%s", test.re, diff)
}
@@ -397,3 +389,80 @@
checkVendor(env.Implementations(refLoc), false)
})
}
+
+// This test can't be expressed as a marker test because the marker
+// test framework opens all files (which is a bit of a hack), creating
+// a <command-line-arguments> package for packages that otherwise
+// wouldn't be found from the go.work file.
+func TestReferencesFromWorkspacePackages59674(t *testing.T) {
+ testenv.NeedsGo1Point(t, 18) // for go.work support
+ const src = `
+-- a/go.mod --
+module example.com/a
+go 1.12
+
+-- b/go.mod --
+module example.com/b
+go 1.12
+
+-- c/go.mod --
+module example.com/c
+go 1.12
+
+-- lib/go.mod --
+module example.com/lib
+go 1.12
+
+-- go.work --
+use ./a
+use ./b
+// don't use ./c
+use ./lib
+
+-- a/a.go --
+package a
+
+import "example.com/lib"
+
+var _ = lib.F // query here
+
+-- b/b.go --
+package b
+
+import "example.com/lib"
+
+var _ = lib.F // also found by references
+
+-- c/c.go --
+package c
+
+import "example.com/lib"
+
+var _ = lib.F // this reference should not be reported
+
+-- lib/lib.go --
+package lib
+
+func F() {} // declaration
+`
+ Run(t, src, func(t *testing.T, env *Env) {
+ env.OpenFile("a/a.go")
+ refLoc := env.RegexpSearch("a/a.go", "F")
+ got := fileLocations(env, env.References(refLoc))
+ want := []string{"a/a.go", "b/b.go", "lib/lib.go"}
+ if diff := cmp.Diff(want, got); diff != "" {
+ t.Errorf("incorrect References (-want +got):\n%s", diff)
+ }
+ })
+}
+
+// fileLocations returns a new sorted array of the relative
+// file name of each location. Duplicates are not removed.
+func fileLocations(env *regtest.Env, locs []protocol.Location) []string {
+ got := make([]string, 0, len(locs))
+ for _, loc := range locs {
+ got = append(got, env.Sandbox.Workdir.URIToPath(loc.URI))
+ }
+ sort.Strings(got)
+ return got
+}