| // Copyright 2020 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 ( |
| "fmt" |
| "sort" |
| "strings" |
| "testing" |
| |
| "golang.org/x/tools/gopls/internal/lsp/protocol" |
| "golang.org/x/tools/gopls/internal/lsp/tests" |
| "golang.org/x/tools/gopls/internal/span" |
| ) |
| |
| func (r *runner) CallHierarchy(t *testing.T, spn span.Span, expectedCalls *tests.CallHierarchyResult) { |
| collectCallSpansString := func(callItems []protocol.CallHierarchyItem) string { |
| var callSpans []string |
| for _, call := range callItems { |
| mapper, err := r.data.Mapper(call.URI.SpanURI()) |
| if err != nil { |
| t.Fatal(err) |
| } |
| callSpan, err := mapper.Span(protocol.Location{URI: call.URI, Range: call.Range}) |
| if err != nil { |
| t.Fatal(err) |
| } |
| callSpans = append(callSpans, fmt.Sprint(callSpan)) |
| } |
| // to make tests deterministic |
| sort.Strings(callSpans) |
| return r.Normalize(strings.Join(callSpans, "\n")) |
| } |
| |
| expectIn, expectOut := collectCallSpansString(expectedCalls.IncomingCalls), collectCallSpansString(expectedCalls.OutgoingCalls) |
| expectIdent := r.Normalize(fmt.Sprint(spn)) |
| |
| uri := spn.URI() |
| filename := uri.Filename() |
| target := filename + fmt.Sprintf(":%v:%v", spn.Start().Line(), spn.Start().Column()) |
| |
| got, stderr := r.NormalizeGoplsCmd(t, "call_hierarchy", target) |
| if stderr != "" { |
| t.Fatalf("call_hierarchy failed for %s: %s", target, stderr) |
| } |
| |
| gotIn, gotIdent, gotOut := cleanCallHierarchyCmdResult(got) |
| if expectIn != gotIn { |
| t.Errorf("incoming calls call_hierarchy failed for %s expected:\n%s\ngot:\n%s", target, expectIn, gotIn) |
| } |
| if expectIdent != gotIdent { |
| t.Errorf("call_hierarchy failed for %s expected:\n%s\ngot:\n%s", target, expectIdent, gotIdent) |
| } |
| if expectOut != gotOut { |
| t.Errorf("outgoing calls call_hierarchy failed for %s expected:\n%s\ngot:\n%s", target, expectOut, gotOut) |
| } |
| |
| } |
| |
| // parses function URI and Range from call hierarchy cmd output to |
| // incoming, identifier and outgoing calls (returned in that order) |
| // ex: "identifier: function d at .../callhierarchy/callhierarchy.go:19:6-7" -> ".../callhierarchy/callhierarchy.go:19:6-7" |
| func cleanCallHierarchyCmdResult(output string) (incoming, ident, outgoing string) { |
| var incomingCalls, outgoingCalls []string |
| for _, out := range strings.Split(output, "\n") { |
| if out == "" { |
| continue |
| } |
| |
| callLocation := out[strings.LastIndex(out, " ")+1:] |
| if strings.HasPrefix(out, "caller") { |
| incomingCalls = append(incomingCalls, callLocation) |
| } else if strings.HasPrefix(out, "callee") { |
| outgoingCalls = append(outgoingCalls, callLocation) |
| } else { |
| ident = callLocation |
| } |
| } |
| sort.Strings(incomingCalls) |
| sort.Strings(outgoingCalls) |
| incoming, outgoing = strings.Join(incomingCalls, "\n"), strings.Join(outgoingCalls, "\n") |
| return |
| } |