mirror of
https://github.com/golang/go
synced 2024-11-18 19:44:46 -07:00
d3bf790afa
Now that we support authoring cgo packages better, we need a way to regenerate the C definitions. Doing it automatically is very difficult; in particular, referencing a new symbol from the C package may require regeneration, but we don't want to do that for every typo. For now, give the user a button and make them push it. We attach a code lens to the import "C" line. This is vulnerable to the usual user-didn't-save glitches. Updates golang/go#35721. Change-Id: Iaa3540a9a12bbd8705e7f0e43ad0be1b22e87067 Reviewed-on: https://go-review.googlesource.com/c/tools/+/234103 Run-TryBot: Heschi Kreinick <heschi@google.com> TryBot-Result: Gobot Gobot <gobot@golang.org> Reviewed-by: Robert Findley <rfindley@google.com>
191 lines
4.6 KiB
Go
191 lines
4.6 KiB
Go
// 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 source
|
|
|
|
import (
|
|
"context"
|
|
"go/ast"
|
|
"go/token"
|
|
"go/types"
|
|
"path/filepath"
|
|
"regexp"
|
|
"strings"
|
|
|
|
"golang.org/x/tools/internal/lsp/protocol"
|
|
)
|
|
|
|
type lensFunc func(context.Context, Snapshot, FileHandle, *ast.File, *protocol.ColumnMapper) ([]protocol.CodeLens, error)
|
|
|
|
var lensFuncs = map[string]lensFunc{
|
|
CommandGenerate: goGenerateCodeLens,
|
|
CommandTest: runTestCodeLens,
|
|
CommandRegenerateCgo: regenerateCgoLens,
|
|
}
|
|
|
|
// CodeLens computes code lens for Go source code.
|
|
func CodeLens(ctx context.Context, snapshot Snapshot, fh FileHandle) ([]protocol.CodeLens, error) {
|
|
f, _, m, _, err := snapshot.View().Session().Cache().ParseGoHandle(fh, ParseFull).Parse(ctx)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
var result []protocol.CodeLens
|
|
for lens, lf := range lensFuncs {
|
|
if !snapshot.View().Options().EnabledCodeLens[lens] {
|
|
continue
|
|
}
|
|
added, err := lf(ctx, snapshot, fh, f, m)
|
|
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
result = append(result, added...)
|
|
}
|
|
return result, nil
|
|
}
|
|
|
|
var testMatcher = regexp.MustCompile("^Test[^a-z]")
|
|
var benchMatcher = regexp.MustCompile("^Benchmark[^a-z]")
|
|
|
|
func runTestCodeLens(ctx context.Context, snapshot Snapshot, fh FileHandle, f *ast.File, m *protocol.ColumnMapper) ([]protocol.CodeLens, error) {
|
|
codeLens := make([]protocol.CodeLens, 0)
|
|
|
|
pkg, _, err := getParsedFile(ctx, snapshot, fh, WidestPackageHandle)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
if !strings.HasSuffix(fh.Identity().URI.Filename(), "_test.go") {
|
|
return nil, nil
|
|
}
|
|
|
|
for _, d := range f.Decls {
|
|
fn, ok := d.(*ast.FuncDecl)
|
|
if !ok {
|
|
continue
|
|
}
|
|
|
|
if isTestFunc(fn, pkg) {
|
|
fset := snapshot.View().Session().Cache().FileSet()
|
|
rng, err := newMappedRange(fset, m, d.Pos(), d.Pos()).Range()
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
uri := fh.Identity().URI
|
|
codeLens = append(codeLens, protocol.CodeLens{
|
|
Range: rng,
|
|
Command: protocol.Command{
|
|
Title: "run test",
|
|
Command: "test",
|
|
Arguments: []interface{}{fn.Name.Name, uri},
|
|
},
|
|
})
|
|
}
|
|
}
|
|
|
|
return codeLens, nil
|
|
}
|
|
|
|
func isTestFunc(fn *ast.FuncDecl, pkg Package) bool {
|
|
typesInfo := pkg.GetTypesInfo()
|
|
if typesInfo == nil {
|
|
return false
|
|
}
|
|
|
|
sig, ok := typesInfo.ObjectOf(fn.Name).Type().(*types.Signature)
|
|
if !ok {
|
|
return false
|
|
}
|
|
|
|
// test funcs should have a single parameter, so we can exit early if that's not the case.
|
|
if sig.Params().Len() != 1 {
|
|
return false
|
|
}
|
|
|
|
firstParam, ok := sig.Params().At(0).Type().(*types.Pointer)
|
|
if !ok {
|
|
return false
|
|
}
|
|
|
|
firstParamElem, ok := firstParam.Elem().(*types.Named)
|
|
if !ok {
|
|
return false
|
|
}
|
|
|
|
firstParamObj := firstParamElem.Obj()
|
|
if firstParamObj.Pkg().Path() != "testing" {
|
|
return false
|
|
}
|
|
|
|
firstParamName := firstParamObj.Id()
|
|
return (firstParamName == "T" && testMatcher.MatchString(fn.Name.Name)) ||
|
|
(firstParamName == "B" && benchMatcher.MatchString(fn.Name.Name))
|
|
}
|
|
|
|
func goGenerateCodeLens(ctx context.Context, snapshot Snapshot, fh FileHandle, f *ast.File, m *protocol.ColumnMapper) ([]protocol.CodeLens, error) {
|
|
const ggDirective = "//go:generate"
|
|
for _, c := range f.Comments {
|
|
for _, l := range c.List {
|
|
if !strings.HasPrefix(l.Text, ggDirective) {
|
|
continue
|
|
}
|
|
fset := snapshot.View().Session().Cache().FileSet()
|
|
rng, err := newMappedRange(fset, m, l.Pos(), l.Pos()+token.Pos(len(ggDirective))).Range()
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
dir := filepath.Dir(fh.Identity().URI.Filename())
|
|
return []protocol.CodeLens{
|
|
{
|
|
Range: rng,
|
|
Command: protocol.Command{
|
|
Title: "run go generate",
|
|
Command: CommandGenerate,
|
|
Arguments: []interface{}{dir, false},
|
|
},
|
|
},
|
|
{
|
|
Range: rng,
|
|
Command: protocol.Command{
|
|
Title: "run go generate ./...",
|
|
Command: CommandGenerate,
|
|
Arguments: []interface{}{dir, true},
|
|
},
|
|
},
|
|
}, nil
|
|
|
|
}
|
|
}
|
|
return nil, nil
|
|
}
|
|
|
|
func regenerateCgoLens(ctx context.Context, snapshot Snapshot, fh FileHandle, f *ast.File, m *protocol.ColumnMapper) ([]protocol.CodeLens, error) {
|
|
var c *ast.ImportSpec
|
|
for _, imp := range f.Imports {
|
|
if imp.Path.Value == `"C"` {
|
|
c = imp
|
|
}
|
|
}
|
|
if c == nil {
|
|
return nil, nil
|
|
}
|
|
fset := snapshot.View().Session().Cache().FileSet()
|
|
rng, err := newMappedRange(fset, m, c.Pos(), c.EndPos).Range()
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
return []protocol.CodeLens{
|
|
{
|
|
Range: rng,
|
|
Command: protocol.Command{
|
|
Title: "regenerate cgo definitions",
|
|
Command: CommandRegenerateCgo,
|
|
Arguments: []interface{}{fh.Identity().URI},
|
|
},
|
|
},
|
|
}, nil
|
|
}
|