| // Copyright 2021 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 misc |
| |
| import ( |
| "fmt" |
| "regexp" |
| "strings" |
| "testing" |
| |
| "github.com/google/go-cmp/cmp" |
| "golang.org/x/tools/gopls/internal/protocol" |
| . "golang.org/x/tools/gopls/internal/test/integration" |
| "golang.org/x/tools/gopls/internal/test/integration/fake" |
| "golang.org/x/tools/internal/testenv" |
| ) |
| |
| func TestHoverUnexported(t *testing.T) { |
| const proxy = ` |
| -- golang.org/x/structs@v1.0.0/go.mod -- |
| module golang.org/x/structs |
| |
| go 1.12 |
| |
| -- golang.org/x/structs@v1.0.0/types.go -- |
| package structs |
| |
| type Mixed struct { |
| // Exported comment |
| Exported int |
| unexported string |
| } |
| |
| func printMixed(m Mixed) { |
| println(m) |
| } |
| ` |
| const mod = ` |
| -- go.mod -- |
| module mod.com |
| |
| go 1.12 |
| |
| require golang.org/x/structs v1.0.0 |
| -- go.sum -- |
| golang.org/x/structs v1.0.0 h1:Ito/a7hBYZaNKShFrZKjfBA/SIPvmBrcPCBWPx5QeKk= |
| golang.org/x/structs v1.0.0/go.mod h1:47gkSIdo5AaQaWJS0upVORsxfEr1LL1MWv9dmYF3iq4= |
| -- main.go -- |
| package main |
| |
| import "golang.org/x/structs" |
| |
| func main() { |
| var m structs.Mixed |
| _ = m.Exported |
| } |
| ` |
| |
| // TODO: use a nested workspace folder here. |
| WithOptions( |
| ProxyFiles(proxy), |
| ).Run(t, mod, func(t *testing.T, env *Env) { |
| env.OpenFile("main.go") |
| mixedLoc := env.RegexpSearch("main.go", "Mixed") |
| got, _ := env.Hover(mixedLoc) |
| if !strings.Contains(got.Value, "unexported") { |
| t.Errorf("Workspace hover: missing expected field 'unexported'. Got:\n%q", got.Value) |
| } |
| |
| cacheLoc := env.GoToDefinition(mixedLoc) |
| cacheFile := env.Sandbox.Workdir.URIToPath(cacheLoc.URI) |
| argLoc := env.RegexpSearch(cacheFile, "printMixed.*(Mixed)") |
| got, _ = env.Hover(argLoc) |
| if !strings.Contains(got.Value, "unexported") { |
| t.Errorf("Non-workspace hover: missing expected field 'unexported'. Got:\n%q", got.Value) |
| } |
| |
| exportedFieldLoc := env.RegexpSearch("main.go", "Exported") |
| got, _ = env.Hover(exportedFieldLoc) |
| if !strings.Contains(got.Value, "comment") { |
| t.Errorf("Workspace hover: missing comment for field 'Exported'. Got:\n%q", got.Value) |
| } |
| }) |
| } |
| |
| func TestHoverIntLiteral(t *testing.T) { |
| const source = ` |
| -- main.go -- |
| package main |
| |
| var ( |
| bigBin = 0b1001001 |
| ) |
| |
| var hex = 0xe34e |
| |
| func main() { |
| } |
| ` |
| Run(t, source, func(t *testing.T, env *Env) { |
| env.OpenFile("main.go") |
| hexExpected := "58190" |
| got, _ := env.Hover(env.RegexpSearch("main.go", "0xe")) |
| if got != nil && !strings.Contains(got.Value, hexExpected) { |
| t.Errorf("Hover: missing expected field '%s'. Got:\n%q", hexExpected, got.Value) |
| } |
| |
| binExpected := "73" |
| got, _ = env.Hover(env.RegexpSearch("main.go", "0b1")) |
| if got != nil && !strings.Contains(got.Value, binExpected) { |
| t.Errorf("Hover: missing expected field '%s'. Got:\n%q", binExpected, got.Value) |
| } |
| }) |
| } |
| |
| // Tests that hovering does not trigger the panic in golang/go#48249. |
| func TestPanicInHoverBrokenCode(t *testing.T) { |
| // Note: this test can not be expressed as a marker test, as it must use |
| // content without a trailing newline. |
| const source = ` |
| -- main.go -- |
| package main |
| |
| type Example struct` |
| Run(t, source, func(t *testing.T, env *Env) { |
| env.OpenFile("main.go") |
| env.Editor.Hover(env.Ctx, env.RegexpSearch("main.go", "Example")) |
| }) |
| } |
| |
| func TestHoverRune_48492(t *testing.T) { |
| const files = ` |
| -- go.mod -- |
| module mod.com |
| |
| go 1.18 |
| -- main.go -- |
| package main |
| ` |
| Run(t, files, func(t *testing.T, env *Env) { |
| env.OpenFile("main.go") |
| env.EditBuffer("main.go", fake.NewEdit(0, 0, 1, 0, "package main\nfunc main() {\nconst x = `\nfoo\n`\n}")) |
| env.Editor.Hover(env.Ctx, env.RegexpSearch("main.go", "foo")) |
| }) |
| } |
| |
| func TestHoverImport(t *testing.T) { |
| const packageDoc1 = "Package lib1 hover documentation" |
| const packageDoc2 = "Package lib2 hover documentation" |
| tests := []struct { |
| hoverPackage string |
| want string |
| wantError bool |
| }{ |
| { |
| "mod.com/lib1", |
| packageDoc1, |
| false, |
| }, |
| { |
| "mod.com/lib2", |
| packageDoc2, |
| false, |
| }, |
| { |
| "mod.com/lib3", |
| "", |
| false, |
| }, |
| { |
| "mod.com/lib4", |
| "", |
| true, |
| }, |
| } |
| source := fmt.Sprintf(` |
| -- go.mod -- |
| module mod.com |
| |
| go 1.12 |
| -- lib1/a.go -- |
| // %s |
| package lib1 |
| |
| const C = 1 |
| |
| -- lib1/b.go -- |
| package lib1 |
| |
| const D = 1 |
| |
| -- lib2/a.go -- |
| // %s |
| package lib2 |
| |
| const E = 1 |
| |
| -- lib3/a.go -- |
| package lib3 |
| |
| const F = 1 |
| |
| -- main.go -- |
| package main |
| |
| import ( |
| "mod.com/lib1" |
| "mod.com/lib2" |
| "mod.com/lib3" |
| "mod.com/lib4" |
| ) |
| |
| func main() { |
| println("Hello") |
| } |
| `, packageDoc1, packageDoc2) |
| Run(t, source, func(t *testing.T, env *Env) { |
| env.OpenFile("main.go") |
| for _, test := range tests { |
| got, _, err := env.Editor.Hover(env.Ctx, env.RegexpSearch("main.go", test.hoverPackage)) |
| if test.wantError { |
| if err == nil { |
| t.Errorf("Hover(%q) succeeded unexpectedly", test.hoverPackage) |
| } |
| } else if !strings.Contains(got.Value, test.want) { |
| t.Errorf("Hover(%q): got:\n%q\nwant:\n%q", test.hoverPackage, got.Value, test.want) |
| } |
| } |
| }) |
| } |
| |
| // for x/tools/gopls: unhandled named anchor on the hover #57048 |
| func TestHoverTags(t *testing.T) { |
| const source = ` |
| -- go.mod -- |
| module mod.com |
| |
| go 1.19 |
| |
| -- lib/a.go -- |
| |
| // variety of execution modes. |
| // |
| // # Test package setup |
| // |
| // The regression test package uses a couple of uncommon patterns to reduce |
| package lib |
| |
| -- a.go -- |
| package main |
| import "mod.com/lib" |
| |
| const A = 1 |
| |
| } |
| ` |
| Run(t, source, func(t *testing.T, env *Env) { |
| t.Run("tags", func(t *testing.T) { |
| env.OpenFile("a.go") |
| z := env.RegexpSearch("a.go", "lib") |
| t.Logf("%#v", z) |
| got, _ := env.Hover(env.RegexpSearch("a.go", "lib")) |
| if strings.Contains(got.Value, "{#hdr-") { |
| t.Errorf("Hover: got {#hdr- tag:\n%q", got) |
| } |
| }) |
| }) |
| } |
| |
| // This is a regression test for Go issue #57625. |
| func TestHoverModMissingModuleStmt(t *testing.T) { |
| const source = ` |
| -- go.mod -- |
| go 1.16 |
| ` |
| Run(t, source, func(t *testing.T, env *Env) { |
| env.OpenFile("go.mod") |
| env.Hover(env.RegexpSearch("go.mod", "go")) // no panic |
| }) |
| } |
| |
| func TestHoverCompletionMarkdown(t *testing.T) { |
| testenv.NeedsGo1Point(t, 19) |
| const source = ` |
| -- go.mod -- |
| module mod.com |
| go 1.19 |
| -- main.go -- |
| package main |
| // Just says [hello]. |
| // |
| // [hello]: https://en.wikipedia.org/wiki/Hello |
| func Hello() string { |
| Hello() //Here |
| return "hello" |
| } |
| ` |
| Run(t, source, func(t *testing.T, env *Env) { |
| // Hover, Completion, and SignatureHelp should all produce markdown |
| // check that the markdown for SignatureHelp and Completion are |
| // the same, and contained in that for Hover (up to trailing \n) |
| env.OpenFile("main.go") |
| loc := env.RegexpSearch("main.go", "func (Hello)") |
| hover, _ := env.Hover(loc) |
| hoverContent := hover.Value |
| |
| loc = env.RegexpSearch("main.go", "//Here") |
| loc.Range.Start.Character -= 3 // Hello(_) //Here |
| completions := env.Completion(loc) |
| signatures := env.SignatureHelp(loc) |
| |
| if len(completions.Items) != 1 { |
| t.Errorf("got %d completions, expected 1", len(completions.Items)) |
| } |
| if len(signatures.Signatures) != 1 { |
| t.Errorf("got %d signatures, expected 1", len(signatures.Signatures)) |
| } |
| item := completions.Items[0].Documentation.Value |
| var itemContent string |
| if x, ok := item.(protocol.MarkupContent); !ok || x.Kind != protocol.Markdown { |
| t.Fatalf("%#v is not markdown", item) |
| } else { |
| itemContent = strings.Trim(x.Value, "\n") |
| } |
| sig := signatures.Signatures[0].Documentation.Value |
| var sigContent string |
| if x, ok := sig.(protocol.MarkupContent); !ok || x.Kind != protocol.Markdown { |
| t.Fatalf("%#v is not markdown", item) |
| } else { |
| sigContent = x.Value |
| } |
| if itemContent != sigContent { |
| t.Errorf("item:%q not sig:%q", itemContent, sigContent) |
| } |
| if !strings.Contains(hoverContent, itemContent) { |
| t.Errorf("hover:%q does not containt sig;%q", hoverContent, sigContent) |
| } |
| }) |
| } |
| |
| // Test that the generated markdown contains links for Go references. |
| // https://github.com/golang/go/issues/58352 |
| func TestHoverLinks(t *testing.T) { |
| testenv.NeedsGo1Point(t, 19) |
| const input = ` |
| -- go.mod -- |
| go 1.19 |
| module mod.com |
| -- main.go -- |
| package main |
| // [fmt] |
| var A int |
| // [fmt.Println] |
| var B int |
| // [golang.org/x/tools/go/packages.Package.String] |
| var C int |
| ` |
| var tests = []struct { |
| pat string |
| ans string |
| }{ |
| {"A", "fmt"}, |
| {"B", "fmt#Println"}, |
| {"C", "golang.org/x/tools/go/packages#Package.String"}, |
| } |
| for _, test := range tests { |
| Run(t, input, func(t *testing.T, env *Env) { |
| env.OpenFile("main.go") |
| loc := env.RegexpSearch("main.go", test.pat) |
| hover, _ := env.Hover(loc) |
| hoverContent := hover.Value |
| want := fmt.Sprintf("%s/%s", "https://pkg.go.dev", test.ans) |
| if !strings.Contains(hoverContent, want) { |
| t.Errorf("hover:%q does not contain link %q", hoverContent, want) |
| } |
| }) |
| } |
| } |
| |
| const linknameHover = ` |
| -- go.mod -- |
| module mod.com |
| |
| -- upper/upper.go -- |
| package upper |
| |
| import ( |
| _ "unsafe" |
| _ "mod.com/lower" |
| ) |
| |
| //go:linkname foo mod.com/lower.bar |
| func foo() string |
| |
| -- lower/lower.go -- |
| package lower |
| |
| // bar does foo. |
| func bar() string { |
| return "foo by bar" |
| }` |
| |
| func TestHoverLinknameDirective(t *testing.T) { |
| Run(t, linknameHover, func(t *testing.T, env *Env) { |
| // Jump from directives 2nd arg. |
| env.OpenFile("upper/upper.go") |
| from := env.RegexpSearch("upper/upper.go", `lower.bar`) |
| |
| hover, _ := env.Hover(from) |
| content := hover.Value |
| |
| expect := "bar does foo" |
| if !strings.Contains(content, expect) { |
| t.Errorf("hover: %q does not contain: %q", content, expect) |
| } |
| }) |
| } |
| |
| func TestHoverGoWork_Issue60821(t *testing.T) { |
| const files = ` |
| -- go.work -- |
| go 1.19 |
| |
| use ( |
| moda |
| modb |
| ) |
| -- moda/go.mod -- |
| |
| ` |
| Run(t, files, func(t *testing.T, env *Env) { |
| env.OpenFile("go.work") |
| // Neither of the requests below should crash gopls. |
| _, _, _ = env.Editor.Hover(env.Ctx, env.RegexpSearch("go.work", "moda")) |
| _, _, _ = env.Editor.Hover(env.Ctx, env.RegexpSearch("go.work", "modb")) |
| }) |
| } |
| |
| const embedHover = ` |
| -- go.mod -- |
| module mod.com |
| go 1.19 |
| -- main.go -- |
| package main |
| |
| import "embed" |
| |
| //go:embed *.txt |
| var foo embed.FS |
| |
| func main() { |
| } |
| -- foo.txt -- |
| FOO |
| -- bar.txt -- |
| BAR |
| -- baz.txt -- |
| BAZ |
| -- other.sql -- |
| SKIPPED |
| -- dir.txt/skip.txt -- |
| SKIPPED |
| ` |
| |
| func TestHoverEmbedDirective(t *testing.T) { |
| testenv.NeedsGo1Point(t, 19) |
| Run(t, embedHover, func(t *testing.T, env *Env) { |
| env.OpenFile("main.go") |
| from := env.RegexpSearch("main.go", `\*.txt`) |
| |
| got, _ := env.Hover(from) |
| if got == nil { |
| t.Fatalf("hover over //go:embed arg not found") |
| } |
| content := got.Value |
| |
| wants := []string{"foo.txt", "bar.txt", "baz.txt"} |
| for _, want := range wants { |
| if !strings.Contains(content, want) { |
| t.Errorf("hover: %q does not contain: %q", content, want) |
| } |
| } |
| |
| // A directory should never be matched, even if it happens to have a matching name. |
| // Content in subdirectories should not match on only one asterisk. |
| skips := []string{"other.sql", "dir.txt", "skip.txt"} |
| for _, skip := range skips { |
| if strings.Contains(content, skip) { |
| t.Errorf("hover: %q should not contain: %q", content, skip) |
| } |
| } |
| }) |
| } |
| |
| func TestHoverBrokenImport_Issue60592(t *testing.T) { |
| const files = ` |
| -- go.mod -- |
| module testdata |
| go 1.18 |
| |
| -- p.go -- |
| package main |
| |
| import foo "a" |
| |
| func _() { |
| foo.Print() |
| } |
| |
| ` |
| Run(t, files, func(t *testing.T, env *Env) { |
| env.OpenFile("p.go") |
| // This request should not crash gopls. |
| _, _, _ = env.Editor.Hover(env.Ctx, env.RegexpSearch("p.go", "foo[.]")) |
| }) |
| } |
| |
| func TestHoverInternalLinks(t *testing.T) { |
| const src = ` |
| -- main.go -- |
| package main |
| |
| import "errors" |
| |
| func main() { |
| errors.New("oops") |
| } |
| ` |
| for _, test := range []struct { |
| linksInHover any // JSON configuration value |
| wantRE string // pattern to match the Hover Markdown output |
| }{ |
| { |
| true, // default: use options.LinkTarget domain |
| regexp.QuoteMeta("[`errors.New` on pkg.go.dev](https://pkg.go.dev/errors#New)"), |
| }, |
| { |
| "gopls", // use gopls' internal viewer |
| "\\[`errors.New` in gopls doc viewer\\]\\(http://127.0.0.1:[0-9]+/gopls/[^/]+/pkg/errors\\?view=[0-9]+#New\\)", |
| }, |
| } { |
| WithOptions( |
| Settings{"linksInHover": test.linksInHover}, |
| ).Run(t, src, func(t *testing.T, env *Env) { |
| env.OpenFile("main.go") |
| got, _ := env.Hover(env.RegexpSearch("main.go", "New")) |
| if m, err := regexp.MatchString(test.wantRE, got.Value); err != nil { |
| t.Fatalf("bad regexp in test: %v", err) |
| } else if !m { |
| t.Fatalf("hover output does not match %q; got:\n\n%s", test.wantRE, got.Value) |
| } |
| }) |
| } |
| } |
| |
| func TestHoverInternalLinksIssue68116(t *testing.T) { |
| // Links for the internal viewer should not include a module version suffix: |
| // the package path and the view are an unambiguous key; see #68116. |
| |
| const proxy = ` |
| -- example.com@v1.2.3/go.mod -- |
| module example.com |
| |
| go 1.12 |
| |
| -- example.com@v1.2.3/a/a.go -- |
| package a |
| |
| // F is a function. |
| func F() |
| ` |
| |
| const mod = ` |
| -- go.mod -- |
| module main |
| |
| go 1.12 |
| |
| require example.com v1.2.3 |
| |
| -- main.go -- |
| package main |
| |
| import "example.com/a" |
| |
| func main() { |
| a.F() |
| } |
| ` |
| WithOptions( |
| ProxyFiles(proxy), |
| Settings{"linksInHover": "gopls"}, |
| WriteGoSum("."), |
| ).Run(t, mod, func(t *testing.T, env *Env) { |
| env.OpenFile("main.go") |
| got, _ := env.Hover(env.RegexpSearch("main.go", "F")) |
| const wantRE = "\\[`a.F` in gopls doc viewer\\]\\(http://127.0.0.1:[0-9]+/gopls/[^/]+/pkg/example.com\\?view=[0-9]+#F\\)" // no version |
| if m, err := regexp.MatchString(wantRE, got.Value); err != nil { |
| t.Fatalf("bad regexp in test: %v", err) |
| } else if !m { |
| t.Fatalf("hover output does not match %q; got:\n\n%s", wantRE, got.Value) |
| } |
| }) |
| } |
| |
| func TestHoverBuiltinFile(t *testing.T) { |
| testenv.NeedsGo1Point(t, 21) // uses 'min' |
| |
| // This test verifies that hovering in the builtin file provides the same |
| // hover content as hovering over a use of a builtin. |
| |
| const src = ` |
| -- p.go -- |
| package p |
| |
| func _() { |
| const ( |
| _ = iota |
| _ = true |
| ) |
| var ( |
| _ any |
| err error = e{} // avoid nil deref warning |
| ) |
| _ = err.Error |
| println("Hello") |
| _ = min(1, 2) |
| } |
| |
| // e implements Error, for use above. |
| type e struct{} |
| func (e) Error() string |
| ` |
| |
| // Test hovering over various builtins with different kinds of declarations. |
| tests := []string{ |
| "iota", |
| "true", |
| "any", |
| "error", |
| "Error", |
| "println", |
| "min", |
| } |
| |
| Run(t, src, func(t *testing.T, env *Env) { |
| env.OpenFile("p.go") |
| env.AfterChange(NoDiagnostics()) // avoid accidental compiler errors |
| |
| for _, builtin := range tests { |
| useLocation := env.RegexpSearch("p.go", builtin) |
| calleeHover, _ := env.Hover(useLocation) |
| declLocation := env.GoToDefinition(useLocation) |
| declHover, _ := env.Hover(declLocation) |
| if diff := cmp.Diff(calleeHover, declHover); diff != "" { |
| t.Errorf("Hover mismatch (-callee hover +decl hover):\n%s", diff) |
| } |
| } |
| }) |
| } |
| |
| func TestHoverStdlibWithAvailableVersion(t *testing.T) { |
| const src = ` |
| -- stdlib.go -- |
| package stdlib |
| |
| import "fmt" |
| import "context" |
| import "crypto" |
| import "regexp" |
| |
| type testRegexp = *regexp.Regexp |
| |
| func _() { |
| var ctx context.Context |
| ctx = context.Background() |
| if ctx.Err(); e == context.Canceled { |
| fmt.Println("Canceled") |
| fmt.Printf("%v", crypto.SHA512_224) |
| } |
| _ := fmt.Appendf(make([]byte, 100), "world, %d", 23) |
| |
| var re = regexp.MustCompile("\n{2,}") |
| copy := re.Copy() |
| var testRE testRegexp |
| testRE.Longest() |
| } |
| ` |
| |
| testcases := []struct { |
| symbolRE string // regexp matching symbol to hover over |
| shouldContain bool |
| targetString string |
| }{ |
| {"Println", false, "go1.0"}, // package-level func |
| {"Appendf", true, "go1.19"}, // package-level func |
| {"Background", true, "go1.7"}, // package-level func |
| {"Canceled", true, "go1.7"}, // package-level var |
| {"Context", true, "go1.7"}, // package-level type |
| {"SHA512_224", true, "go1.5"}, // package-level const |
| {"Copy", true, "go1.6"}, // method |
| {"Longest", true, "go1.1"}, // method with alias receiver |
| // TODO(hxjiang): add test for symbol type Field. |
| } |
| |
| Run(t, src, func(t *testing.T, env *Env) { |
| env.OpenFile("stdlib.go") |
| for _, tc := range testcases { |
| content, _ := env.Hover(env.RegexpSearch("stdlib.go", tc.symbolRE)) |
| if tc.shouldContain && !strings.Contains(content.Value, tc.targetString) { |
| t.Errorf("Hover(%q) should contain string %s", tc.symbolRE, tc.targetString) |
| } |
| if !tc.shouldContain && strings.Contains(content.Value, tc.targetString) { |
| t.Errorf("Hover(%q) should not contain string %s", tc.symbolRE, tc.targetString) |
| } |
| } |
| }) |
| } |