| // 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 lsp |
| |
| import ( |
| "context" |
| |
| "golang.org/x/tools/internal/lsp/protocol" |
| "golang.org/x/tools/internal/lsp/source" |
| "golang.org/x/tools/internal/lsp/telemetry" |
| "golang.org/x/tools/internal/span" |
| "golang.org/x/tools/internal/telemetry/log" |
| ) |
| |
| func (s *Server) implementation(ctx context.Context, params *protocol.ImplementationParams) ([]protocol.Location, error) { |
| uri := span.NewURI(params.TextDocument.URI) |
| view, err := s.session.ViewOf(uri) |
| if err != nil { |
| return nil, err |
| } |
| snapshot := view.Snapshot() |
| f, err := view.GetFile(ctx, uri) |
| if err != nil { |
| return nil, err |
| } |
| if f.Kind() != source.Go { |
| return nil, nil |
| } |
| phs, err := snapshot.PackageHandles(ctx, snapshot.Handle(ctx, f)) |
| if err != nil { |
| return nil, err |
| } |
| var ( |
| allLocs []protocol.Location |
| seen = make(map[protocol.Location]bool) |
| ) |
| for _, ph := range phs { |
| ctx := telemetry.Package.With(ctx, ph.ID()) |
| |
| ident, err := source.Identifier(ctx, snapshot, f, params.Position, source.SpecificPackageHandle(ph.ID())) |
| if err != nil { |
| if err == source.ErrNoIdentFound { |
| return nil, err |
| } |
| log.Error(ctx, "failed to find Identifer", err) |
| continue |
| } |
| |
| locs, err := ident.Implementation(ctx) |
| if err != nil { |
| if err == source.ErrNotAnInterface { |
| return nil, err |
| } |
| log.Error(ctx, "failed to find Implemenation", err) |
| continue |
| } |
| |
| for _, loc := range locs { |
| if seen[loc] { |
| continue |
| } |
| seen[loc] = true |
| allLocs = append(allLocs, loc) |
| } |
| } |
| |
| return allLocs, nil |
| } |