190 lines
		
	
	
		
			4.9 KiB
		
	
	
	
		
			Go
		
	
	
	
			
		
		
	
	
			190 lines
		
	
	
		
			4.9 KiB
		
	
	
	
		
			Go
		
	
	
	
| // 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 cmd_test
 | |
| 
 | |
| import (
 | |
| 	"context"
 | |
| 	"flag"
 | |
| 	"fmt"
 | |
| 	"os"
 | |
| 	"os/exec"
 | |
| 	"path/filepath"
 | |
| 	"regexp"
 | |
| 	"runtime"
 | |
| 	"strconv"
 | |
| 	"strings"
 | |
| 	"testing"
 | |
| 
 | |
| 	"golang.org/x/tools/go/packages/packagestest"
 | |
| 	"golang.org/x/tools/internal/lsp/cmd"
 | |
| 	"golang.org/x/tools/internal/span"
 | |
| 	"golang.org/x/tools/internal/tool"
 | |
| )
 | |
| 
 | |
| const (
 | |
| 	expectedDefinitionsCount     = 25
 | |
| 	expectedTypeDefinitionsCount = 2
 | |
| )
 | |
| 
 | |
| type definition struct {
 | |
| 	src   span.Span
 | |
| 	flags string
 | |
| 	def   span.Span
 | |
| 	match string
 | |
| }
 | |
| 
 | |
| type definitions map[span.Span]definition
 | |
| 
 | |
| var verifyGuru = flag.Bool("verify-guru", false, "Check that the guru compatability matches")
 | |
| 
 | |
| func TestDefinitionHelpExample(t *testing.T) {
 | |
| 	if runtime.GOOS == "android" {
 | |
| 		t.Skip("not all source files are available on android")
 | |
| 	}
 | |
| 	dir, err := os.Getwd()
 | |
| 	if err != nil {
 | |
| 		t.Errorf("could not get wd: %v", err)
 | |
| 		return
 | |
| 	}
 | |
| 	thisFile := filepath.Join(dir, "definition.go")
 | |
| 	baseArgs := []string{"query", "definition"}
 | |
| 	expect := regexp.MustCompile(`^[\w/\\:_-]+flag[/\\]flag.go:\d+:\d+-\d+: defined here as type flag.FlagSet struct{.*}$`)
 | |
| 	for _, query := range []string{
 | |
| 		fmt.Sprintf("%v:%v:%v", thisFile, cmd.ExampleLine, cmd.ExampleColumn),
 | |
| 		fmt.Sprintf("%v:#%v", thisFile, cmd.ExampleOffset)} {
 | |
| 		args := append(baseArgs, query)
 | |
| 		got := captureStdOut(t, func() {
 | |
| 			tool.Main(context.Background(), &cmd.Application{}, args)
 | |
| 		})
 | |
| 		if !expect.MatchString(got) {
 | |
| 			t.Errorf("test with %v\nexpected:\n%s\ngot:\n%s", args, expect, got)
 | |
| 		}
 | |
| 	}
 | |
| }
 | |
| 
 | |
| func (l definitions) godef(src, def span.Span) {
 | |
| 	l[src] = definition{
 | |
| 		src: src,
 | |
| 		def: def,
 | |
| 	}
 | |
| }
 | |
| 
 | |
| func (l definitions) typdef(src, def span.Span) {
 | |
| 	l[src] = definition{
 | |
| 		src: src,
 | |
| 		def: def,
 | |
| 	}
 | |
| }
 | |
| 
 | |
| func (l definitions) definition(src span.Span, flags string, def span.Span, match string) {
 | |
| 	l[src] = definition{
 | |
| 		src:   src,
 | |
| 		flags: flags,
 | |
| 		def:   def,
 | |
| 		match: match,
 | |
| 	}
 | |
| }
 | |
| 
 | |
| func (l definitions) testDefinitions(t *testing.T, e *packagestest.Exported) {
 | |
| 	if len(l) != expectedDefinitionsCount {
 | |
| 		t.Errorf("got %v definitions expected %v", len(l), expectedDefinitionsCount)
 | |
| 	}
 | |
| 	for _, d := range l {
 | |
| 		args := []string{"query"}
 | |
| 		if d.flags != "" {
 | |
| 			args = append(args, strings.Split(d.flags, " ")...)
 | |
| 		}
 | |
| 		args = append(args, "definition")
 | |
| 		src := span.New(d.src.URI(), span.NewPoint(0, 0, d.src.Start().Offset()), span.Point{})
 | |
| 		args = append(args, fmt.Sprint(src))
 | |
| 		app := &cmd.Application{}
 | |
| 		app.Config = *e.Config
 | |
| 		got := captureStdOut(t, func() {
 | |
| 			tool.Main(context.Background(), app, args)
 | |
| 		})
 | |
| 		if d.match == "" {
 | |
| 			expect := fmt.Sprint(d.def)
 | |
| 			if !strings.HasPrefix(got, expect) {
 | |
| 				t.Errorf("definition %v\nexpected:\n%s\ngot:\n%s", args, expect, got)
 | |
| 			}
 | |
| 		} else {
 | |
| 			expect := os.Expand(d.match, func(name string) string {
 | |
| 				switch name {
 | |
| 				case "file":
 | |
| 					fname, _ := d.def.URI().Filename()
 | |
| 					return fname
 | |
| 				case "efile":
 | |
| 					fname, _ := d.def.URI().Filename()
 | |
| 					qfile := strconv.Quote(fname)
 | |
| 					return qfile[1 : len(qfile)-1]
 | |
| 				case "euri":
 | |
| 					quri := strconv.Quote(string(d.def.URI()))
 | |
| 					return quri[1 : len(quri)-1]
 | |
| 				case "line":
 | |
| 					return fmt.Sprint(d.def.Start().Line())
 | |
| 				case "col":
 | |
| 					return fmt.Sprint(d.def.Start().Column())
 | |
| 				case "offset":
 | |
| 					return fmt.Sprint(d.def.Start().Offset())
 | |
| 				case "eline":
 | |
| 					return fmt.Sprint(d.def.End().Line())
 | |
| 				case "ecol":
 | |
| 					return fmt.Sprint(d.def.End().Column())
 | |
| 				case "eoffset":
 | |
| 					return fmt.Sprint(d.def.End().Offset())
 | |
| 				default:
 | |
| 					return name
 | |
| 				}
 | |
| 			})
 | |
| 			if expect != got {
 | |
| 				t.Errorf("definition %v\nexpected:\n%s\ngot:\n%s", args, expect, got)
 | |
| 			}
 | |
| 			if *verifyGuru {
 | |
| 				moduleMode := e.File(e.Modules[0].Name, "go.mod") != ""
 | |
| 				var guruArgs []string
 | |
| 				runGuru := false
 | |
| 				if !moduleMode {
 | |
| 					for _, arg := range args {
 | |
| 						switch {
 | |
| 						case arg == "query":
 | |
| 							// just ignore this one
 | |
| 						case arg == "-json":
 | |
| 							guruArgs = append(guruArgs, arg)
 | |
| 						case arg == "-emulate=guru":
 | |
| 							// if we don't see this one we should not run guru
 | |
| 							runGuru = true
 | |
| 						case strings.HasPrefix(arg, "-"):
 | |
| 							// unknown flag, ignore it
 | |
| 							break
 | |
| 						default:
 | |
| 							guruArgs = append(guruArgs, arg)
 | |
| 						}
 | |
| 					}
 | |
| 				}
 | |
| 				if runGuru {
 | |
| 					cmd := exec.Command("guru", guruArgs...)
 | |
| 					cmd.Env = e.Config.Env
 | |
| 					out, err := cmd.CombinedOutput()
 | |
| 					if err != nil {
 | |
| 						t.Errorf("Could not run guru %v: %v\n%s", guruArgs, err, out)
 | |
| 					} else {
 | |
| 						guru := strings.TrimSpace(string(out))
 | |
| 						if !strings.HasPrefix(expect, guru) {
 | |
| 							t.Errorf("definition %v\nexpected:\n%s\nguru gave:\n%s", args, expect, guru)
 | |
| 						}
 | |
| 					}
 | |
| 				}
 | |
| 			}
 | |
| 		}
 | |
| 	}
 | |
| }
 | |
| 
 | |
| func (l definitions) testTypeDefinitions(t *testing.T, e *packagestest.Exported) {
 | |
| 	if len(l) != expectedTypeDefinitionsCount {
 | |
| 		t.Errorf("got %v definitions expected %v", len(l), expectedTypeDefinitionsCount)
 | |
| 	}
 | |
| 	//TODO: add command line type definition tests when it works
 | |
| }
 |