blob: 35959c2de7a96e230f0cb7fc6363d59275fbfa8a [file] [log] [blame]
// 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 golang
import (
"context"
"fmt"
"go/ast"
"go/token"
"go/types"
"golang.org/x/tools/gopls/internal/cache"
"golang.org/x/tools/gopls/internal/cache/parsego"
"golang.org/x/tools/gopls/internal/file"
"golang.org/x/tools/gopls/internal/protocol"
"golang.org/x/tools/internal/event"
)
func DocumentSymbols(ctx context.Context, snapshot *cache.Snapshot, fh file.Handle) ([]protocol.DocumentSymbol, error) {
ctx, done := event.Start(ctx, "golang.DocumentSymbols")
defer done()
pgf, err := snapshot.ParseGo(ctx, fh, parsego.Full)
if err != nil {
return nil, fmt.Errorf("getting file for DocumentSymbols: %w", err)
}
// Build symbols for file declarations. When encountering a declaration with
// errors (typically because positions are invalid), we skip the declaration
// entirely. VS Code fails to show any symbols if one of the top-level
// symbols is missing position information.
var symbols []protocol.DocumentSymbol
for _, decl := range pgf.File.Decls {
switch decl := decl.(type) {
case *ast.FuncDecl:
if decl.Name.Name == "_" {
continue
}
fs, err := funcSymbol(pgf.Mapper, pgf.Tok, decl)
if err == nil {
// If function is a method, prepend the type of the method.
if decl.Recv != nil && len(decl.Recv.List) > 0 {
fs.Name = fmt.Sprintf("(%s).%s", types.ExprString(decl.Recv.List[0].Type), fs.Name)
}
symbols = append(symbols, fs)
}
case *ast.GenDecl:
for _, spec := range decl.Specs {
switch spec := spec.(type) {
case *ast.TypeSpec:
if spec.Name.Name == "_" {
continue
}
ts, err := typeSymbol(pgf.Mapper, pgf.Tok, spec)
if err == nil {
symbols = append(symbols, ts)
}
case *ast.ValueSpec:
for _, name := range spec.Names {
if name.Name == "_" {
continue
}
vs, err := varSymbol(pgf.Mapper, pgf.Tok, spec, name, decl.Tok == token.CONST)
if err == nil {
symbols = append(symbols, vs)
}
}
}
}
}
}
return symbols, nil
}
func funcSymbol(m *protocol.Mapper, tf *token.File, decl *ast.FuncDecl) (protocol.DocumentSymbol, error) {
s := protocol.DocumentSymbol{
Name: decl.Name.Name,
Kind: protocol.Function,
}
if decl.Recv != nil {
s.Kind = protocol.Method
}
var err error
s.Range, err = m.NodeRange(tf, decl)
if err != nil {
return protocol.DocumentSymbol{}, err
}
s.SelectionRange, err = m.NodeRange(tf, decl.Name)
if err != nil {
return protocol.DocumentSymbol{}, err
}
s.Detail = types.ExprString(decl.Type)
return s, nil
}
func typeSymbol(m *protocol.Mapper, tf *token.File, spec *ast.TypeSpec) (protocol.DocumentSymbol, error) {
s := protocol.DocumentSymbol{
Name: spec.Name.Name,
}
var err error
s.Range, err = m.NodeRange(tf, spec)
if err != nil {
return protocol.DocumentSymbol{}, err
}
s.SelectionRange, err = m.NodeRange(tf, spec.Name)
if err != nil {
return protocol.DocumentSymbol{}, err
}
s.Kind, s.Detail, s.Children = typeDetails(m, tf, spec.Type)
return s, nil
}
func typeDetails(m *protocol.Mapper, tf *token.File, typExpr ast.Expr) (kind protocol.SymbolKind, detail string, children []protocol.DocumentSymbol) {
switch typExpr := typExpr.(type) {
case *ast.StructType:
kind = protocol.Struct
children = fieldListSymbols(m, tf, typExpr.Fields, protocol.Field)
if len(children) > 0 {
detail = "struct{...}"
} else {
detail = "struct{}"
}
// Find interface methods and embedded types.
case *ast.InterfaceType:
kind = protocol.Interface
children = fieldListSymbols(m, tf, typExpr.Methods, protocol.Method)
if len(children) > 0 {
detail = "interface{...}"
} else {
detail = "interface{}"
}
case *ast.FuncType:
kind = protocol.Function
detail = types.ExprString(typExpr)
default:
kind = protocol.Class // catch-all, for cases where we don't know the kind syntactically
detail = types.ExprString(typExpr)
}
return
}
func fieldListSymbols(m *protocol.Mapper, tf *token.File, fields *ast.FieldList, fieldKind protocol.SymbolKind) []protocol.DocumentSymbol {
if fields == nil {
return nil
}
var symbols []protocol.DocumentSymbol
for _, field := range fields.List {
detail, children := "", []protocol.DocumentSymbol(nil)
if field.Type != nil {
_, detail, children = typeDetails(m, tf, field.Type)
}
if len(field.Names) == 0 { // embedded interface or struct field
// By default, use the formatted type details as the name of this field.
// This handles potentially invalid syntax, as well as type embeddings in
// interfaces.
child := protocol.DocumentSymbol{
Name: detail,
Kind: protocol.Field, // consider all embeddings to be fields
Children: children,
}
// If the field is a valid embedding, promote the type name to field
// name.
selection := field.Type
if id := embeddedIdent(field.Type); id != nil {
child.Name = id.Name
child.Detail = detail
selection = id
}
if rng, err := m.NodeRange(tf, field.Type); err == nil {
child.Range = rng
}
if rng, err := m.NodeRange(tf, selection); err == nil {
child.SelectionRange = rng
}
symbols = append(symbols, child)
} else {
for _, name := range field.Names {
child := protocol.DocumentSymbol{
Name: name.Name,
Kind: fieldKind,
Detail: detail,
Children: children,
}
if rng, err := m.NodeRange(tf, field); err == nil {
child.Range = rng
}
if rng, err := m.NodeRange(tf, name); err == nil {
child.SelectionRange = rng
}
symbols = append(symbols, child)
}
}
}
return symbols
}
func varSymbol(m *protocol.Mapper, tf *token.File, spec *ast.ValueSpec, name *ast.Ident, isConst bool) (protocol.DocumentSymbol, error) {
s := protocol.DocumentSymbol{
Name: name.Name,
Kind: protocol.Variable,
}
if isConst {
s.Kind = protocol.Constant
}
var err error
s.Range, err = m.NodeRange(tf, spec)
if err != nil {
return protocol.DocumentSymbol{}, err
}
s.SelectionRange, err = m.NodeRange(tf, name)
if err != nil {
return protocol.DocumentSymbol{}, err
}
if spec.Type != nil { // type may be missing from the syntax
_, s.Detail, s.Children = typeDetails(m, tf, spec.Type)
}
return s, nil
}