go/types/objectpath: a stable naming scheme for types.Object
Type-checker objects are canonical, so they are usually identified by their address in memory (a pointer), but a pointer has meaning only within one address space. By contrast, objectpath names allow the identity of a logical object to be sent from one program to another, establishing a correspondence between types.Object variables that are distinct but logically equivalent. This package was developed for Google's internal fork of guru. It is needed for lemma support in the analysis API; see docs.google.com/document/d/1-azPLXaLgTCKeKDNg0HVMq2ovMlD-e7n1ZHzZVzOlJk Change-Id: I9899ce14d57909858a68f84e90d58a039f2bb7a0 Reviewed-on: https://go-review.googlesource.com/135675 Reviewed-by: Robert Griesemer <gri@golang.org>
This commit is contained in:
parent
84988e2dba
commit
73ed285d4c
|
@ -0,0 +1,523 @@
|
||||||
|
// Copyright 2018 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 objectpath defines a naming scheme for types.Objects
|
||||||
|
// (that is, named entities in Go programs) relative to their enclosing
|
||||||
|
// package.
|
||||||
|
//
|
||||||
|
// Type-checker objects are canonical, so they are usually identified by
|
||||||
|
// their address in memory (a pointer), but a pointer has meaning only
|
||||||
|
// within one address space. By contrast, objectpath names allow the
|
||||||
|
// identity of an object to be sent from one program to another,
|
||||||
|
// establishing a correspondence between types.Object variables that are
|
||||||
|
// distinct but logically equivalent.
|
||||||
|
//
|
||||||
|
// A single object may have multiple paths. In this example,
|
||||||
|
// type A struct{ X int }
|
||||||
|
// type B A
|
||||||
|
// the field X has two paths due to its membership of both A and B.
|
||||||
|
// The For(obj) function always returns one of these paths, arbitrarily
|
||||||
|
// but consistently.
|
||||||
|
package objectpath
|
||||||
|
|
||||||
|
import (
|
||||||
|
"fmt"
|
||||||
|
"strconv"
|
||||||
|
"strings"
|
||||||
|
|
||||||
|
"go/types"
|
||||||
|
)
|
||||||
|
|
||||||
|
// A Path is an opaque name that identifies a types.Object
|
||||||
|
// relative to its package. Conceptually, the name consists of a
|
||||||
|
// sequence of destructuring operations applied to the package scope
|
||||||
|
// to obtain the original object.
|
||||||
|
// The name does not include the package itself.
|
||||||
|
type Path string
|
||||||
|
|
||||||
|
// Encoding
|
||||||
|
//
|
||||||
|
// An object path is a textual and (with training) human-readable encoding
|
||||||
|
// of a sequence of destructuring operators, starting from a types.Package.
|
||||||
|
// The sequences represent a path through the package/object/type graph.
|
||||||
|
// We classify these operators by their type:
|
||||||
|
//
|
||||||
|
// PO package->object Package.Scope.Lookup
|
||||||
|
// OT object->type Object.Type
|
||||||
|
// TT type->type Type.{Elem,Key,Params,Results,Underlying} [EKPRU]
|
||||||
|
// TO type->object Type.{At,Field,Method,Obj} [AFMO]
|
||||||
|
//
|
||||||
|
// All valid paths start with a package and end at an object
|
||||||
|
// and thus may be defined by the regular language:
|
||||||
|
//
|
||||||
|
// objectpath = PO (OT TT* TO)*
|
||||||
|
//
|
||||||
|
// The concrete encoding follows directly:
|
||||||
|
// - The only PO operator is Package.Scope.Lookup, which requires an identifier.
|
||||||
|
// - The only OT operator is Object.Type,
|
||||||
|
// which we encode as '.' because dot cannot appear in an identifier.
|
||||||
|
// - The TT operators are encoded as [EKPRU].
|
||||||
|
// - The OT operators are encoded as [AFMO];
|
||||||
|
// three of these (At,Field,Method) require an integer operand,
|
||||||
|
// which is encoded as a string of decimal digits.
|
||||||
|
// These indices are stable across different representations
|
||||||
|
// of the same package, even source and export data.
|
||||||
|
//
|
||||||
|
// In the example below,
|
||||||
|
//
|
||||||
|
// package p
|
||||||
|
//
|
||||||
|
// type T interface {
|
||||||
|
// f() (a string, b struct{ X int })
|
||||||
|
// }
|
||||||
|
//
|
||||||
|
// field X has the path "T.UM0.RA1.F0",
|
||||||
|
// representing the following sequence of operations:
|
||||||
|
//
|
||||||
|
// p.Lookup("T") T
|
||||||
|
// .Type().Underlying().Method(0). f
|
||||||
|
// .Type().Results().At(1) b
|
||||||
|
// .Type().Field(0) X
|
||||||
|
//
|
||||||
|
// The encoding is not maximally compact---every R or P is
|
||||||
|
// followed by an A, for example---but this simplifies the
|
||||||
|
// encoder and decoder.
|
||||||
|
//
|
||||||
|
const (
|
||||||
|
// object->type operators
|
||||||
|
opType = '.' // .Type() (Object)
|
||||||
|
|
||||||
|
// type->type operators
|
||||||
|
opElem = 'E' // .Elem() (Pointer, Slice, Array, Chan, Map)
|
||||||
|
opKey = 'K' // .Key() (Map)
|
||||||
|
opParams = 'P' // .Params() (Signature)
|
||||||
|
opResults = 'R' // .Results() (Signature)
|
||||||
|
opUnderlying = 'U' // .Underlying() (Named)
|
||||||
|
|
||||||
|
// type->object operators
|
||||||
|
opAt = 'A' // .At(i) (Tuple)
|
||||||
|
opField = 'F' // .Field(i) (Struct)
|
||||||
|
opMethod = 'M' // .Method(i) (Named or Interface; not Struct: "promoted" names are ignored)
|
||||||
|
opObj = 'O' // .Obj() (Named)
|
||||||
|
)
|
||||||
|
|
||||||
|
// The For function returns the path to an object relative to its package,
|
||||||
|
// or an error if the object is not accessible from the package's Scope.
|
||||||
|
//
|
||||||
|
// The For function guarantees to return a path only for the following objects:
|
||||||
|
// - package-level types
|
||||||
|
// - exported package-level non-types
|
||||||
|
// - methods
|
||||||
|
// - parameter and result variables
|
||||||
|
// - struct fields
|
||||||
|
// These objects are sufficient to define the API of their package.
|
||||||
|
// The objects described by a package's export data are drawn from this set.
|
||||||
|
//
|
||||||
|
// For does not return a path for predeclared names, imported package
|
||||||
|
// names, local names, and unexported package-level names (except
|
||||||
|
// types).
|
||||||
|
//
|
||||||
|
// Example: given this definition,
|
||||||
|
//
|
||||||
|
// package p
|
||||||
|
//
|
||||||
|
// type T interface {
|
||||||
|
// f() (a string, b struct{ X int })
|
||||||
|
// }
|
||||||
|
//
|
||||||
|
// For(X) would return a path that denotes the following sequence of operations:
|
||||||
|
//
|
||||||
|
// p.Scope().Lookup("T") (TypeName T)
|
||||||
|
// .Type().Underlying().Method(0). (method Func f)
|
||||||
|
// .Type().Results().At(1) (field Var b)
|
||||||
|
// .Type().Field(0) (field Var X)
|
||||||
|
//
|
||||||
|
// where p is the package (*types.Package) to which X belongs.
|
||||||
|
func For(obj types.Object) (Path, error) {
|
||||||
|
pkg := obj.Pkg()
|
||||||
|
|
||||||
|
// This table lists the cases of interest.
|
||||||
|
//
|
||||||
|
// Object Action
|
||||||
|
// ------ ------
|
||||||
|
// nil reject
|
||||||
|
// builtin reject
|
||||||
|
// pkgname reject
|
||||||
|
// label reject
|
||||||
|
// var
|
||||||
|
// package-level accept
|
||||||
|
// func param/result accept
|
||||||
|
// local reject
|
||||||
|
// struct field accept
|
||||||
|
// const
|
||||||
|
// package-level accept
|
||||||
|
// local reject
|
||||||
|
// func
|
||||||
|
// package-level accept
|
||||||
|
// init functions reject
|
||||||
|
// concrete method accept
|
||||||
|
// interface method accept
|
||||||
|
// type
|
||||||
|
// package-level accept
|
||||||
|
// local reject
|
||||||
|
//
|
||||||
|
// The only accessible package-level objects are members of pkg itself.
|
||||||
|
//
|
||||||
|
// The cases are handled in four steps:
|
||||||
|
//
|
||||||
|
// 1. reject nil and builtin
|
||||||
|
// 2. accept package-level objects
|
||||||
|
// 3. reject obviously invalid objects
|
||||||
|
// 4. search the API for the path to the param/result/field/method.
|
||||||
|
|
||||||
|
// 1. reference to nil or builtin?
|
||||||
|
if pkg == nil {
|
||||||
|
return "", fmt.Errorf("predeclared %s has no path", obj)
|
||||||
|
}
|
||||||
|
scope := pkg.Scope()
|
||||||
|
|
||||||
|
// 2. package-level object?
|
||||||
|
if scope.Lookup(obj.Name()) == obj {
|
||||||
|
// Only exported objects (and non-exported types) have a path.
|
||||||
|
// Non-exported types may be referenced by other objects.
|
||||||
|
if _, ok := obj.(*types.TypeName); !ok && !obj.Exported() {
|
||||||
|
return "", fmt.Errorf("no path for non-exported %v", obj)
|
||||||
|
}
|
||||||
|
return Path(obj.Name()), nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// 3. Not a package-level object.
|
||||||
|
// Reject obviously non-viable cases.
|
||||||
|
switch obj := obj.(type) {
|
||||||
|
case *types.Const, // Only package-level constants have a path.
|
||||||
|
*types.TypeName, // Only package-level types have a path.
|
||||||
|
*types.Label, // Labels are function-local.
|
||||||
|
*types.PkgName: // PkgNames are file-local.
|
||||||
|
return "", fmt.Errorf("no path for %v", obj)
|
||||||
|
|
||||||
|
case *types.Var:
|
||||||
|
// Could be:
|
||||||
|
// - a field (obj.IsField())
|
||||||
|
// - a func parameter or result
|
||||||
|
// - a local var.
|
||||||
|
// Sadly there is no way to distinguish
|
||||||
|
// a param/result from a local
|
||||||
|
// so we must proceed to the find.
|
||||||
|
|
||||||
|
case *types.Func:
|
||||||
|
// A func, if not package-level, must be a method.
|
||||||
|
if recv := obj.Type().(*types.Signature).Recv(); recv == nil {
|
||||||
|
return "", fmt.Errorf("func is not a method: %v", obj)
|
||||||
|
}
|
||||||
|
// TODO(adonovan): opt: if the method is concrete,
|
||||||
|
// do a specialized version of the rest of this function so
|
||||||
|
// that it's O(1) not O(|scope|). Basically 'find' is needed
|
||||||
|
// only for struct fields and interface methods.
|
||||||
|
|
||||||
|
default:
|
||||||
|
panic(obj)
|
||||||
|
}
|
||||||
|
|
||||||
|
// 4. Search the API for the path to the var (field/param/result) or method.
|
||||||
|
|
||||||
|
// First inspect package-level named types.
|
||||||
|
// In the presence of path aliases, these give
|
||||||
|
// the best paths because non-types may
|
||||||
|
// refer to types, but not the reverse.
|
||||||
|
empty := make([]byte, 0, 48) // initial space
|
||||||
|
for _, name := range scope.Names() {
|
||||||
|
o := scope.Lookup(name)
|
||||||
|
tname, ok := o.(*types.TypeName)
|
||||||
|
if !ok {
|
||||||
|
continue // handle non-types in second pass
|
||||||
|
}
|
||||||
|
|
||||||
|
path := append(empty, name...)
|
||||||
|
path = append(path, opType)
|
||||||
|
|
||||||
|
T := o.Type()
|
||||||
|
|
||||||
|
if tname.IsAlias() {
|
||||||
|
// type alias
|
||||||
|
if r := find(obj, T, path); r != nil {
|
||||||
|
return Path(r), nil
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
// defined (named) type
|
||||||
|
if r := find(obj, T.Underlying(), append(path, opUnderlying)); r != nil {
|
||||||
|
return Path(r), nil
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Then inspect everything else:
|
||||||
|
// non-types, and declared methods of defined types.
|
||||||
|
for _, name := range scope.Names() {
|
||||||
|
o := scope.Lookup(name)
|
||||||
|
path := append(empty, name...)
|
||||||
|
if _, ok := o.(*types.TypeName); !ok {
|
||||||
|
if o.Exported() {
|
||||||
|
// exported non-type (const, var, func)
|
||||||
|
if r := find(obj, o.Type(), append(path, opType)); r != nil {
|
||||||
|
return Path(r), nil
|
||||||
|
}
|
||||||
|
}
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
|
||||||
|
// Inspect declared methods of defined types.
|
||||||
|
if T, ok := o.Type().(*types.Named); ok {
|
||||||
|
path = append(path, opType)
|
||||||
|
for i := 0; i < T.NumMethods(); i++ {
|
||||||
|
m := T.Method(i)
|
||||||
|
path2 := appendOpArg(path, opMethod, i)
|
||||||
|
if m == obj {
|
||||||
|
return Path(path2), nil // found declared method
|
||||||
|
}
|
||||||
|
if r := find(obj, m.Type(), append(path2, opType)); r != nil {
|
||||||
|
return Path(r), nil
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return "", fmt.Errorf("can't find path for %v in %s", obj, pkg.Path())
|
||||||
|
}
|
||||||
|
|
||||||
|
func appendOpArg(path []byte, op byte, arg int) []byte {
|
||||||
|
path = append(path, op)
|
||||||
|
path = strconv.AppendInt(path, int64(arg), 10)
|
||||||
|
return path
|
||||||
|
}
|
||||||
|
|
||||||
|
// find finds obj within type T, returning the path to it, or nil if not found.
|
||||||
|
func find(obj types.Object, T types.Type, path []byte) []byte {
|
||||||
|
switch T := T.(type) {
|
||||||
|
case *types.Basic, *types.Named:
|
||||||
|
// Named types belonging to pkg were handled already,
|
||||||
|
// so T must belong to another package. No path.
|
||||||
|
return nil
|
||||||
|
case *types.Pointer:
|
||||||
|
return find(obj, T.Elem(), append(path, opElem))
|
||||||
|
case *types.Slice:
|
||||||
|
return find(obj, T.Elem(), append(path, opElem))
|
||||||
|
case *types.Array:
|
||||||
|
return find(obj, T.Elem(), append(path, opElem))
|
||||||
|
case *types.Chan:
|
||||||
|
return find(obj, T.Elem(), append(path, opElem))
|
||||||
|
case *types.Map:
|
||||||
|
if r := find(obj, T.Key(), append(path, opKey)); r != nil {
|
||||||
|
return r
|
||||||
|
}
|
||||||
|
return find(obj, T.Elem(), append(path, opElem))
|
||||||
|
case *types.Signature:
|
||||||
|
if r := find(obj, T.Params(), append(path, opParams)); r != nil {
|
||||||
|
return r
|
||||||
|
}
|
||||||
|
return find(obj, T.Results(), append(path, opResults))
|
||||||
|
case *types.Struct:
|
||||||
|
for i := 0; i < T.NumFields(); i++ {
|
||||||
|
f := T.Field(i)
|
||||||
|
path2 := appendOpArg(path, opField, i)
|
||||||
|
if f == obj {
|
||||||
|
return path2 // found field var
|
||||||
|
}
|
||||||
|
if r := find(obj, f.Type(), append(path2, opType)); r != nil {
|
||||||
|
return r
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return nil
|
||||||
|
case *types.Tuple:
|
||||||
|
for i := 0; i < T.Len(); i++ {
|
||||||
|
v := T.At(i)
|
||||||
|
path2 := appendOpArg(path, opAt, i)
|
||||||
|
if v == obj {
|
||||||
|
return path2 // found param/result var
|
||||||
|
}
|
||||||
|
if r := find(obj, v.Type(), append(path2, opType)); r != nil {
|
||||||
|
return r
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return nil
|
||||||
|
case *types.Interface:
|
||||||
|
for i := 0; i < T.NumMethods(); i++ {
|
||||||
|
m := T.Method(i)
|
||||||
|
path2 := appendOpArg(path, opMethod, i)
|
||||||
|
if m == obj {
|
||||||
|
return path2 // found interface method
|
||||||
|
}
|
||||||
|
if r := find(obj, m.Type(), append(path2, opType)); r != nil {
|
||||||
|
return r
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
panic(T)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Object returns the object denoted by path p within the package pkg.
|
||||||
|
func Object(pkg *types.Package, p Path) (types.Object, error) {
|
||||||
|
if p == "" {
|
||||||
|
return nil, fmt.Errorf("empty path")
|
||||||
|
}
|
||||||
|
|
||||||
|
pathstr := string(p)
|
||||||
|
var pkgobj, suffix string
|
||||||
|
if dot := strings.IndexByte(pathstr, opType); dot < 0 {
|
||||||
|
pkgobj = pathstr
|
||||||
|
} else {
|
||||||
|
pkgobj = pathstr[:dot]
|
||||||
|
suffix = pathstr[dot:] // suffix starts with "."
|
||||||
|
}
|
||||||
|
|
||||||
|
obj := pkg.Scope().Lookup(pkgobj)
|
||||||
|
if obj == nil {
|
||||||
|
return nil, fmt.Errorf("package %s does not contain %q", pkg.Path(), pkgobj)
|
||||||
|
}
|
||||||
|
|
||||||
|
// abtraction of *types.{Pointer,Slice,Array,Chan,Map}
|
||||||
|
type hasElem interface {
|
||||||
|
Elem() types.Type
|
||||||
|
}
|
||||||
|
// abstraction of *types.{Interface,Named}
|
||||||
|
type hasMethods interface {
|
||||||
|
Method(int) *types.Func
|
||||||
|
NumMethods() int
|
||||||
|
}
|
||||||
|
|
||||||
|
// The loop state is the pair (t, obj),
|
||||||
|
// exactly one of which is non-nil, initially obj.
|
||||||
|
// All suffixes start with '.' (the only object->type operation),
|
||||||
|
// followed by optional type->type operations,
|
||||||
|
// then a type->object operation.
|
||||||
|
// The cycle then repeats.
|
||||||
|
var t types.Type
|
||||||
|
for suffix != "" {
|
||||||
|
code := suffix[0]
|
||||||
|
suffix = suffix[1:]
|
||||||
|
|
||||||
|
// Codes [AFM] have an integer operand.
|
||||||
|
var index int
|
||||||
|
switch code {
|
||||||
|
case opAt, opField, opMethod:
|
||||||
|
rest := strings.TrimLeft(suffix, "0123456789")
|
||||||
|
numerals := suffix[:len(suffix)-len(rest)]
|
||||||
|
suffix = rest
|
||||||
|
i, err := strconv.Atoi(numerals)
|
||||||
|
if err != nil {
|
||||||
|
return nil, fmt.Errorf("invalid path: bad numeric operand %q for code %q", numerals, code)
|
||||||
|
}
|
||||||
|
index = int(i)
|
||||||
|
case opObj:
|
||||||
|
// no operand
|
||||||
|
default:
|
||||||
|
// The suffix must end with a type->object operation.
|
||||||
|
if suffix == "" {
|
||||||
|
return nil, fmt.Errorf("invalid path: ends with %q, want [AFMO]", code)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if code == opType {
|
||||||
|
if t != nil {
|
||||||
|
return nil, fmt.Errorf("invalid path: unexpected %q in type context", opType)
|
||||||
|
}
|
||||||
|
t = obj.Type()
|
||||||
|
obj = nil
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
|
||||||
|
if t == nil {
|
||||||
|
return nil, fmt.Errorf("invalid path: code %q in object context", code)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Inv: t != nil, obj == nil
|
||||||
|
|
||||||
|
switch code {
|
||||||
|
case opElem:
|
||||||
|
hasElem, ok := t.(hasElem) // Pointer, Slice, Array, Chan, Map
|
||||||
|
if !ok {
|
||||||
|
return nil, fmt.Errorf("cannot apply %q to %s (got %T, want pointer, slice, array, chan or map)", code, t, t)
|
||||||
|
}
|
||||||
|
t = hasElem.Elem()
|
||||||
|
|
||||||
|
case opKey:
|
||||||
|
mapType, ok := t.(*types.Map)
|
||||||
|
if !ok {
|
||||||
|
return nil, fmt.Errorf("cannot apply %q to %s (got %T, want map)", code, t, t)
|
||||||
|
}
|
||||||
|
t = mapType.Key()
|
||||||
|
|
||||||
|
case opParams:
|
||||||
|
sig, ok := t.(*types.Signature)
|
||||||
|
if !ok {
|
||||||
|
return nil, fmt.Errorf("cannot apply %q to %s (got %T, want signature)", code, t, t)
|
||||||
|
}
|
||||||
|
t = sig.Params()
|
||||||
|
|
||||||
|
case opResults:
|
||||||
|
sig, ok := t.(*types.Signature)
|
||||||
|
if !ok {
|
||||||
|
return nil, fmt.Errorf("cannot apply %q to %s (got %T, want signature)", code, t, t)
|
||||||
|
}
|
||||||
|
t = sig.Results()
|
||||||
|
|
||||||
|
case opUnderlying:
|
||||||
|
named, ok := t.(*types.Named)
|
||||||
|
if !ok {
|
||||||
|
return nil, fmt.Errorf("cannot apply %q to %s (got %s, want named)", code, t, t)
|
||||||
|
}
|
||||||
|
t = named.Underlying()
|
||||||
|
|
||||||
|
case opAt:
|
||||||
|
tuple, ok := t.(*types.Tuple)
|
||||||
|
if !ok {
|
||||||
|
return nil, fmt.Errorf("cannot apply %q to %s (got %s, want tuple)", code, t, t)
|
||||||
|
}
|
||||||
|
if n := tuple.Len(); index >= n {
|
||||||
|
return nil, fmt.Errorf("tuple index %d out of range [0-%d)", index, n)
|
||||||
|
}
|
||||||
|
obj = tuple.At(index)
|
||||||
|
t = nil
|
||||||
|
|
||||||
|
case opField:
|
||||||
|
structType, ok := t.(*types.Struct)
|
||||||
|
if !ok {
|
||||||
|
return nil, fmt.Errorf("cannot apply %q to %s (got %T, want struct)", code, t, t)
|
||||||
|
}
|
||||||
|
if n := structType.NumFields(); index >= n {
|
||||||
|
return nil, fmt.Errorf("field index %d out of range [0-%d)", index, n)
|
||||||
|
}
|
||||||
|
obj = structType.Field(index)
|
||||||
|
t = nil
|
||||||
|
|
||||||
|
case opMethod:
|
||||||
|
hasMethods, ok := t.(hasMethods) // Interface or Named
|
||||||
|
if !ok {
|
||||||
|
return nil, fmt.Errorf("cannot apply %q to %s (got %s, want interface or named)", code, t, t)
|
||||||
|
}
|
||||||
|
if n := hasMethods.NumMethods(); index >= n {
|
||||||
|
return nil, fmt.Errorf("method index %d out of range [0-%d)", index, n)
|
||||||
|
}
|
||||||
|
obj = hasMethods.Method(index)
|
||||||
|
t = nil
|
||||||
|
|
||||||
|
case opObj:
|
||||||
|
named, ok := t.(*types.Named)
|
||||||
|
if !ok {
|
||||||
|
return nil, fmt.Errorf("cannot apply %q to %s (got %s, want named)", code, t, t)
|
||||||
|
}
|
||||||
|
obj = named.Obj()
|
||||||
|
t = nil
|
||||||
|
|
||||||
|
default:
|
||||||
|
return nil, fmt.Errorf("invalid path: unknown code %q", code)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if obj.Pkg() != pkg {
|
||||||
|
return nil, fmt.Errorf("path denotes %s, which belongs to a different package", obj)
|
||||||
|
}
|
||||||
|
|
||||||
|
return obj, nil // success
|
||||||
|
}
|
|
@ -0,0 +1,290 @@
|
||||||
|
// Copyright 2018 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 objectpath_test
|
||||||
|
|
||||||
|
import (
|
||||||
|
"bytes"
|
||||||
|
"go/ast"
|
||||||
|
"go/importer"
|
||||||
|
"go/parser"
|
||||||
|
"go/token"
|
||||||
|
"go/types"
|
||||||
|
"testing"
|
||||||
|
|
||||||
|
"golang.org/x/tools/go/buildutil"
|
||||||
|
"golang.org/x/tools/go/gcexportdata"
|
||||||
|
"golang.org/x/tools/go/loader"
|
||||||
|
"golang.org/x/tools/go/types/objectpath"
|
||||||
|
)
|
||||||
|
|
||||||
|
func TestPaths(t *testing.T) {
|
||||||
|
pkgs := map[string]map[string]string{
|
||||||
|
"b": {"b.go": `
|
||||||
|
package b
|
||||||
|
|
||||||
|
import "a"
|
||||||
|
|
||||||
|
const C = a.Int(0)
|
||||||
|
|
||||||
|
func F(a, b, c int, d a.T)
|
||||||
|
|
||||||
|
type T struct{ A int; b int; a.T }
|
||||||
|
|
||||||
|
func (T) M() *interface{ f() }
|
||||||
|
|
||||||
|
type U T
|
||||||
|
|
||||||
|
type A = struct{ x int }
|
||||||
|
|
||||||
|
var V []*a.T
|
||||||
|
|
||||||
|
type M map[struct{x int}]struct{y int}
|
||||||
|
|
||||||
|
func unexportedFunc()
|
||||||
|
type unexportedType struct{}
|
||||||
|
`},
|
||||||
|
"a": {"a.go": `
|
||||||
|
package a
|
||||||
|
|
||||||
|
type Int int
|
||||||
|
|
||||||
|
type T struct{x, y int}
|
||||||
|
|
||||||
|
`},
|
||||||
|
}
|
||||||
|
conf := loader.Config{Build: buildutil.FakeContext(pkgs)}
|
||||||
|
conf.Import("a")
|
||||||
|
conf.Import("b")
|
||||||
|
prog, err := conf.Load()
|
||||||
|
if err != nil {
|
||||||
|
t.Fatal(err)
|
||||||
|
}
|
||||||
|
a := prog.Imported["a"].Pkg
|
||||||
|
b := prog.Imported["b"].Pkg
|
||||||
|
|
||||||
|
// We test objectpath by enumerating a set of paths
|
||||||
|
// and ensuring that Path(pkg, Object(pkg, path)) == path.
|
||||||
|
//
|
||||||
|
// It might seem more natural to invert the test:
|
||||||
|
// identify a set of objects and for each one,
|
||||||
|
// ensure that Object(pkg, Path(pkg, obj)) == obj.
|
||||||
|
// However, for most interesting test cases there is no
|
||||||
|
// easy way to identify the object short of applying
|
||||||
|
// a series of destructuring operations to pkg---which
|
||||||
|
// is essentially what objectpath.Object does.
|
||||||
|
// (We do a little of that when testing bad paths, below.)
|
||||||
|
//
|
||||||
|
// The downside is that the test depends on the path encoding.
|
||||||
|
// The upside is that the test exercises the encoding.
|
||||||
|
|
||||||
|
// good paths
|
||||||
|
for _, test := range []struct {
|
||||||
|
pkg *types.Package
|
||||||
|
path objectpath.Path
|
||||||
|
wantobj string
|
||||||
|
}{
|
||||||
|
{b, "C", "const b.C a.Int"},
|
||||||
|
{b, "F", "func b.F(a int, b int, c int, d a.T)"},
|
||||||
|
{b, "F.PA0", "var a int"},
|
||||||
|
{b, "F.PA1", "var b int"},
|
||||||
|
{b, "F.PA2", "var c int"},
|
||||||
|
{b, "F.PA3", "var d a.T"},
|
||||||
|
{b, "T", "type b.T struct{A int; b int; a.T}"},
|
||||||
|
{b, "T.O", "type b.T struct{A int; b int; a.T}"},
|
||||||
|
{b, "T.UF0", "field A int"},
|
||||||
|
{b, "T.UF1", "field b int"},
|
||||||
|
{b, "T.UF2", "field T a.T"},
|
||||||
|
{b, "U.UF2", "field T a.T"}, // U.U... are aliases for T.U...
|
||||||
|
{b, "A", "type b.A = struct{x int}"},
|
||||||
|
{b, "A.F0", "field x int"},
|
||||||
|
{b, "V", "var b.V []*a.T"},
|
||||||
|
{b, "M", "type b.M map[struct{x int}]struct{y int}"},
|
||||||
|
{b, "M.UKF0", "field x int"},
|
||||||
|
{b, "M.UEF0", "field y int"},
|
||||||
|
{b, "T.M0", "func (b.T).M() *interface{f()}"}, // concrete method
|
||||||
|
{b, "T.M0.RA0", "var *interface{f()}"}, // parameter
|
||||||
|
{b, "T.M0.RA0.EM0", "func (interface).f()"}, // interface method
|
||||||
|
{b, "unexportedType", "type b.unexportedType struct{}"},
|
||||||
|
{a, "T", "type a.T struct{x int; y int}"},
|
||||||
|
{a, "T.UF0", "field x int"},
|
||||||
|
} {
|
||||||
|
// check path -> object
|
||||||
|
obj, err := objectpath.Object(test.pkg, test.path)
|
||||||
|
if err != nil {
|
||||||
|
t.Errorf("Object(%s, %q) failed: %v",
|
||||||
|
test.pkg.Path(), test.path, err)
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
if obj.String() != test.wantobj {
|
||||||
|
t.Errorf("Object(%s, %q) = %v, want %s",
|
||||||
|
test.pkg.Path(), test.path, obj, test.wantobj)
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
if obj.Pkg() != test.pkg {
|
||||||
|
t.Errorf("Object(%s, %q) = %v, which belongs to package %s",
|
||||||
|
test.pkg.Path(), test.path, obj, obj.Pkg().Path())
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
|
||||||
|
// check object -> path
|
||||||
|
path2, err := objectpath.For(obj)
|
||||||
|
if err != nil {
|
||||||
|
t.Errorf("For(%v) failed: %v, want %q", obj, err, test.path)
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
// We do not require that test.path == path2. Aliases are legal.
|
||||||
|
// But we do require that Object(path2) finds the same object.
|
||||||
|
obj2, err := objectpath.Object(test.pkg, path2)
|
||||||
|
if err != nil {
|
||||||
|
t.Errorf("Object(%s, %q) failed: %v (roundtrip from %q)",
|
||||||
|
test.pkg.Path(), path2, err, test.path)
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
if obj2 != obj {
|
||||||
|
t.Errorf("Object(%s, For(obj)) != obj: got %s, obj is %s (path1=%q, path2=%q)",
|
||||||
|
test.pkg.Path(), obj2, obj, test.path, path2)
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// bad paths (all relative to package b)
|
||||||
|
for _, test := range []struct {
|
||||||
|
pkg *types.Package
|
||||||
|
path objectpath.Path
|
||||||
|
wantErr string
|
||||||
|
}{
|
||||||
|
{b, "", "empty path"},
|
||||||
|
{b, "missing", `package b does not contain "missing"`},
|
||||||
|
{b, "F.U", "invalid path: ends with 'U', want [AFMO]"},
|
||||||
|
{b, "F.PA3.O", "path denotes type a.T struct{x int; y int}, which belongs to a different package"},
|
||||||
|
{b, "F.PA!", `invalid path: bad numeric operand "" for code 'A'`},
|
||||||
|
{b, "F.PA3.UF0", "path denotes field x int, which belongs to a different package"},
|
||||||
|
{b, "F.PA3.UF5", "field index 5 out of range [0-2)"},
|
||||||
|
{b, "V.EE", "invalid path: ends with 'E', want [AFMO]"},
|
||||||
|
{b, "F..O", "invalid path: unexpected '.' in type context"},
|
||||||
|
{b, "T.OO", "invalid path: code 'O' in object context"},
|
||||||
|
{b, "T.EO", "cannot apply 'E' to b.T (got *types.Named, want pointer, slice, array, chan or map)"},
|
||||||
|
{b, "A.O", "cannot apply 'O' to struct{x int} (got struct{x int}, want named)"},
|
||||||
|
{b, "A.UF0", "cannot apply 'U' to struct{x int} (got struct{x int}, want named)"},
|
||||||
|
{b, "M.UPO", "cannot apply 'P' to map[struct{x int}]struct{y int} (got *types.Map, want signature)"},
|
||||||
|
{b, "C.O", "path denotes type a.Int int, which belongs to a different package"},
|
||||||
|
} {
|
||||||
|
obj, err := objectpath.Object(test.pkg, test.path)
|
||||||
|
if err == nil {
|
||||||
|
t.Errorf("Object(%s, %q) = %s, want error",
|
||||||
|
test.pkg.Path(), test.path, obj)
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
if err.Error() != test.wantErr {
|
||||||
|
t.Errorf("Object(%s, %q) error was %q, want %q",
|
||||||
|
test.pkg.Path(), test.path, err, test.wantErr)
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// bad objects
|
||||||
|
bInfo := prog.Imported["b"]
|
||||||
|
for _, test := range []struct {
|
||||||
|
obj types.Object
|
||||||
|
wantErr string
|
||||||
|
}{
|
||||||
|
{types.Universe.Lookup("nil"), "predeclared nil has no path"},
|
||||||
|
{types.Universe.Lookup("len"), "predeclared builtin len has no path"},
|
||||||
|
{types.Universe.Lookup("int"), "predeclared type int has no path"},
|
||||||
|
{bInfo.Info.Implicits[bInfo.Files[0].Imports[0]], "no path for package a"}, // import "a"
|
||||||
|
{b.Scope().Lookup("unexportedFunc"), "no path for non-exported func b.unexportedFunc()"},
|
||||||
|
} {
|
||||||
|
path, err := objectpath.For(test.obj)
|
||||||
|
if err == nil {
|
||||||
|
t.Errorf("Object(%s) = %q, want error", test.obj, path)
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
if err.Error() != test.wantErr {
|
||||||
|
t.Errorf("Object(%s) error was %q, want %q", test.obj, err, test.wantErr)
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// TestSourceAndExportData uses objectpath to compute a correspondence
|
||||||
|
// of objects between two versions of the same package, one loaded from
|
||||||
|
// source, the other from export data.
|
||||||
|
func TestSourceAndExportData(t *testing.T) {
|
||||||
|
const src = `
|
||||||
|
package p
|
||||||
|
|
||||||
|
type I int
|
||||||
|
|
||||||
|
func (I) F() *struct{ X, Y int } {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
type Foo interface {
|
||||||
|
Method() (string, func(int) struct{ X int })
|
||||||
|
}
|
||||||
|
|
||||||
|
var X chan struct{ Z int }
|
||||||
|
var Z map[string]struct{ A int }
|
||||||
|
`
|
||||||
|
|
||||||
|
// Parse source file and type-check it as a package, "src".
|
||||||
|
fset := token.NewFileSet()
|
||||||
|
f, err := parser.ParseFile(fset, "src.go", src, 0)
|
||||||
|
if err != nil {
|
||||||
|
t.Fatal(err)
|
||||||
|
}
|
||||||
|
conf := types.Config{Importer: importer.For("source", nil)}
|
||||||
|
info := &types.Info{
|
||||||
|
Defs: make(map[*ast.Ident]types.Object),
|
||||||
|
}
|
||||||
|
srcpkg, err := conf.Check("src/p", fset, []*ast.File{f}, info)
|
||||||
|
if err != nil {
|
||||||
|
t.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Export binary export data then reload it as a new package, "bin".
|
||||||
|
var buf bytes.Buffer
|
||||||
|
if err := gcexportdata.Write(&buf, fset, srcpkg); err != nil {
|
||||||
|
t.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
imports := make(map[string]*types.Package)
|
||||||
|
binpkg, err := gcexportdata.Read(&buf, fset, imports, "bin/p")
|
||||||
|
if err != nil {
|
||||||
|
t.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Now find the correspondences between them.
|
||||||
|
for _, srcobj := range info.Defs {
|
||||||
|
if srcobj == nil {
|
||||||
|
continue // e.g. package declaration
|
||||||
|
}
|
||||||
|
if _, ok := srcobj.(*types.PkgName); ok {
|
||||||
|
continue // PkgName has no objectpath
|
||||||
|
}
|
||||||
|
|
||||||
|
path, err := objectpath.For(srcobj)
|
||||||
|
if err != nil {
|
||||||
|
t.Errorf("For(%v): %v", srcobj, err)
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
binobj, err := objectpath.Object(binpkg, path)
|
||||||
|
if err != nil {
|
||||||
|
t.Errorf("Object(%s, %q): %v", binpkg.Path(), path, err)
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
|
||||||
|
// Check the object strings match.
|
||||||
|
// (We can't check that types are identical because the
|
||||||
|
// objects belong to different type-checker realms.)
|
||||||
|
srcstr := types.ObjectString(srcobj, (*types.Package).Name)
|
||||||
|
binstr := types.ObjectString(binobj, (*types.Package).Name)
|
||||||
|
if srcstr != binstr {
|
||||||
|
t.Errorf("ObjectStrings do not match: Object(For(%q)) = %s, want %s",
|
||||||
|
path, srcstr, binstr)
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
Loading…
Reference in New Issue