mirror of
https://github.com/golang/go
synced 2024-11-18 14:14:46 -07:00
ecd3fc4348
We use file identities pervasively throughout gopls. Prior to this change, the identity is the modification date of an unopened file, or the hash of an opened file. That means that opening a file changes its identity, which causes unnecessary churn in the cache. Unfortunately, there isn't an easy way to fix this. Changing the cache key to something else, such as the modification time, means that we won't unify cache entries if a change is made and then undone. The approach here is to read files eagerly in GetFile, so that we know their hashes immediately. That resolves the churn, but means that we do a ton of file IO at startup. Incidental changes: Remove the FileSystem interface; there was only one implementation and it added a fair amount of cruft. We have many other places that assume os.Stat and such work. Add direct accessors to FileHandle for URI, Kind, and Version. Most uses of (FileHandle).Identity were for stuff that we derive solely from the URI, and this helped me disentangle them. It is a *ton* of churn, though. I can revert it if you want. Change-Id: Ia2133bc527f71daf81c9d674951726a232ca5bc9 Reviewed-on: https://go-review.googlesource.com/c/tools/+/237037 Run-TryBot: Heschi Kreinick <heschi@google.com> TryBot-Result: Gobot Gobot <gobot@golang.org> Reviewed-by: Rebecca Stambler <rstambler@golang.org>
372 lines
9.2 KiB
Go
372 lines
9.2 KiB
Go
// Copyright 2019 The Go Authors. All rights reserved.
|
|
// Use of this source code is governed by a BSD-style
|
|
// license that can be found in the LICENSE file.
|
|
|
|
package source
|
|
|
|
import (
|
|
"context"
|
|
"errors"
|
|
"fmt"
|
|
"go/ast"
|
|
"go/token"
|
|
"go/types"
|
|
|
|
"golang.org/x/tools/internal/event"
|
|
"golang.org/x/tools/internal/lsp/protocol"
|
|
)
|
|
|
|
func Implementation(ctx context.Context, s Snapshot, f FileHandle, pp protocol.Position) ([]protocol.Location, error) {
|
|
ctx, done := event.Start(ctx, "source.Implementation")
|
|
defer done()
|
|
|
|
impls, err := implementations(ctx, s, f, pp)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
var locations []protocol.Location
|
|
for _, impl := range impls {
|
|
if impl.pkg == nil || len(impl.pkg.CompiledGoFiles()) == 0 {
|
|
continue
|
|
}
|
|
rng, err := objToMappedRange(s.View(), impl.pkg, impl.obj)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
pr, err := rng.Range()
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
locations = append(locations, protocol.Location{
|
|
URI: protocol.URIFromSpanURI(rng.URI()),
|
|
Range: pr,
|
|
})
|
|
}
|
|
return locations, nil
|
|
}
|
|
|
|
var ErrNotAType = errors.New("not a type name or method")
|
|
|
|
// implementations returns the concrete implementations of the specified
|
|
// interface, or the interfaces implemented by the specified concrete type.
|
|
func implementations(ctx context.Context, s Snapshot, f FileHandle, pp protocol.Position) ([]qualifiedObject, error) {
|
|
var (
|
|
impls []qualifiedObject
|
|
seen = make(map[token.Position]bool)
|
|
fset = s.View().Session().Cache().FileSet()
|
|
)
|
|
|
|
qos, err := qualifiedObjsAtProtocolPos(ctx, s, f, pp)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
for _, qo := range qos {
|
|
var (
|
|
queryType types.Type
|
|
queryMethod *types.Func
|
|
)
|
|
|
|
switch obj := qo.obj.(type) {
|
|
case *types.Func:
|
|
queryMethod = obj
|
|
if recv := obj.Type().(*types.Signature).Recv(); recv != nil {
|
|
queryType = ensurePointer(recv.Type())
|
|
}
|
|
case *types.TypeName:
|
|
queryType = ensurePointer(obj.Type())
|
|
}
|
|
|
|
if queryType == nil {
|
|
return nil, ErrNotAType
|
|
}
|
|
|
|
if types.NewMethodSet(queryType).Len() == 0 {
|
|
return nil, nil
|
|
}
|
|
|
|
// Find all named types, even local types (which can have methods
|
|
// due to promotion).
|
|
var (
|
|
allNamed []*types.Named
|
|
pkgs = make(map[*types.Package]Package)
|
|
)
|
|
knownPkgs, err := s.KnownPackages(ctx)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
for _, ph := range knownPkgs {
|
|
pkg, err := ph.Check(ctx)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
pkgs[pkg.GetTypes()] = pkg
|
|
info := pkg.GetTypesInfo()
|
|
for _, obj := range info.Defs {
|
|
obj, ok := obj.(*types.TypeName)
|
|
// We ignore aliases 'type M = N' to avoid duplicate reporting
|
|
// of the Named type N.
|
|
if !ok || obj.IsAlias() {
|
|
continue
|
|
}
|
|
if named, ok := obj.Type().(*types.Named); ok {
|
|
allNamed = append(allNamed, named)
|
|
}
|
|
}
|
|
}
|
|
|
|
// Find all the named types that match our query.
|
|
for _, named := range allNamed {
|
|
var (
|
|
candObj types.Object = named.Obj()
|
|
candType = ensurePointer(named)
|
|
)
|
|
|
|
if !concreteImplementsIntf(candType, queryType) {
|
|
continue
|
|
}
|
|
|
|
ms := types.NewMethodSet(candType)
|
|
if ms.Len() == 0 {
|
|
// Skip empty interfaces.
|
|
continue
|
|
}
|
|
|
|
// If client queried a method, look up corresponding candType method.
|
|
if queryMethod != nil {
|
|
sel := ms.Lookup(queryMethod.Pkg(), queryMethod.Name())
|
|
if sel == nil {
|
|
continue
|
|
}
|
|
candObj = sel.Obj()
|
|
}
|
|
|
|
pos := fset.Position(candObj.Pos())
|
|
if candObj == queryMethod || seen[pos] {
|
|
continue
|
|
}
|
|
|
|
seen[pos] = true
|
|
|
|
impls = append(impls, qualifiedObject{
|
|
obj: candObj,
|
|
pkg: pkgs[candObj.Pkg()],
|
|
})
|
|
}
|
|
}
|
|
|
|
return impls, nil
|
|
}
|
|
|
|
// concreteImplementsIntf returns true if a is an interface type implemented by
|
|
// concrete type b, or vice versa.
|
|
func concreteImplementsIntf(a, b types.Type) bool {
|
|
aIsIntf, bIsIntf := isInterface(a), isInterface(b)
|
|
|
|
// Make sure exactly one is an interface type.
|
|
if aIsIntf == bIsIntf {
|
|
return false
|
|
}
|
|
|
|
// Rearrange if needed so "a" is the concrete type.
|
|
if aIsIntf {
|
|
a, b = b, a
|
|
}
|
|
|
|
return types.AssignableTo(a, b)
|
|
}
|
|
|
|
// ensurePointer wraps T in a *types.Pointer if T is a named, non-interface
|
|
// type. This is useful to make sure you consider a named type's full method
|
|
// set.
|
|
func ensurePointer(T types.Type) types.Type {
|
|
if _, ok := T.(*types.Named); ok && !isInterface(T) {
|
|
return types.NewPointer(T)
|
|
}
|
|
|
|
return T
|
|
}
|
|
|
|
type qualifiedObject struct {
|
|
obj types.Object
|
|
|
|
// pkg is the Package that contains obj's definition.
|
|
pkg Package
|
|
|
|
// node is the *ast.Ident or *ast.ImportSpec we followed to find obj, if any.
|
|
node ast.Node
|
|
|
|
// sourcePkg is the Package that contains node, if any.
|
|
sourcePkg Package
|
|
}
|
|
|
|
var errBuiltin = errors.New("builtin object")
|
|
|
|
// qualifiedObjsAtProtocolPos returns info for all the type.Objects
|
|
// referenced at the given position. An object will be returned for
|
|
// every package that the file belongs to.
|
|
func qualifiedObjsAtProtocolPos(ctx context.Context, s Snapshot, fh FileHandle, pp protocol.Position) ([]qualifiedObject, error) {
|
|
phs, err := s.PackageHandles(ctx, fh)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
// Check all the packages that the file belongs to.
|
|
var qualifiedObjs []qualifiedObject
|
|
for _, ph := range phs {
|
|
searchpkg, err := ph.Check(ctx)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
astFile, pos, err := getASTFile(searchpkg, fh, pp)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
path := pathEnclosingObjNode(astFile, pos)
|
|
if path == nil {
|
|
return nil, ErrNoIdentFound
|
|
}
|
|
var objs []types.Object
|
|
switch leaf := path[0].(type) {
|
|
case *ast.Ident:
|
|
// If leaf represents an implicit type switch object or the type
|
|
// switch "assign" variable, expand to all of the type switch's
|
|
// implicit objects.
|
|
if implicits := typeSwitchImplicits(searchpkg, path); len(implicits) > 0 {
|
|
objs = append(objs, implicits...)
|
|
} else {
|
|
obj := searchpkg.GetTypesInfo().ObjectOf(leaf)
|
|
if obj == nil {
|
|
return nil, fmt.Errorf("no object for %q", leaf.Name)
|
|
}
|
|
objs = append(objs, obj)
|
|
}
|
|
case *ast.ImportSpec:
|
|
// Look up the implicit *types.PkgName.
|
|
obj := searchpkg.GetTypesInfo().Implicits[leaf]
|
|
if obj == nil {
|
|
return nil, fmt.Errorf("no object for import %q", importPath(leaf))
|
|
}
|
|
objs = append(objs, obj)
|
|
}
|
|
// Get all of the transitive dependencies of the search package.
|
|
pkgs := make(map[*types.Package]Package)
|
|
var addPkg func(pkg Package)
|
|
addPkg = func(pkg Package) {
|
|
pkgs[pkg.GetTypes()] = pkg
|
|
for _, imp := range pkg.Imports() {
|
|
if _, ok := pkgs[imp.GetTypes()]; !ok {
|
|
addPkg(imp)
|
|
}
|
|
}
|
|
}
|
|
addPkg(searchpkg)
|
|
for _, obj := range objs {
|
|
if obj.Parent() == types.Universe {
|
|
return nil, fmt.Errorf("%w %q", errBuiltin, obj.Name())
|
|
}
|
|
pkg, ok := pkgs[obj.Pkg()]
|
|
if !ok {
|
|
event.Error(ctx, fmt.Sprintf("no package for obj %s: %v", obj, obj.Pkg()), err)
|
|
continue
|
|
}
|
|
qualifiedObjs = append(qualifiedObjs, qualifiedObject{
|
|
obj: obj,
|
|
pkg: pkg,
|
|
sourcePkg: searchpkg,
|
|
node: path[0],
|
|
})
|
|
}
|
|
}
|
|
// Return an error if no objects were found since callers will assume that
|
|
// the slice has at least 1 element.
|
|
if len(qualifiedObjs) == 0 {
|
|
return nil, fmt.Errorf("no object found")
|
|
}
|
|
return qualifiedObjs, nil
|
|
}
|
|
|
|
func getASTFile(pkg Package, f FileHandle, pos protocol.Position) (*ast.File, token.Pos, error) {
|
|
pgh, err := pkg.File(f.URI())
|
|
if err != nil {
|
|
return nil, 0, err
|
|
}
|
|
file, _, m, _, err := pgh.Cached()
|
|
if err != nil {
|
|
return nil, 0, err
|
|
}
|
|
spn, err := m.PointSpan(pos)
|
|
if err != nil {
|
|
return nil, 0, err
|
|
}
|
|
rng, err := spn.Range(m.Converter)
|
|
if err != nil {
|
|
return nil, 0, err
|
|
}
|
|
return file, rng.Start, nil
|
|
}
|
|
|
|
// pathEnclosingObjNode returns the AST path to the object-defining
|
|
// node associated with pos. "Object-defining" means either an
|
|
// *ast.Ident mapped directly to a types.Object or an ast.Node mapped
|
|
// implicitly to a types.Object.
|
|
func pathEnclosingObjNode(f *ast.File, pos token.Pos) []ast.Node {
|
|
var (
|
|
path []ast.Node
|
|
found bool
|
|
)
|
|
|
|
ast.Inspect(f, func(n ast.Node) bool {
|
|
if found {
|
|
return false
|
|
}
|
|
|
|
if n == nil {
|
|
path = path[:len(path)-1]
|
|
return false
|
|
}
|
|
|
|
path = append(path, n)
|
|
|
|
switch n := n.(type) {
|
|
case *ast.Ident:
|
|
// Include the position directly after identifier. This handles
|
|
// the common case where the cursor is right after the
|
|
// identifier the user is currently typing. Previously we
|
|
// handled this by calling astutil.PathEnclosingInterval twice,
|
|
// once for "pos" and once for "pos-1".
|
|
found = n.Pos() <= pos && pos <= n.End()
|
|
case *ast.ImportSpec:
|
|
if n.Path.Pos() <= pos && pos < n.Path.End() {
|
|
found = true
|
|
// If import spec has a name, add name to path even though
|
|
// position isn't in the name.
|
|
if n.Name != nil {
|
|
path = append(path, n.Name)
|
|
}
|
|
}
|
|
case *ast.StarExpr:
|
|
// Follow star expressions to the inner identifier.
|
|
if pos == n.Star {
|
|
pos = n.X.Pos()
|
|
}
|
|
case *ast.SelectorExpr:
|
|
// If pos is on the ".", move it into the selector.
|
|
if pos == n.X.End() {
|
|
pos = n.Sel.Pos()
|
|
}
|
|
}
|
|
|
|
return !found
|
|
})
|
|
|
|
if len(path) == 0 {
|
|
return nil
|
|
}
|
|
|
|
// Reverse path so leaf is first element.
|
|
for i := 0; i < len(path)/2; i++ {
|
|
path[i], path[len(path)-1-i] = path[len(path)-1-i], path[i]
|
|
}
|
|
|
|
return path
|
|
}
|