mirror of
https://github.com/golang/go
synced 2024-11-18 17:54:57 -07:00
eaaaedc6af
Our logic to generate documentation links did not account for embedded fields and methods. The types.Info.ObjectOf an embedded field returns the *types.Var created for the field, not its types.TypeName, so we have to navigate back to the actual definition of the field. This requires traversing through all of the named types in the top-level type. Fixes golang/go#40294 Change-Id: Ia6573aebe66b7f60e2d6861a381cd7b07e7d7eaa Reviewed-on: https://go-review.googlesource.com/c/tools/+/244178 Run-TryBot: Rebecca Stambler <rstambler@golang.org> TryBot-Result: Gobot Gobot <gobot@golang.org> Reviewed-by: Robert Findley <rfindley@google.com>
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
|
|
}
|