tools/gopls: add cmd support for symbols

This change adds command line support for symbols.
Symbols are formatted as '{name} {type} {range}', with
children being preceded by a \t.

Example:

$ gopls symbols ~/tmp/foo/main.go
$
$ x Variable 7:5-7:6
$ y Constant 9:7-9:8
$ Quux Struct 29:6-29:10
$ 	Do Method 37:16-37:18
$ 	X Field 30:2-30:3
$ 	Y Field 30:5-30:6

Updates golang/go#32875

Change-Id: I1272fce733fb12b67e3d6fb948f5bf3de4ca2ca1
Reviewed-on: https://go-review.googlesource.com/c/tools/+/203609
Run-TryBot: Rebecca Stambler <rstambler@golang.org>
Reviewed-by: Rebecca Stambler <rstambler@golang.org>
This commit is contained in:
Francesco Renzi 2019-10-27 17:41:48 +00:00 committed by Rebecca Stambler
parent b8f202ca5e
commit f02a19dded
8 changed files with 147 additions and 9 deletions

View File

@ -150,6 +150,7 @@ func (app *Application) commands() []tool.Application {
&rename{app: app}, &rename{app: app},
&signature{app: app}, &signature{app: app},
&suggestedfix{app: app}, &suggestedfix{app: app},
&symbols{app: app},
&version{app: app}, &version{app: app},
} }
} }

View File

@ -0,0 +1,80 @@
// 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 cmd
import (
"context"
"flag"
"fmt"
"sort"
"golang.org/x/tools/internal/lsp/protocol"
"golang.org/x/tools/internal/span"
"golang.org/x/tools/internal/tool"
)
// references implements the references verb for gopls
type symbols struct {
app *Application
}
func (r *symbols) Name() string { return "symbols" }
func (r *symbols) Usage() string { return "<file>" }
func (r *symbols) ShortHelp() string { return "display selected file's symbols" }
func (r *symbols) DetailedHelp(f *flag.FlagSet) {
fmt.Fprint(f.Output(), `
Example:
$ gopls symbols helper/helper.go
`)
f.PrintDefaults()
}
func (r *symbols) Run(ctx context.Context, args ...string) error {
if len(args) != 1 {
return tool.CommandLineErrorf("symbols expects 1 argument (position)")
}
conn, err := r.app.connect(ctx)
if err != nil {
return err
}
defer conn.terminate(ctx)
from := span.Parse(args[0])
p := protocol.DocumentSymbolParams{
TextDocument: protocol.TextDocumentIdentifier{
URI: string(from.URI()),
},
}
symbols, err := conn.DocumentSymbol(ctx, &p)
if err != nil {
return err
}
for _, s := range symbols {
fmt.Println(symbolToString(s))
// Sort children for consistency
sort.Slice(s.Children, func(i, j int) bool {
return s.Children[i].Name < s.Children[j].Name
})
for _, c := range s.Children {
fmt.Println("\t" + symbolToString(c))
}
}
return nil
}
func symbolToString(symbol protocol.DocumentSymbol) string {
r := symbol.SelectionRange
// convert ranges to user friendly 1-based positions
position := fmt.Sprintf("%v:%v-%v:%v",
r.Start.Line+1,
r.Start.Character+1,
r.End.Line+1,
r.End.Character+1,
)
return fmt.Sprintf("%s %s %s", symbol.Name, symbol.Kind, position)
}

View File

@ -16,7 +16,6 @@ import (
"testing" "testing"
"golang.org/x/tools/go/packages/packagestest" "golang.org/x/tools/go/packages/packagestest"
"golang.org/x/tools/internal/lsp/protocol"
"golang.org/x/tools/internal/lsp/source" "golang.org/x/tools/internal/lsp/source"
"golang.org/x/tools/internal/lsp/tests" "golang.org/x/tools/internal/lsp/tests"
"golang.org/x/tools/internal/span" "golang.org/x/tools/internal/span"
@ -78,10 +77,6 @@ func (r *runner) PrepareRename(t *testing.T, src span.Span, want *source.Prepare
//TODO: add command line prepare rename tests when it works //TODO: add command line prepare rename tests when it works
} }
func (r *runner) Symbol(t *testing.T, uri span.URI, expectedSymbols []protocol.DocumentSymbol) {
//TODO: add command line symbol tests when it works
}
func (r *runner) Implementation(t *testing.T, spn span.Span, imp tests.Implementations) { func (r *runner) Implementation(t *testing.T, spn span.Span, imp tests.Implementations) {
//TODO: add implements tests when it works //TODO: add implements tests when it works
} }

View File

@ -0,0 +1,33 @@
// 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 cmdtest
import (
"testing"
"fmt"
"golang.org/x/tools/internal/lsp/cmd"
"golang.org/x/tools/internal/lsp/protocol"
"golang.org/x/tools/internal/span"
"golang.org/x/tools/internal/tool"
)
func (r *runner) Symbols(t *testing.T, uri span.URI, expectedSymbols []protocol.DocumentSymbol) {
filename := uri.Filename()
app := cmd.New("gopls-test", r.data.Config.Dir, r.data.Config.Env, r.options)
got := CaptureStdOut(t, func() {
err := tool.Run(r.ctx, app, append([]string{"-remote=internal", "symbols"}, filename))
if err != nil {
fmt.Println(err)
}
})
expect := string(r.data.Golden("symbols", filename, func() ([]byte, error) {
return []byte(got), nil
}))
if expect != got {
t.Errorf("symbols failed for %s expected:\n%s\ngot:\n%s", filename, expect, got)
}
}

View File

@ -654,7 +654,7 @@ func applyEdits(contents string, edits []diff.TextEdit) string {
return res return res
} }
func (r *runner) Symbol(t *testing.T, uri span.URI, expectedSymbols []protocol.DocumentSymbol) { func (r *runner) Symbols(t *testing.T, uri span.URI, expectedSymbols []protocol.DocumentSymbol) {
params := &protocol.DocumentSymbolParams{ params := &protocol.DocumentSymbolParams{
TextDocument: protocol.TextDocumentIdentifier{ TextDocument: protocol.TextDocumentIdentifier{
URI: string(uri), URI: string(uri),

View File

@ -770,7 +770,7 @@ func (r *runner) PrepareRename(t *testing.T, src span.Span, want *source.Prepare
} }
} }
func (r *runner) Symbol(t *testing.T, uri span.URI, expectedSymbols []protocol.DocumentSymbol) { func (r *runner) Symbols(t *testing.T, uri span.URI, expectedSymbols []protocol.DocumentSymbol) {
ctx := r.ctx ctx := r.ctx
f, err := r.view.GetFile(ctx, uri) f, err := r.view.GetFile(ctx, uri)
if err != nil { if err != nil {

View File

@ -0,0 +1,29 @@
-- symbols --
x Variable 7:5-7:6
y Constant 9:7-9:8
Number Number 11:6-11:12
Alias String 13:6-13:11
NumberAlias Number 15:6-15:17
Boolean Boolean 18:2-18:9
BoolAlias Boolean 19:2-19:11
Foo Struct 22:6-22:9
Bar Field 25:2-25:5
Baz Method 33:14-33:17
Quux Field 23:2-23:6
W Field 24:2-24:3
baz Field 26:2-26:5
Quux Struct 29:6-29:10
Do Method 37:16-37:18
X Field 30:2-30:3
Y Field 30:5-30:6
main Function 39:6-39:10
Stringer Interface 43:6-43:14
String Method 44:2-44:8
ABer Interface 47:6-47:10
A Method 49:2-49:3
B Method 48:2-48:3
WithEmbeddeds Interface 52:6-52:19
ABer Interface 54:2-54:6
Do Method 53:2-53:4
io.Writer Interface 55:2-55:11

View File

@ -116,7 +116,7 @@ type Tests interface {
References(*testing.T, span.Span, []span.Span) References(*testing.T, span.Span, []span.Span)
Rename(*testing.T, span.Span, string) Rename(*testing.T, span.Span, string)
PrepareRename(*testing.T, span.Span, *source.PrepareItem) PrepareRename(*testing.T, span.Span, *source.PrepareItem)
Symbol(*testing.T, span.URI, []protocol.DocumentSymbol) Symbols(*testing.T, span.URI, []protocol.DocumentSymbol)
SignatureHelp(*testing.T, span.Span, *source.SignatureInformation) SignatureHelp(*testing.T, span.Span, *source.SignatureInformation)
Link(*testing.T, span.URI, []Link) Link(*testing.T, span.URI, []Link)
} }
@ -534,7 +534,7 @@ func Run(t *testing.T, tests Tests, data *Data) {
for uri, expectedSymbols := range data.Symbols { for uri, expectedSymbols := range data.Symbols {
t.Run(uriName(uri), func(t *testing.T) { t.Run(uriName(uri), func(t *testing.T) {
t.Helper() t.Helper()
tests.Symbol(t, uri, expectedSymbols) tests.Symbols(t, uri, expectedSymbols)
}) })
} }
}) })