blob: ac74e6deed530a55ce9294b8270703d8d6d17198 [file] [log] [blame]
// 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 workspace
import (
"context"
"fmt"
"os"
"sort"
"strings"
"testing"
"github.com/google/go-cmp/cmp"
"github.com/google/go-cmp/cmp/cmpopts"
"golang.org/x/tools/gopls/internal/cache"
"golang.org/x/tools/gopls/internal/protocol"
"golang.org/x/tools/gopls/internal/protocol/command"
"golang.org/x/tools/gopls/internal/test/integration/fake"
"golang.org/x/tools/gopls/internal/util/bug"
"golang.org/x/tools/gopls/internal/util/goversion"
"golang.org/x/tools/internal/gocommand"
"golang.org/x/tools/internal/testenv"
. "golang.org/x/tools/gopls/internal/test/integration"
)
func TestMain(m *testing.M) {
bug.PanicOnBugs = true
os.Exit(Main(m))
}
const workspaceProxy = `
-- example.com@v1.2.3/go.mod --
module example.com
go 1.12
-- example.com@v1.2.3/blah/blah.go --
package blah
import "fmt"
func SaySomething() {
fmt.Println("something")
}
-- random.org@v1.2.3/go.mod --
module random.org
go 1.12
-- random.org@v1.2.3/bye/bye.go --
package bye
func Goodbye() {
println("Bye")
}
`
// TODO: Add a replace directive.
const workspaceModule = `
-- pkg/go.mod --
module mod.com
go 1.14
require (
example.com v1.2.3
random.org v1.2.3
)
-- pkg/go.sum --
example.com v1.2.3 h1:veRD4tUnatQRgsULqULZPjeoBGFr2qBhevSCZllD2Ds=
example.com v1.2.3/go.mod h1:Y2Rc5rVWjWur0h3pd9aEvK5Pof8YKDANh9gHA2Maujo=
random.org v1.2.3 h1:+JE2Fkp7gS0zsHXGEQJ7hraom3pNTlkxC4b2qPfA+/Q=
random.org v1.2.3/go.mod h1:E9KM6+bBX2g5ykHZ9H27w16sWo3QwgonyjM44Dnej3I=
-- pkg/main.go --
package main
import (
"example.com/blah"
"mod.com/inner"
"random.org/bye"
)
func main() {
blah.SaySomething()
inner.Hi()
bye.Goodbye()
}
-- pkg/main2.go --
package main
import "fmt"
func _() {
fmt.Print("%s")
}
-- pkg/inner/inner.go --
package inner
import "example.com/blah"
func Hi() {
blah.SaySomething()
}
-- goodbye/bye/bye.go --
package bye
func Bye() {}
-- goodbye/go.mod --
module random.org
go 1.12
`
// Confirm that find references returns all of the references in the module,
// regardless of what the workspace root is.
func TestReferences(t *testing.T) {
for _, tt := range []struct {
name, rootPath string
}{
{
name: "module root",
rootPath: "pkg",
},
{
name: "subdirectory",
rootPath: "pkg/inner",
},
} {
t.Run(tt.name, func(t *testing.T) {
opts := []RunOption{ProxyFiles(workspaceProxy)}
if tt.rootPath != "" {
opts = append(opts, WorkspaceFolders(tt.rootPath))
}
WithOptions(opts...).Run(t, workspaceModule, func(t *testing.T, env *Env) {
f := "pkg/inner/inner.go"
env.OpenFile(f)
locations := env.References(env.RegexpSearch(f, `SaySomething`))
want := 3
if got := len(locations); got != want {
t.Fatalf("expected %v locations, got %v", want, got)
}
})
})
}
}
// Make sure that analysis diagnostics are cleared for the whole package when
// the only opened file is closed. This test was inspired by the experience in
// VS Code, where clicking on a reference result triggers a
// textDocument/didOpen without a corresponding textDocument/didClose.
func TestClearAnalysisDiagnostics(t *testing.T) {
WithOptions(
ProxyFiles(workspaceProxy),
WorkspaceFolders("pkg/inner"),
).Run(t, workspaceModule, func(t *testing.T, env *Env) {
env.OpenFile("pkg/main.go")
env.AfterChange(
Diagnostics(env.AtRegexp("pkg/main2.go", "fmt.Print")),
)
env.CloseBuffer("pkg/main.go")
env.AfterChange(
NoDiagnostics(ForFile("pkg/main2.go")),
)
})
}
// TestReloadOnlyOnce checks that changes to the go.mod file do not result in
// redundant package loads (golang/go#54473).
//
// Note that this test may be fragile, as it depends on specific structure to
// log messages around reinitialization. Nevertheless, it is important for
// guarding against accidentally duplicate reloading.
func TestReloadOnlyOnce(t *testing.T) {
WithOptions(
ProxyFiles(workspaceProxy),
WorkspaceFolders("pkg"),
).Run(t, workspaceModule, func(t *testing.T, env *Env) {
dir := env.Sandbox.Workdir.URI("goodbye").Path()
goModWithReplace := fmt.Sprintf(`%s
replace random.org => %s
`, env.ReadWorkspaceFile("pkg/go.mod"), dir)
env.WriteWorkspaceFile("pkg/go.mod", goModWithReplace)
env.Await(
LogMatching(protocol.Info, `packages\.Load #\d+\n`, 2, false),
)
})
}
const workspaceModuleProxy = `
-- example.com@v1.2.3/go.mod --
module example.com
go 1.12
-- example.com@v1.2.3/blah/blah.go --
package blah
import "fmt"
func SaySomething() {
fmt.Println("something")
}
-- b.com@v1.2.3/go.mod --
module b.com
go 1.12
-- b.com@v1.2.3/b/b.go --
package b
func Hello() {}
`
const multiModule = `
-- moda/a/go.mod --
module a.com
require b.com v1.2.3
-- moda/a/go.sum --
b.com v1.2.3 h1:tXrlXP0rnjRpKNmkbLYoWBdq0ikb3C3bKK9//moAWBI=
b.com v1.2.3/go.mod h1:D+J7pfFBZK5vdIdZEFquR586vKKIkqG7Qjw9AxG5BQ8=
-- moda/a/a.go --
package a
import (
"b.com/b"
)
func main() {
var x int
_ = b.Hello()
}
-- modb/go.mod --
module b.com
-- modb/b/b.go --
package b
func Hello() int {
var x int
}
`
func TestAutomaticWorkspaceModule_Interdependent(t *testing.T) {
WithOptions(
ProxyFiles(workspaceModuleProxy),
).Run(t, multiModule, func(t *testing.T, env *Env) {
env.RunGoCommand("work", "init")
env.RunGoCommand("work", "use", "-r", ".")
env.AfterChange(
Diagnostics(env.AtRegexp("moda/a/a.go", "x")),
Diagnostics(env.AtRegexp("modb/b/b.go", "x")),
NoDiagnostics(env.AtRegexp("moda/a/a.go", `"b.com/b"`)),
)
})
}
func TestWorkspaceVendoring(t *testing.T) {
testenv.NeedsGo1Point(t, 22)
WithOptions(
ProxyFiles(workspaceModuleProxy),
).Run(t, multiModule, func(t *testing.T, env *Env) {
env.RunGoCommand("work", "init")
env.RunGoCommand("work", "use", "moda/a")
env.AfterChange()
env.OpenFile("moda/a/a.go")
env.RunGoCommand("work", "vendor")
env.AfterChange()
loc := env.GoToDefinition(env.RegexpSearch("moda/a/a.go", "b.(Hello)"))
const want = "vendor/b.com/b/b.go"
if got := env.Sandbox.Workdir.URIToPath(loc.URI); got != want {
t.Errorf("Definition: got location %q, want %q", got, want)
}
})
}
func TestModuleWithExclude(t *testing.T) {
const proxy = `
-- c.com@v1.2.3/go.mod --
module c.com
go 1.12
require b.com v1.2.3
-- c.com@v1.2.3/blah/blah.go --
package blah
import "fmt"
func SaySomething() {
fmt.Println("something")
}
-- b.com@v1.2.3/go.mod --
module b.com
go 1.12
-- b.com@v1.2.4/b/b.go --
package b
func Hello() {}
-- b.com@v1.2.4/go.mod --
module b.com
go 1.12
`
const files = `
-- go.mod --
module a.com
require c.com v1.2.3
exclude b.com v1.2.3
-- go.sum --
c.com v1.2.3 h1:n07Dz9fYmpNqvZMwZi5NEqFcSHbvLa9lacMX+/g25tw=
c.com v1.2.3/go.mod h1:/4TyYgU9Nu5tA4NymP5xyqE8R2VMzGD3TbJCwCOvHAg=
-- main.go --
package a
func main() {
var x int
}
`
WithOptions(
ProxyFiles(proxy),
).Run(t, files, func(t *testing.T, env *Env) {
env.OnceMet(
InitialWorkspaceLoad,
Diagnostics(env.AtRegexp("main.go", "x")),
)
})
}
// This change tests that the version of the module used changes after it has
// been deleted from the workspace.
//
// TODO(golang/go#55331): delete this placeholder along with experimental
// workspace module.
func TestDeleteModule_Interdependent(t *testing.T) {
const multiModule = `
-- go.work --
go 1.18
use (
moda/a
modb
)
-- moda/a/go.mod --
module a.com
require b.com v1.2.3
-- moda/a/go.sum --
b.com v1.2.3 h1:tXrlXP0rnjRpKNmkbLYoWBdq0ikb3C3bKK9//moAWBI=
b.com v1.2.3/go.mod h1:D+J7pfFBZK5vdIdZEFquR586vKKIkqG7Qjw9AxG5BQ8=
-- moda/a/a.go --
package a
import (
"b.com/b"
)
func main() {
var x int
_ = b.Hello()
}
-- modb/go.mod --
module b.com
-- modb/b/b.go --
package b
func Hello() int {
var x int
}
`
WithOptions(
ProxyFiles(workspaceModuleProxy),
).Run(t, multiModule, func(t *testing.T, env *Env) {
env.OpenFile("moda/a/a.go")
env.Await(env.DoneWithOpen())
originalLoc := env.GoToDefinition(env.RegexpSearch("moda/a/a.go", "Hello"))
original := env.Sandbox.Workdir.URIToPath(originalLoc.URI)
if want := "modb/b/b.go"; !strings.HasSuffix(original, want) {
t.Errorf("expected %s, got %v", want, original)
}
env.CloseBuffer(original)
env.AfterChange()
env.RemoveWorkspaceFile("modb/b/b.go")
env.RemoveWorkspaceFile("modb/go.mod")
env.WriteWorkspaceFile("go.work", "go 1.18\nuse moda/a")
env.AfterChange()
gotLoc := env.GoToDefinition(env.RegexpSearch("moda/a/a.go", "Hello"))
got := env.Sandbox.Workdir.URIToPath(gotLoc.URI)
if want := "b.com@v1.2.3/b/b.go"; !strings.HasSuffix(got, want) {
t.Errorf("expected %s, got %v", want, got)
}
})
}
// Tests that the version of the module used changes after it has been added
// to the workspace.
func TestCreateModule_Interdependent(t *testing.T) {
const multiModule = `
-- go.work --
go 1.18
use (
moda/a
)
-- moda/a/go.mod --
module a.com
require b.com v1.2.3
-- moda/a/go.sum --
b.com v1.2.3 h1:tXrlXP0rnjRpKNmkbLYoWBdq0ikb3C3bKK9//moAWBI=
b.com v1.2.3/go.mod h1:D+J7pfFBZK5vdIdZEFquR586vKKIkqG7Qjw9AxG5BQ8=
-- moda/a/a.go --
package a
import (
"b.com/b"
)
func main() {
var x int
_ = b.Hello()
}
`
WithOptions(
ProxyFiles(workspaceModuleProxy),
).Run(t, multiModule, func(t *testing.T, env *Env) {
env.OpenFile("moda/a/a.go")
loc := env.GoToDefinition(env.RegexpSearch("moda/a/a.go", "Hello"))
original := env.Sandbox.Workdir.URIToPath(loc.URI)
if want := "b.com@v1.2.3/b/b.go"; !strings.HasSuffix(original, want) {
t.Errorf("expected %s, got %v", want, original)
}
env.CloseBuffer(original)
env.WriteWorkspaceFiles(map[string]string{
"go.work": `go 1.18
use (
moda/a
modb
)
`,
"modb/go.mod": "module b.com",
"modb/b/b.go": `package b
func Hello() int {
var x int
}
`,
})
env.AfterChange(Diagnostics(env.AtRegexp("modb/b/b.go", "x")))
gotLoc := env.GoToDefinition(env.RegexpSearch("moda/a/a.go", "Hello"))
got := env.Sandbox.Workdir.URIToPath(gotLoc.URI)
if want := "modb/b/b.go"; !strings.HasSuffix(got, want) {
t.Errorf("expected %s, got %v", want, original)
}
})
}
// This test confirms that a gopls workspace can recover from initialization
// with one invalid module.
func TestOneBrokenModule(t *testing.T) {
const multiModule = `
-- go.work --
go 1.18
use (
moda/a
modb
)
-- moda/a/go.mod --
module a.com
require b.com v1.2.3
-- moda/a/a.go --
package a
import (
"b.com/b"
)
func main() {
var x int
_ = b.Hello()
}
-- modb/go.mod --
modul b.com // typo here
-- modb/b/b.go --
package b
func Hello() int {
var x int
}
`
WithOptions(
ProxyFiles(workspaceModuleProxy),
).Run(t, multiModule, func(t *testing.T, env *Env) {
env.OpenFile("modb/go.mod")
env.AfterChange(
Diagnostics(AtPosition("modb/go.mod", 0, 0)),
)
env.RegexpReplace("modb/go.mod", "modul", "module")
env.SaveBufferWithoutActions("modb/go.mod")
env.AfterChange(
Diagnostics(env.AtRegexp("modb/b/b.go", "x")),
)
})
}
// TestBadGoWork exercises the panic from golang/vscode-go#2121.
func TestBadGoWork(t *testing.T) {
const files = `
-- go.work --
use ./bar
-- bar/go.mod --
module example.com/bar
`
Run(t, files, func(t *testing.T, env *Env) {
env.OpenFile("go.work")
})
}
func TestUseGoWork(t *testing.T) {
// This test validates certain functionality related to using a go.work
// file to specify workspace modules.
const multiModule = `
-- moda/a/go.mod --
module a.com
require b.com v1.2.3
-- moda/a/go.sum --
b.com v1.2.3 h1:tXrlXP0rnjRpKNmkbLYoWBdq0ikb3C3bKK9//moAWBI=
b.com v1.2.3/go.mod h1:D+J7pfFBZK5vdIdZEFquR586vKKIkqG7Qjw9AxG5BQ8=
-- moda/a/a.go --
package a
import (
"b.com/b"
)
func main() {
var x int
_ = b.Hello()
}
-- modb/go.mod --
module b.com
require example.com v1.2.3
-- modb/go.sum --
example.com v1.2.3 h1:Yryq11hF02fEf2JlOS2eph+ICE2/ceevGV3C9dl5V/c=
example.com v1.2.3/go.mod h1:Y2Rc5rVWjWur0h3pd9aEvK5Pof8YKDANh9gHA2Maujo=
-- modb/b/b.go --
package b
func Hello() int {
var x int
}
-- go.work --
go 1.17
use (
./moda/a
)
`
WithOptions(
ProxyFiles(workspaceModuleProxy),
Settings{
"subdirWatchPatterns": "on",
},
).Run(t, multiModule, func(t *testing.T, env *Env) {
// Initially, the go.work should cause only the a.com module to be loaded,
// so we shouldn't get any file watches for modb. Further validate this by
// jumping to a definition in b.com and ensuring that we go to the module
// cache.
env.OnceMet(
InitialWorkspaceLoad,
NoFileWatchMatching("modb"),
)
env.OpenFile("moda/a/a.go")
env.Await(env.DoneWithOpen())
// To verify which modules are loaded, we'll jump to the definition of
// b.Hello.
checkHelloLocation := func(want string) error {
loc := env.GoToDefinition(env.RegexpSearch("moda/a/a.go", "Hello"))
file := env.Sandbox.Workdir.URIToPath(loc.URI)
if !strings.HasSuffix(file, want) {
return fmt.Errorf("expected %s, got %v", want, file)
}
return nil
}
// Initially this should be in the module cache, as b.com is not replaced.
if err := checkHelloLocation("b.com@v1.2.3/b/b.go"); err != nil {
t.Fatal(err)
}
// Now, modify the go.work file on disk to activate the b.com module in
// the workspace.
env.WriteWorkspaceFile("go.work", `
go 1.17
use (
./moda/a
./modb
)
`)
// As of golang/go#54069, writing go.work to the workspace triggers a
// workspace reload, and new file watches.
env.AfterChange(
Diagnostics(env.AtRegexp("modb/b/b.go", "x")),
// TODO(golang/go#60340): we don't get a file watch yet, because
// updateWatchedDirectories runs before snapshot.load. Instead, we get it
// after the next change (the didOpen below).
// FileWatchMatching("modb"),
)
// Jumping to definition should now go to b.com in the workspace.
if err := checkHelloLocation("modb/b/b.go"); err != nil {
t.Fatal(err)
}
// Now, let's modify the go.work *overlay* (not on disk), and verify that
// this change is only picked up once it is saved.
env.OpenFile("go.work")
env.AfterChange(
// TODO(golang/go#60340): delete this expectation in favor of
// the commented-out expectation above, once we fix the evaluation order
// of file watches. We should not have to wait for a second change to get
// the correct watches.
FileWatchMatching("modb"),
)
env.SetBufferContent("go.work", `go 1.17
use (
./moda/a
)`)
// Simply modifying the go.work file does not cause a reload, so we should
// still jump within the workspace.
//
// TODO: should editing the go.work above cause modb diagnostics to be
// suppressed?
env.Await(env.DoneWithChange())
if err := checkHelloLocation("modb/b/b.go"); err != nil {
t.Fatal(err)
}
// Saving should reload the workspace.
env.SaveBufferWithoutActions("go.work")
if err := checkHelloLocation("b.com@v1.2.3/b/b.go"); err != nil {
t.Fatal(err)
}
// This fails if guarded with a OnceMet(DoneWithSave(), ...), because it is
// delayed (and therefore not synchronous with the change).
//
// Note: this check used to assert on NoDiagnostics, but with zero-config
// gopls we still have diagnostics.
env.Await(Diagnostics(ForFile("modb/go.mod"), WithMessage("example.com is not used")))
// Test Formatting.
env.SetBufferContent("go.work", `go 1.18
use (
./moda/a
)
`) // TODO(matloob): For some reason there's a "start position 7:0 is out of bounds" error when the ")" is on the last character/line in the file. Rob probably knows what's going on.
env.SaveBuffer("go.work")
env.Await(env.DoneWithSave())
gotWorkContents := env.ReadWorkspaceFile("go.work")
wantWorkContents := `go 1.18
use (
./moda/a
)
`
if gotWorkContents != wantWorkContents {
t.Fatalf("formatted contents of workspace: got %q; want %q", gotWorkContents, wantWorkContents)
}
})
}
func TestUseGoWorkDiagnosticMissingModule(t *testing.T) {
const files = `
-- go.work --
go 1.18
use ./foo
-- bar/go.mod --
module example.com/bar
`
Run(t, files, func(t *testing.T, env *Env) {
env.OpenFile("go.work")
env.AfterChange(
Diagnostics(env.AtRegexp("go.work", "use"), WithMessage("directory ./foo does not contain a module")),
)
// The following tests is a regression test against an issue where we weren't
// copying the workFile struct field on workspace when a new one was created in
// (*workspace).invalidate. Set the buffer content to a working file so that
// invalidate recognizes the workspace to be change and copies over the workspace
// struct, and then set the content back to the old contents to make sure
// the diagnostic still shows up.
env.SetBufferContent("go.work", "go 1.18 \n\n use ./bar\n")
env.AfterChange(
NoDiagnostics(env.AtRegexp("go.work", "use")),
)
env.SetBufferContent("go.work", "go 1.18 \n\n use ./foo\n")
env.AfterChange(
Diagnostics(env.AtRegexp("go.work", "use"), WithMessage("directory ./foo does not contain a module")),
)
})
}
func TestUseGoWorkDiagnosticSyntaxError(t *testing.T) {
const files = `
-- go.work --
go 1.18
usa ./foo
replace
`
Run(t, files, func(t *testing.T, env *Env) {
env.OpenFile("go.work")
env.AfterChange(
Diagnostics(env.AtRegexp("go.work", "usa"), WithMessage("unknown directive: usa")),
Diagnostics(env.AtRegexp("go.work", "replace"), WithMessage("usage: replace")),
)
})
}
func TestUseGoWorkHover(t *testing.T) {
const files = `
-- go.work --
go 1.18
use ./foo
use (
./bar
./bar/baz
)
-- foo/go.mod --
module example.com/foo
-- bar/go.mod --
module example.com/bar
-- bar/baz/go.mod --
module example.com/bar/baz
`
Run(t, files, func(t *testing.T, env *Env) {
env.OpenFile("go.work")
tcs := map[string]string{
`\./foo`: "example.com/foo",
`(?m)\./bar$`: "example.com/bar",
`\./bar/baz`: "example.com/bar/baz",
}
for hoverRE, want := range tcs {
got, _ := env.Hover(env.RegexpSearch("go.work", hoverRE))
if got.Value != want {
t.Errorf(`hover on %q: got %q, want %q`, hoverRE, got, want)
}
}
})
}
func TestExpandToGoWork(t *testing.T) {
const workspace = `
-- moda/a/go.mod --
module a.com
require b.com v1.2.3
-- moda/a/a.go --
package a
import (
"b.com/b"
)
func main() {
var x int
_ = b.Hello()
}
-- modb/go.mod --
module b.com
require example.com v1.2.3
-- modb/b/b.go --
package b
func Hello() int {
var x int
}
-- go.work --
go 1.17
use (
./moda/a
./modb
)
`
WithOptions(
WorkspaceFolders("moda/a"),
).Run(t, workspace, func(t *testing.T, env *Env) {
env.OpenFile("moda/a/a.go")
env.Await(env.DoneWithOpen())
loc := env.GoToDefinition(env.RegexpSearch("moda/a/a.go", "Hello"))
file := env.Sandbox.Workdir.URIToPath(loc.URI)
want := "modb/b/b.go"
if !strings.HasSuffix(file, want) {
t.Errorf("expected %s, got %v", want, file)
}
})
}
func TestInnerGoWork(t *testing.T) {
// This test checks that gopls honors a go.work file defined
// inside a go module (golang/go#63917).
const workspace = `
-- go.mod --
module a.com
require b.com v1.2.3
-- a/go.work --
go 1.18
use (
..
../b
)
-- a/a.go --
package a
import "b.com/b"
var _ = b.B
-- b/go.mod --
module b.com/b
-- b/b.go --
package b
const B = 0
`
WithOptions(
// This doesn't work if we open the outer module. I'm not sure it should,
// since the go.work file does not apply to the entire module, just a
// subdirectory.
WorkspaceFolders("a"),
).Run(t, workspace, func(t *testing.T, env *Env) {
env.OpenFile("a/a.go")
loc := env.GoToDefinition(env.RegexpSearch("a/a.go", "b.(B)"))
got := env.Sandbox.Workdir.URIToPath(loc.URI)
want := "b/b.go"
if got != want {
t.Errorf("Definition(b.B): got %q, want %q", got, want)
}
})
}
func TestNonWorkspaceFileCreation(t *testing.T) {
const files = `
-- work/go.mod --
module mod.com
go 1.12
-- work/x.go --
package x
`
const code = `
package foo
import "fmt"
var _ = fmt.Printf
`
WithOptions(
WorkspaceFolders("work"), // so that outside/... is outside the workspace
).Run(t, files, func(t *testing.T, env *Env) {
env.CreateBuffer("outside/foo.go", "")
env.EditBuffer("outside/foo.go", fake.NewEdit(0, 0, 0, 0, code))
env.GoToDefinition(env.RegexpSearch("outside/foo.go", `Printf`))
})
}
func TestGoWork_V2Module(t *testing.T) {
// When using a go.work, we must have proxy content even if it is replaced.
const proxy = `
-- b.com/v2@v2.1.9/go.mod --
module b.com/v2
go 1.12
-- b.com/v2@v2.1.9/b/b.go --
package b
func Ciao()() int {
return 0
}
`
const multiModule = `
-- go.work --
go 1.18
use (
moda/a
modb
modb/v2
modc
)
-- moda/a/go.mod --
module a.com
require b.com/v2 v2.1.9
-- moda/a/a.go --
package a
import (
"b.com/v2/b"
)
func main() {
var x int
_ = b.Hi()
}
-- modb/go.mod --
module b.com
-- modb/b/b.go --
package b
func Hello() int {
var x int
}
-- modb/v2/go.mod --
module b.com/v2
-- modb/v2/b/b.go --
package b
func Hi() int {
var x int
}
-- modc/go.mod --
module gopkg.in/yaml.v1 // test gopkg.in versions
-- modc/main.go --
package main
func main() {
var x int
}
`
WithOptions(
ProxyFiles(proxy),
).Run(t, multiModule, func(t *testing.T, env *Env) {
env.OnceMet(
InitialWorkspaceLoad,
// TODO(rfindley): assert on the full set of diagnostics here. We
// should ensure that we don't have a diagnostic at b.Hi in a.go.
Diagnostics(env.AtRegexp("moda/a/a.go", "x")),
Diagnostics(env.AtRegexp("modb/b/b.go", "x")),
Diagnostics(env.AtRegexp("modb/v2/b/b.go", "x")),
Diagnostics(env.AtRegexp("modc/main.go", "x")),
)
})
}
// Confirm that a fix for a tidy module will correct all modules in the
// workspace.
func TestMultiModule_OneBrokenModule(t *testing.T) {
// In the earlier 'experimental workspace mode', gopls would aggregate go.sum
// entries for the workspace module, allowing it to correctly associate
// missing go.sum with diagnostics. With go.work files, this doesn't work:
// the go.command will happily write go.work.sum.
t.Skip("golang/go#57509: go.mod diagnostics do not work in go.work mode")
const files = `
-- go.work --
go 1.18
use (
a
b
)
-- go.work.sum --
-- a/go.mod --
module a.com
go 1.12
-- a/main.go --
package main
-- b/go.mod --
module b.com
go 1.12
require (
example.com v1.2.3
)
-- b/go.sum --
-- b/main.go --
package b
import "example.com/blah"
func main() {
blah.Hello()
}
`
WithOptions(
ProxyFiles(workspaceProxy),
).Run(t, files, func(t *testing.T, env *Env) {
params := &protocol.PublishDiagnosticsParams{}
env.OpenFile("b/go.mod")
env.AfterChange(
Diagnostics(
env.AtRegexp("go.mod", `example.com v1.2.3`),
WithMessage("go.sum is out of sync"),
),
ReadDiagnostics("b/go.mod", params),
)
for _, d := range params.Diagnostics {
if !strings.Contains(d.Message, "go.sum is out of sync") {
continue
}
actions := env.GetQuickFixes("b/go.mod", []protocol.Diagnostic{d})
if len(actions) != 2 {
t.Fatalf("expected 2 code actions, got %v", len(actions))
}
env.ApplyQuickFixes("b/go.mod", []protocol.Diagnostic{d})
}
env.AfterChange(
NoDiagnostics(ForFile("b/go.mod")),
)
})
}
// Tests the fix for golang/go#52500.
func TestChangeTestVariant_Issue52500(t *testing.T) {
const src = `
-- go.mod --
module mod.test
go 1.12
-- main_test.go --
package main_test
type Server struct{}
const mainConst = otherConst
-- other_test.go --
package main_test
const otherConst = 0
func (Server) Foo() {}
`
Run(t, src, func(t *testing.T, env *Env) {
env.OpenFile("other_test.go")
env.RegexpReplace("other_test.go", "main_test", "main")
// For this test to function, it is necessary to wait on both of the
// expectations below: the bug is that when switching the package name in
// other_test.go from main->main_test, metadata for main_test is not marked
// as invalid. So we need to wait for the metadata of main_test.go to be
// updated before moving other_test.go back to the main_test package.
env.Await(
Diagnostics(env.AtRegexp("other_test.go", "Server")),
Diagnostics(env.AtRegexp("main_test.go", "otherConst")),
)
env.RegexpReplace("other_test.go", "main", "main_test")
env.AfterChange(
NoDiagnostics(ForFile("other_test.go")),
NoDiagnostics(ForFile("main_test.go")),
)
// This will cause a test failure if other_test.go is not in any package.
_ = env.GoToDefinition(env.RegexpSearch("other_test.go", "Server"))
})
}
// Test for golang/go#48929.
func TestClearNonWorkspaceDiagnostics(t *testing.T) {
const ws = `
-- go.work --
go 1.18
use (
./b
)
-- a/go.mod --
module a
go 1.17
-- a/main.go --
package main
func main() {
var V string
}
-- b/go.mod --
module b
go 1.17
-- b/main.go --
package b
import (
_ "fmt"
)
`
Run(t, ws, func(t *testing.T, env *Env) {
env.OpenFile("b/main.go")
env.AfterChange(
NoDiagnostics(ForFile("a/main.go")),
)
env.OpenFile("a/main.go")
env.AfterChange(
Diagnostics(env.AtRegexp("a/main.go", "V"), WithMessage("not used")),
)
// Here, diagnostics are added because of zero-config gopls.
// In the past, they were added simply due to diagnosing changed files.
// (see TestClearNonWorkspaceDiagnostics_NoView below for a
// reimplementation of that test).
if got, want := len(env.Views()), 2; got != want {
t.Errorf("after opening a/main.go, got %d views, want %d", got, want)
}
env.CloseBuffer("a/main.go")
env.AfterChange(
NoDiagnostics(ForFile("a/main.go")),
)
if got, want := len(env.Views()), 1; got != want {
t.Errorf("after closing a/main.go, got %d views, want %d", got, want)
}
})
}
// This test is like TestClearNonWorkspaceDiagnostics, but bypasses the
// zero-config algorithm by opening a nested workspace folder.
//
// We should still compute diagnostics correctly for open packages.
func TestClearNonWorkspaceDiagnostics_NoView(t *testing.T) {
const ws = `
-- a/go.mod --
module example.com/a
go 1.18
require example.com/b v1.2.3
replace example.com/b => ../b
-- a/a.go --
package a
import "example.com/b"
func _() {
V := b.B // unused
}
-- b/go.mod --
module b
go 1.18
-- b/b.go --
package b
const B = 2
func _() {
var V int // unused
}
-- b/b2.go --
package b
const B2 = B
-- c/c.go --
package main
func main() {
var V int // unused
}
`
WithOptions(
WorkspaceFolders("a"),
).Run(t, ws, func(t *testing.T, env *Env) {
env.OpenFile("a/a.go")
env.AfterChange(
Diagnostics(env.AtRegexp("a/a.go", "V"), WithMessage("not used")),
NoDiagnostics(ForFile("b/b.go")),
NoDiagnostics(ForFile("c/c.go")),
)
env.OpenFile("b/b.go")
env.AfterChange(
Diagnostics(env.AtRegexp("a/a.go", "V"), WithMessage("not used")),
Diagnostics(env.AtRegexp("b/b.go", "V"), WithMessage("not used")),
NoDiagnostics(ForFile("c/c.go")),
)
// Opening b/b.go should not result in a new view, because b is not
// contained in a workspace folder.
//
// Yet we should get diagnostics for b, because it is open.
if got, want := len(env.Views()), 1; got != want {
t.Errorf("after opening b/b.go, got %d views, want %d", got, want)
}
env.CloseBuffer("b/b.go")
env.AfterChange(
Diagnostics(env.AtRegexp("a/a.go", "V"), WithMessage("not used")),
NoDiagnostics(ForFile("b/b.go")),
NoDiagnostics(ForFile("c/c.go")),
)
// We should get references in the b package.
bUse := env.RegexpSearch("a/a.go", `b\.(B)`)
refs := env.References(bUse)
wantRefs := []string{"a/a.go", "b/b.go", "b/b2.go"}
var gotRefs []string
for _, ref := range refs {
gotRefs = append(gotRefs, env.Sandbox.Workdir.URIToPath(ref.URI))
}
sort.Strings(gotRefs)
if diff := cmp.Diff(wantRefs, gotRefs); diff != "" {
t.Errorf("references(b.B) mismatch (-want +got)\n%s", diff)
}
// Opening c/c.go should also not result in a new view, yet we should get
// orphaned file diagnostics.
env.OpenFile("c/c.go")
env.AfterChange(
Diagnostics(env.AtRegexp("a/a.go", "V"), WithMessage("not used")),
NoDiagnostics(ForFile("b/b.go")),
Diagnostics(env.AtRegexp("c/c.go", "V"), WithMessage("not used")),
)
if got, want := len(env.Views()), 1; got != want {
t.Errorf("after opening b/b.go, got %d views, want %d", got, want)
}
env.CloseBuffer("c/c.go")
env.AfterChange(
Diagnostics(env.AtRegexp("a/a.go", "V"), WithMessage("not used")),
NoDiagnostics(ForFile("b/b.go")),
NoDiagnostics(ForFile("c/c.go")),
)
env.CloseBuffer("a/a.go")
env.AfterChange(
Diagnostics(env.AtRegexp("a/a.go", "V"), WithMessage("not used")),
NoDiagnostics(ForFile("b/b.go")),
NoDiagnostics(ForFile("c/c.go")),
)
})
}
// Test that we don't get a version warning when the Go version in PATH is
// supported.
func TestOldGoNotification_SupportedVersion(t *testing.T) {
v := goVersion(t)
if v < goversion.OldestSupported() {
t.Skipf("go version 1.%d is unsupported", v)
}
Run(t, "", func(t *testing.T, env *Env) {
env.OnceMet(
InitialWorkspaceLoad,
NoShownMessage("upgrade"),
)
})
}
// Test that we do get a version warning when the Go version in PATH is
// unsupported, though this test may never execute if we stop running CI at
// legacy Go versions (see also TestOldGoNotification_Fake)
func TestOldGoNotification_UnsupportedVersion(t *testing.T) {
v := goVersion(t)
if v >= goversion.OldestSupported() {
t.Skipf("go version 1.%d is supported", v)
}
Run(t, "", func(t *testing.T, env *Env) {
env.Await(
// Note: cannot use OnceMet(InitialWorkspaceLoad, ...) here, as the
// upgrade message may race with the IWL.
ShownMessage("Please upgrade"),
)
})
}
func TestOldGoNotification_Fake(t *testing.T) {
// Get the Go version from path, and make sure it's unsupported.
//
// In the future we'll stop running CI on legacy Go versions. By mutating the
// oldest supported Go version here, we can at least ensure that the
// ShowMessage pop-up works.
ctx := context.Background()
version, err := gocommand.GoVersion(ctx, gocommand.Invocation{}, &gocommand.Runner{})
if err != nil {
t.Fatal(err)
}
defer func(t []goversion.Support) {
goversion.Supported = t
}(goversion.Supported)
goversion.Supported = []goversion.Support{
{GoVersion: version, InstallGoplsVersion: "v1.0.0"},
}
Run(t, "", func(t *testing.T, env *Env) {
env.Await(
// Note: cannot use OnceMet(InitialWorkspaceLoad, ...) here, as the
// upgrade message may race with the IWL.
ShownMessage("Please upgrade"),
)
})
}
// goVersion returns the version of the Go command in PATH.
func goVersion(t *testing.T) int {
t.Helper()
ctx := context.Background()
goversion, err := gocommand.GoVersion(ctx, gocommand.Invocation{}, &gocommand.Runner{})
if err != nil {
t.Fatal(err)
}
return goversion
}
func TestGoworkMutation(t *testing.T) {
WithOptions(
ProxyFiles(workspaceModuleProxy),
).Run(t, multiModule, func(t *testing.T, env *Env) {
env.RunGoCommand("work", "init")
env.RunGoCommand("work", "use", "-r", ".")
env.AfterChange(
Diagnostics(env.AtRegexp("moda/a/a.go", "x")),
Diagnostics(env.AtRegexp("modb/b/b.go", "x")),
NoDiagnostics(env.AtRegexp("moda/a/a.go", `b\.Hello`)),
)
env.RunGoCommand("work", "edit", "-dropuse", "modb")
env.Await(
Diagnostics(env.AtRegexp("moda/a/a.go", "x")),
NoDiagnostics(env.AtRegexp("modb/b/b.go", "x")),
Diagnostics(env.AtRegexp("moda/a/a.go", `b\.Hello`)),
)
})
}
func TestInitializeWithNonFileWorkspaceFolders(t *testing.T) {
for _, tt := range []struct {
name string
folders []string
wantViewRoots []string
}{
{
name: "real,virtual",
folders: []string{"modb", "virtual:///virtualpath"},
wantViewRoots: []string{"./modb"},
},
{
name: "virtual,real",
folders: []string{"virtual:///virtualpath", "modb"},
wantViewRoots: []string{"./modb"},
},
{
name: "real,virtual,real",
folders: []string{"moda/a", "virtual:///virtualpath", "modb"},
wantViewRoots: []string{"./moda/a", "./modb"},
},
{
name: "virtual",
folders: []string{"virtual:///virtualpath"},
wantViewRoots: nil,
},
} {
t.Run(tt.name, func(t *testing.T) {
opts := []RunOption{ProxyFiles(workspaceProxy), WorkspaceFolders(tt.folders...)}
WithOptions(opts...).Run(t, multiModule, func(t *testing.T, env *Env) {
summary := func(typ cache.ViewType, root, folder string) command.View {
return command.View{
Type: typ.String(),
Root: env.Sandbox.Workdir.URI(root),
Folder: env.Sandbox.Workdir.URI(folder),
}
}
checkViews := func(want ...command.View) {
got := env.Views()
if diff := cmp.Diff(want, got, cmpopts.IgnoreFields(command.View{}, "ID")); diff != "" {
t.Errorf("SummarizeViews() mismatch (-want +got):\n%s", diff)
}
}
var wantViews []command.View
for _, root := range tt.wantViewRoots {
wantViews = append(wantViews, summary(cache.GoModView, root, root))
}
env.Await(
LogMatching(protocol.Warning, "skip adding virtual folder", 1, false),
)
checkViews(wantViews...)
})
})
}
}
// Test that non-file scheme Document URIs in ChangeWorkspaceFolders
// notification does not produce errors.
func TestChangeNonFileWorkspaceFolders(t *testing.T) {
for _, tt := range []struct {
name string
before []string
after []string
wantViewRoots []string
}{
{
name: "add",
before: []string{"modb"},
after: []string{"modb", "moda/a", "virtual:///virtualpath"},
wantViewRoots: []string{"./modb", "moda/a"},
},
{
name: "remove",
before: []string{"modb", "virtual:///virtualpath", "moda/a"},
after: []string{"modb"},
wantViewRoots: []string{"./modb"},
},
} {
t.Run(tt.name, func(t *testing.T) {
opts := []RunOption{ProxyFiles(workspaceProxy), WorkspaceFolders(tt.before...)}
WithOptions(opts...).Run(t, multiModule, func(t *testing.T, env *Env) {
summary := func(typ cache.ViewType, root, folder string) command.View {
return command.View{
Type: typ.String(),
Root: env.Sandbox.Workdir.URI(root),
Folder: env.Sandbox.Workdir.URI(folder),
}
}
checkViews := func(want ...command.View) {
got := env.Views()
if diff := cmp.Diff(want, got, cmpopts.IgnoreFields(command.View{}, "ID")); diff != "" {
t.Errorf("SummarizeViews() mismatch (-want +got):\n%s", diff)
}
}
var wantViews []command.View
for _, root := range tt.wantViewRoots {
wantViews = append(wantViews, summary(cache.GoModView, root, root))
}
env.ChangeWorkspaceFolders(tt.after...)
env.Await(
LogMatching(protocol.Warning, "skip adding virtual folder", 1, false),
NoOutstandingWork(IgnoreTelemetryPromptWork),
)
checkViews(wantViews...)
})
})
}
}