188 lines
		
	
	
		
			4.6 KiB
		
	
	
	
		
			Go
		
	
	
	
			
		
		
	
	
			188 lines
		
	
	
		
			4.6 KiB
		
	
	
	
		
			Go
		
	
	
	
| // Copyright 2013 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 oracle
 | |
| 
 | |
| import (
 | |
| 	"go/ast"
 | |
| 	"go/token"
 | |
| 	"sort"
 | |
| 
 | |
| 	"code.google.com/p/go.tools/go/types"
 | |
| 	"code.google.com/p/go.tools/oracle/json"
 | |
| )
 | |
| 
 | |
| // freevars displays the lexical (not package-level) free variables of
 | |
| // the selection.
 | |
| //
 | |
| // It treats A.B.C as a separate variable from A to reveal the parts
 | |
| // of an aggregate type that are actually needed.
 | |
| // This aids refactoring.
 | |
| //
 | |
| // TODO(adonovan): optionally display the free references to
 | |
| // file/package scope objects, and to objects from other packages.
 | |
| // Depending on where the resulting function abstraction will go,
 | |
| // these might be interesting.  Perhaps group the results into three
 | |
| // bands.
 | |
| //
 | |
| func freevars(o *oracle) (queryResult, error) {
 | |
| 	file := o.queryPath[len(o.queryPath)-1] // the enclosing file
 | |
| 
 | |
| 	// The id and sel functions return non-nil if they denote an
 | |
| 	// object o or selection o.x.y that is referenced by the
 | |
| 	// selection but defined neither within the selection nor at
 | |
| 	// file scope, i.e. it is in the lexical environment.
 | |
| 	var id func(n *ast.Ident) types.Object
 | |
| 	var sel func(n *ast.SelectorExpr) types.Object
 | |
| 
 | |
| 	sel = func(n *ast.SelectorExpr) types.Object {
 | |
| 		switch x := unparen(n.X).(type) {
 | |
| 		case *ast.SelectorExpr:
 | |
| 			return sel(x)
 | |
| 		case *ast.Ident:
 | |
| 			return id(x)
 | |
| 		}
 | |
| 		return nil
 | |
| 	}
 | |
| 
 | |
| 	id = func(n *ast.Ident) types.Object {
 | |
| 		obj := o.queryPkgInfo.ObjectOf(n)
 | |
| 		if obj == nil {
 | |
| 			return nil // TODO(adonovan): fix: this fails for *types.Label.
 | |
| 			panic(o.errorf(n, "no types.Object for ast.Ident"))
 | |
| 		}
 | |
| 		if _, ok := obj.(*types.Package); ok {
 | |
| 			return nil // imported package
 | |
| 		}
 | |
| 		if n.Pos() == obj.Pos() {
 | |
| 			return nil // this ident is the definition, not a reference
 | |
| 		}
 | |
| 		if !(file.Pos() <= obj.Pos() && obj.Pos() <= file.End()) {
 | |
| 			return nil // not defined in this file
 | |
| 		}
 | |
| 		if obj.Parent() == nil {
 | |
| 			return nil // e.g. interface method  TODO(adonovan): what else?
 | |
| 		}
 | |
| 		if obj.Parent() == o.queryPkgInfo.Scopes[file] {
 | |
| 			return nil // defined at file scope
 | |
| 		}
 | |
| 		if o.startPos <= obj.Pos() && obj.Pos() <= o.endPos {
 | |
| 			return nil // defined within selection => not free
 | |
| 		}
 | |
| 		return obj
 | |
| 	}
 | |
| 
 | |
| 	// Maps each reference that is free in the selection
 | |
| 	// to the object it refers to.
 | |
| 	// The map de-duplicates repeated references.
 | |
| 	refsMap := make(map[string]freevarsRef)
 | |
| 
 | |
| 	// Visit all the identifiers in the selected ASTs.
 | |
| 	ast.Inspect(o.queryPath[0], func(n ast.Node) bool {
 | |
| 		if n == nil {
 | |
| 			return true // popping DFS stack
 | |
| 		}
 | |
| 
 | |
| 		// Is this node contained within the selection?
 | |
| 		// (freevars permits inexact selections,
 | |
| 		// like two stmts in a block.)
 | |
| 		if o.startPos <= n.Pos() && n.End() <= o.endPos {
 | |
| 			var obj types.Object
 | |
| 			var prune bool
 | |
| 			switch n := n.(type) {
 | |
| 			case *ast.Ident:
 | |
| 				obj = id(n)
 | |
| 
 | |
| 			case *ast.SelectorExpr:
 | |
| 				obj = sel(n)
 | |
| 				prune = true
 | |
| 			}
 | |
| 
 | |
| 			if obj != nil {
 | |
| 				var kind string
 | |
| 				switch obj.(type) {
 | |
| 				case *types.Var:
 | |
| 					kind = "var"
 | |
| 				case *types.Func:
 | |
| 					kind = "func"
 | |
| 				case *types.TypeName:
 | |
| 					kind = "type"
 | |
| 				case *types.Const:
 | |
| 					kind = "const"
 | |
| 				case *types.Label:
 | |
| 					kind = "label"
 | |
| 				default:
 | |
| 					panic(obj)
 | |
| 				}
 | |
| 
 | |
| 				typ := o.queryPkgInfo.TypeOf(n.(ast.Expr))
 | |
| 				ref := freevarsRef{kind, o.printNode(n), typ, obj}
 | |
| 				refsMap[ref.ref] = ref
 | |
| 
 | |
| 				if prune {
 | |
| 					return false // don't descend
 | |
| 				}
 | |
| 			}
 | |
| 		}
 | |
| 
 | |
| 		return true // descend
 | |
| 	})
 | |
| 
 | |
| 	refs := make([]freevarsRef, 0, len(refsMap))
 | |
| 	for _, ref := range refsMap {
 | |
| 		refs = append(refs, ref)
 | |
| 	}
 | |
| 	sort.Sort(byRef(refs))
 | |
| 
 | |
| 	return &freevarsResult{
 | |
| 		fset: o.prog.Fset,
 | |
| 		refs: refs,
 | |
| 	}, nil
 | |
| }
 | |
| 
 | |
| type freevarsResult struct {
 | |
| 	fset *token.FileSet
 | |
| 	refs []freevarsRef
 | |
| }
 | |
| 
 | |
| type freevarsRef struct {
 | |
| 	kind string
 | |
| 	ref  string
 | |
| 	typ  types.Type
 | |
| 	obj  types.Object
 | |
| }
 | |
| 
 | |
| func (r *freevarsResult) display(printf printfFunc) {
 | |
| 	if len(r.refs) == 0 {
 | |
| 		printf(false, "No free identifers.")
 | |
| 	} else {
 | |
| 		printf(false, "Free identifers:")
 | |
| 		for _, ref := range r.refs {
 | |
| 			printf(ref.obj, "%s %s %s", ref.kind, ref.ref, ref.typ)
 | |
| 		}
 | |
| 	}
 | |
| }
 | |
| 
 | |
| func (r *freevarsResult) toJSON(res *json.Result, fset *token.FileSet) {
 | |
| 	var refs []*json.FreeVar
 | |
| 	for _, ref := range r.refs {
 | |
| 		refs = append(refs,
 | |
| 			&json.FreeVar{
 | |
| 				Pos:  fset.Position(ref.obj.Pos()).String(),
 | |
| 				Kind: ref.kind,
 | |
| 				Ref:  ref.ref,
 | |
| 				Type: ref.typ.String(),
 | |
| 			})
 | |
| 	}
 | |
| 	res.Freevars = refs
 | |
| }
 | |
| 
 | |
| // -------- utils --------
 | |
| 
 | |
| type byRef []freevarsRef
 | |
| 
 | |
| func (p byRef) Len() int           { return len(p) }
 | |
| func (p byRef) Less(i, j int) bool { return p[i].ref < p[j].ref }
 | |
| func (p byRef) Swap(i, j int)      { p[i], p[j] = p[j], p[i] }
 |