gocomplete: run only runnable go files
when typing 'go run', the completion will complete only go files which are in main package and have a main function.
This commit is contained in:
parent
e00c0546bc
commit
ff8cd4ed39
|
@ -5,7 +5,7 @@ import "github.com/posener/complete"
|
||||||
|
|
||||||
var (
|
var (
|
||||||
ellipsis = complete.PredictSet("./...")
|
ellipsis = complete.PredictSet("./...")
|
||||||
anyPackage = predictPackages("")
|
anyPackage = complete.PredictFunc(predictPackages)
|
||||||
goFiles = complete.PredictFiles("*.go")
|
goFiles = complete.PredictFiles("*.go")
|
||||||
anyFile = complete.PredictFiles("*")
|
anyFile = complete.PredictFiles("*")
|
||||||
anyGo = complete.PredictOr(goFiles, anyPackage, ellipsis)
|
anyGo = complete.PredictOr(goFiles, anyPackage, ellipsis)
|
||||||
|
@ -44,7 +44,7 @@ func main() {
|
||||||
Flags: complete.Flags{
|
Flags: complete.Flags{
|
||||||
"-exec": complete.PredictAnything,
|
"-exec": complete.PredictAnything,
|
||||||
},
|
},
|
||||||
Args: goFiles,
|
Args: complete.PredictFunc(predictRunnableFiles),
|
||||||
}
|
}
|
||||||
|
|
||||||
test := complete.Command{
|
test := complete.Command{
|
||||||
|
@ -53,14 +53,14 @@ func main() {
|
||||||
"-c": complete.PredictNothing,
|
"-c": complete.PredictNothing,
|
||||||
"-exec": complete.PredictAnything,
|
"-exec": complete.PredictAnything,
|
||||||
|
|
||||||
"-bench": predictTest("Benchmark"),
|
"-bench": predictBenchmark,
|
||||||
"-benchtime": complete.PredictAnything,
|
"-benchtime": complete.PredictAnything,
|
||||||
"-count": complete.PredictAnything,
|
"-count": complete.PredictAnything,
|
||||||
"-cover": complete.PredictNothing,
|
"-cover": complete.PredictNothing,
|
||||||
"-covermode": complete.PredictSet("set", "count", "atomic"),
|
"-covermode": complete.PredictSet("set", "count", "atomic"),
|
||||||
"-coverpkg": complete.PredictDirs("*"),
|
"-coverpkg": complete.PredictDirs("*"),
|
||||||
"-cpu": complete.PredictAnything,
|
"-cpu": complete.PredictAnything,
|
||||||
"-run": predictTest("Test", "Example"),
|
"-run": predictTest,
|
||||||
"-short": complete.PredictNothing,
|
"-short": complete.PredictNothing,
|
||||||
"-timeout": complete.PredictAnything,
|
"-timeout": complete.PredictAnything,
|
||||||
|
|
||||||
|
|
|
@ -0,0 +1,28 @@
|
||||||
|
package main
|
||||||
|
|
||||||
|
import (
|
||||||
|
"go/ast"
|
||||||
|
"go/parser"
|
||||||
|
"go/token"
|
||||||
|
"regexp"
|
||||||
|
|
||||||
|
"github.com/posener/complete"
|
||||||
|
)
|
||||||
|
|
||||||
|
func functionsInFile(path string, regexp *regexp.Regexp) (tests []string) {
|
||||||
|
fset := token.NewFileSet()
|
||||||
|
f, err := parser.ParseFile(fset, path, nil, 0)
|
||||||
|
if err != nil {
|
||||||
|
complete.Log("Failed parsing %s: %s", path, err)
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
for _, d := range f.Decls {
|
||||||
|
if f, ok := d.(*ast.FuncDecl); ok {
|
||||||
|
name := f.Name.String()
|
||||||
|
if regexp == nil || regexp.MatchString(name) {
|
||||||
|
tests = append(tests, name)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return
|
||||||
|
}
|
|
@ -4,37 +4,59 @@ import (
|
||||||
"bytes"
|
"bytes"
|
||||||
"encoding/json"
|
"encoding/json"
|
||||||
"os/exec"
|
"os/exec"
|
||||||
|
"path/filepath"
|
||||||
|
"regexp"
|
||||||
"strings"
|
"strings"
|
||||||
|
|
||||||
"github.com/posener/complete"
|
"github.com/posener/complete"
|
||||||
)
|
)
|
||||||
|
|
||||||
const goListFormat = `'{"name": "{{.Name}}", "dir": "{{.Dir}}"}'`
|
const goListFormat = `{"Name": "{{.Name}}", "Path": "{{.Dir}}", "FilesString": "{{.GoFiles}}"}`
|
||||||
|
|
||||||
func predictPackages(packageName string) complete.Predictor {
|
// regexp matches a main function
|
||||||
return complete.PredictFunc(func(a complete.Args) (prediction []string) {
|
var reMainFunc = regexp.MustCompile("^main$")
|
||||||
|
|
||||||
|
func predictPackages(a complete.Args) (prediction []string) {
|
||||||
dir := a.Directory()
|
dir := a.Directory()
|
||||||
dir = strings.TrimRight(dir, "/.") + "/..."
|
|
||||||
|
|
||||||
pkgs := listPackages(dir)
|
pkgs := listPackages(dir)
|
||||||
|
|
||||||
files := make([]string, 0, len(pkgs))
|
files := make([]string, 0, len(pkgs))
|
||||||
for _, p := range pkgs {
|
for _, p := range pkgs {
|
||||||
if packageName != "" && p.Name != packageName {
|
|
||||||
continue
|
|
||||||
}
|
|
||||||
files = append(files, p.Path)
|
files = append(files, p.Path)
|
||||||
}
|
}
|
||||||
return complete.PredictFilesSet(files).Predict(a)
|
return complete.PredictFilesSet(files).Predict(a)
|
||||||
})
|
}
|
||||||
|
|
||||||
|
func predictRunnableFiles(a complete.Args) (prediction []string) {
|
||||||
|
dir := a.Directory()
|
||||||
|
pkgs := listPackages(dir)
|
||||||
|
|
||||||
|
files := []string{}
|
||||||
|
for _, p := range pkgs {
|
||||||
|
// filter non main pacakges
|
||||||
|
if p.Name != "main" {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
for _, f := range p.Files {
|
||||||
|
path := filepath.Join(p.Path, f)
|
||||||
|
if len(functionsInFile(path, reMainFunc)) > 0 {
|
||||||
|
files = append(files, path)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
complete.Log("FILES: %s", files)
|
||||||
|
return complete.PredictFilesSet(files).Predict(a)
|
||||||
}
|
}
|
||||||
|
|
||||||
type pack struct {
|
type pack struct {
|
||||||
Name string
|
Name string
|
||||||
Path string
|
Path string
|
||||||
|
FilesString string
|
||||||
|
Files []string
|
||||||
}
|
}
|
||||||
|
|
||||||
func listPackages(dir string) (pkgs []pack) {
|
func listPackages(dir string) (pkgs []pack) {
|
||||||
|
dir = strings.TrimRight(dir, "/") + "/..."
|
||||||
out, err := exec.Command("go", "list", "-f", goListFormat, dir).Output()
|
out, err := exec.Command("go", "list", "-f", goListFormat, dir).Output()
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return
|
return
|
||||||
|
@ -42,9 +64,13 @@ func listPackages(dir string) (pkgs []pack) {
|
||||||
lines := bytes.Split(out, []byte("\n"))
|
lines := bytes.Split(out, []byte("\n"))
|
||||||
for _, line := range lines {
|
for _, line := range lines {
|
||||||
var p pack
|
var p pack
|
||||||
if err := json.Unmarshal(line, &p); err == nil {
|
err := json.Unmarshal(line, &p)
|
||||||
pkgs = append(pkgs, p)
|
if err != nil {
|
||||||
|
continue
|
||||||
}
|
}
|
||||||
|
// parse the FileString from a string "[file1 file2 file3]" to a list of files
|
||||||
|
p.Files = strings.Split(strings.Trim(p.FilesString, "[]"), " ")
|
||||||
|
pkgs = append(pkgs, p)
|
||||||
}
|
}
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,25 +1,28 @@
|
||||||
package main
|
package main
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"go/ast"
|
|
||||||
"go/parser"
|
|
||||||
"go/token"
|
|
||||||
"os"
|
"os"
|
||||||
"path/filepath"
|
"path/filepath"
|
||||||
|
"regexp"
|
||||||
"strings"
|
"strings"
|
||||||
|
|
||||||
"github.com/posener/complete"
|
"github.com/posener/complete"
|
||||||
"github.com/posener/complete/match"
|
"github.com/posener/complete/match"
|
||||||
)
|
)
|
||||||
|
|
||||||
|
var (
|
||||||
|
predictBenchmark = funcPredict(regexp.MustCompile("^Benchmark"))
|
||||||
|
predictTest = funcPredict(regexp.MustCompile("^(Test|Example)"))
|
||||||
|
)
|
||||||
|
|
||||||
// predictTest predict test names.
|
// predictTest predict test names.
|
||||||
// it searches in the current directory for all the go test files
|
// it searches in the current directory for all the go test files
|
||||||
// and then all the relevant function names.
|
// and then all the relevant function names.
|
||||||
// for test names use prefix of 'Test' or 'Example', and for benchmark
|
// for test names use prefix of 'Test' or 'Example', and for benchmark
|
||||||
// test names use 'Benchmark'
|
// test names use 'Benchmark'
|
||||||
func predictTest(funcPrefix ...string) complete.Predictor {
|
func funcPredict(funcRegexp *regexp.Regexp) complete.Predictor {
|
||||||
return complete.PredictFunc(func(a complete.Args) (prediction []string) {
|
return complete.PredictFunc(func(a complete.Args) (prediction []string) {
|
||||||
tests := testNames(funcPrefix)
|
tests := funcNames(funcRegexp)
|
||||||
for _, t := range tests {
|
for _, t := range tests {
|
||||||
if match.Prefix(t, a.Last) {
|
if match.Prefix(t, a.Last) {
|
||||||
prediction = append(prediction, t)
|
prediction = append(prediction, t)
|
||||||
|
@ -30,36 +33,15 @@ func predictTest(funcPrefix ...string) complete.Predictor {
|
||||||
}
|
}
|
||||||
|
|
||||||
// get all test names in current directory
|
// get all test names in current directory
|
||||||
func testNames(funcPrefix []string) (tests []string) {
|
func funcNames(funcRegexp *regexp.Regexp) (tests []string) {
|
||||||
filepath.Walk("./", func(path string, info os.FileInfo, err error) error {
|
filepath.Walk("./", func(path string, info os.FileInfo, err error) error {
|
||||||
// if not a test file, skip
|
// if not a test file, skip
|
||||||
if !strings.HasSuffix(path, "_test.go") {
|
if !strings.HasSuffix(path, "_test.go") {
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
// inspect test file and append all the test names
|
// inspect test file and append all the test names
|
||||||
tests = append(tests, testsInFile(funcPrefix, path)...)
|
tests = append(tests, functionsInFile(path, funcRegexp)...)
|
||||||
return nil
|
return nil
|
||||||
})
|
})
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
func testsInFile(funcPrefix []string, path string) (tests []string) {
|
|
||||||
fset := token.NewFileSet()
|
|
||||||
f, err := parser.ParseFile(fset, path, nil, 0)
|
|
||||||
if err != nil {
|
|
||||||
complete.Log("Failed parsing %s: %s", path, err)
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
for _, d := range f.Decls {
|
|
||||||
if f, ok := d.(*ast.FuncDecl); ok {
|
|
||||||
name := f.Name.String()
|
|
||||||
for _, prefix := range funcPrefix {
|
|
||||||
if strings.HasPrefix(name, prefix) {
|
|
||||||
tests = append(tests, name)
|
|
||||||
break
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
|
@ -0,0 +1,84 @@
|
||||||
|
package main
|
||||||
|
|
||||||
|
import (
|
||||||
|
"testing"
|
||||||
|
|
||||||
|
"github.com/posener/complete"
|
||||||
|
)
|
||||||
|
|
||||||
|
func TestPredictions(t *testing.T) {
|
||||||
|
t.Parallel()
|
||||||
|
|
||||||
|
tests := []struct {
|
||||||
|
name string
|
||||||
|
predictor complete.Predictor
|
||||||
|
last string
|
||||||
|
completion []string
|
||||||
|
}{
|
||||||
|
{
|
||||||
|
name: "predict tests ok",
|
||||||
|
predictor: predictTest,
|
||||||
|
completion: []string{"TestPredictions", "Example"},
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: "predict tests not found",
|
||||||
|
predictor: predictTest,
|
||||||
|
last: "X",
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: "predict benchmark ok",
|
||||||
|
predictor: predictBenchmark,
|
||||||
|
completion: []string{"BenchmarkFake"},
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: "predict benchmarks not found",
|
||||||
|
predictor: predictBenchmark,
|
||||||
|
last: "X",
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: "predict packages ok",
|
||||||
|
predictor: complete.PredictFunc(predictPackages),
|
||||||
|
completion: []string{"./"},
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: "predict packages not found",
|
||||||
|
predictor: complete.PredictFunc(predictPackages),
|
||||||
|
last: "X",
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: "predict runnable ok",
|
||||||
|
predictor: complete.PredictFunc(predictRunnableFiles),
|
||||||
|
completion: []string{"./complete.go"},
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: "predict runnable not found",
|
||||||
|
predictor: complete.PredictFunc(predictRunnableFiles),
|
||||||
|
last: "X",
|
||||||
|
},
|
||||||
|
}
|
||||||
|
|
||||||
|
for _, tt := range tests {
|
||||||
|
t.Run(tt.name, func(t *testing.T) {
|
||||||
|
a := complete.Args{Last: tt.last}
|
||||||
|
got := tt.predictor.Predict(a)
|
||||||
|
if want := tt.completion; !equal(got, want) {
|
||||||
|
t.Errorf("Failed %s: completion = %q, want %q", t.Name(), got, want)
|
||||||
|
}
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func BenchmarkFake(b *testing.B) {}
|
||||||
|
func Example() {}
|
||||||
|
|
||||||
|
func equal(s1, s2 []string) bool {
|
||||||
|
if len(s1) != len(s2) {
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
for i := range s1 {
|
||||||
|
if s1[i] != s2[i] {
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return true
|
||||||
|
}
|
Loading…
Reference in New Issue