2018-11-02 16:10:49 -06:00
|
|
|
// Copyright 2018 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 source
|
|
|
|
|
|
|
|
import (
|
2018-12-18 14:18:03 -07:00
|
|
|
"context"
|
2018-11-02 16:10:49 -06:00
|
|
|
"go/ast"
|
|
|
|
"go/token"
|
2019-03-06 14:33:47 -07:00
|
|
|
"go/types"
|
2019-04-08 07:22:58 -06:00
|
|
|
"strings"
|
2018-11-07 13:21:31 -07:00
|
|
|
|
2019-03-06 14:33:47 -07:00
|
|
|
"golang.org/x/tools/go/analysis"
|
2018-11-07 13:21:31 -07:00
|
|
|
"golang.org/x/tools/go/packages"
|
2019-04-08 07:22:58 -06:00
|
|
|
"golang.org/x/tools/internal/lsp/diff"
|
2019-03-29 17:04:29 -06:00
|
|
|
"golang.org/x/tools/internal/lsp/xlog"
|
2019-02-19 19:11:15 -07:00
|
|
|
"golang.org/x/tools/internal/span"
|
2018-11-02 16:10:49 -06:00
|
|
|
)
|
|
|
|
|
2018-12-18 13:46:14 -07:00
|
|
|
// View abstracts the underlying architecture of the package using the source
|
|
|
|
// package. The view provides access to files and their contents, so the source
|
|
|
|
// package does not directly access the file system.
|
|
|
|
type View interface {
|
2019-03-29 17:04:29 -06:00
|
|
|
Logger() xlog.Logger
|
2019-03-14 15:19:01 -06:00
|
|
|
FileSet() *token.FileSet
|
2019-02-19 19:11:15 -07:00
|
|
|
GetFile(ctx context.Context, uri span.URI) (File, error)
|
|
|
|
SetContent(ctx context.Context, uri span.URI, content []byte) error
|
2018-12-18 13:46:14 -07:00
|
|
|
}
|
|
|
|
|
2018-12-05 15:00:36 -07:00
|
|
|
// File represents a Go source file that has been type-checked. It is the input
|
|
|
|
// to most of the exported functions in this package, as it wraps up the
|
|
|
|
// building blocks for most queries. Users of the source package can abstract
|
|
|
|
// the loading of packages into their own caching systems.
|
|
|
|
type File interface {
|
2019-02-19 19:11:15 -07:00
|
|
|
URI() span.URI
|
2019-03-05 15:30:44 -07:00
|
|
|
GetAST(ctx context.Context) *ast.File
|
|
|
|
GetFileSet(ctx context.Context) *token.FileSet
|
2019-03-06 14:33:47 -07:00
|
|
|
GetPackage(ctx context.Context) Package
|
2019-03-05 15:30:44 -07:00
|
|
|
GetToken(ctx context.Context) *token.File
|
|
|
|
GetContent(ctx context.Context) []byte
|
2018-11-05 15:54:12 -07:00
|
|
|
}
|
|
|
|
|
2019-03-06 14:33:47 -07:00
|
|
|
// Package represents a Go package that has been type-checked. It maintains
|
|
|
|
// only the relevant fields of a *go/packages.Package.
|
|
|
|
type Package interface {
|
|
|
|
GetFilenames() []string
|
|
|
|
GetSyntax() []*ast.File
|
|
|
|
GetErrors() []packages.Error
|
|
|
|
GetTypes() *types.Package
|
|
|
|
GetTypesInfo() *types.Info
|
2019-03-29 14:39:22 -06:00
|
|
|
GetTypesSizes() types.Sizes
|
2019-03-11 15:14:55 -06:00
|
|
|
IsIllTyped() bool
|
2019-03-06 14:33:47 -07:00
|
|
|
GetActionGraph(ctx context.Context, a *analysis.Analyzer) (*Action, error)
|
|
|
|
}
|
|
|
|
|
2018-11-07 10:58:55 -07:00
|
|
|
// TextEdit represents a change to a section of a document.
|
2019-02-19 19:11:15 -07:00
|
|
|
// The text within the specified span should be replaced by the supplied new text.
|
2018-11-07 10:58:55 -07:00
|
|
|
type TextEdit struct {
|
2019-02-19 19:11:15 -07:00
|
|
|
Span span.Span
|
2018-11-07 10:58:55 -07:00
|
|
|
NewText string
|
|
|
|
}
|
2019-04-08 07:22:58 -06:00
|
|
|
|
|
|
|
// DiffToEdits converts from a sequence of diff operations to a sequence of
|
|
|
|
// source.TextEdit
|
|
|
|
func DiffToEdits(uri span.URI, ops []*diff.Op) []TextEdit {
|
|
|
|
edits := make([]TextEdit, 0, len(ops))
|
|
|
|
for _, op := range ops {
|
|
|
|
s := span.New(uri, span.NewPoint(op.I1+1, 1, 0), span.NewPoint(op.I2+1, 1, 0))
|
|
|
|
switch op.Kind {
|
|
|
|
case diff.Delete:
|
|
|
|
// Delete: unformatted[i1:i2] is deleted.
|
|
|
|
edits = append(edits, TextEdit{Span: s})
|
|
|
|
case diff.Insert:
|
|
|
|
// Insert: formatted[j1:j2] is inserted at unformatted[i1:i1].
|
|
|
|
if content := strings.Join(op.Content, ""); content != "" {
|
|
|
|
edits = append(edits, TextEdit{Span: s, NewText: content})
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return edits
|
|
|
|
}
|
|
|
|
|
|
|
|
func EditsToDiff(edits []TextEdit) []*diff.Op {
|
|
|
|
iToJ := 0
|
|
|
|
ops := make([]*diff.Op, len(edits))
|
|
|
|
for i, edit := range edits {
|
|
|
|
i1 := edit.Span.Start().Line() - 1
|
|
|
|
i2 := edit.Span.End().Line() - 1
|
|
|
|
kind := diff.Insert
|
|
|
|
if edit.NewText == "" {
|
|
|
|
kind = diff.Delete
|
|
|
|
}
|
|
|
|
ops[i] = &diff.Op{
|
|
|
|
Kind: kind,
|
|
|
|
Content: diff.SplitLines(edit.NewText),
|
|
|
|
I1: i1,
|
|
|
|
I2: i2,
|
|
|
|
J1: i1 + iToJ,
|
|
|
|
}
|
|
|
|
if kind == diff.Insert {
|
|
|
|
iToJ += len(ops[i].Content)
|
|
|
|
} else {
|
|
|
|
iToJ -= i2 - i1
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return ops
|
|
|
|
}
|