blob: 850c8c8265baebf3bd48abbc13e14257054d59d4 [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 integration
import (
"errors"
"os"
"path"
"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/internal/xcontext"
)
// RemoveWorkspaceFile deletes a file on disk but does nothing in the
// editor. It calls t.Fatal on any error.
func (e *Env) RemoveWorkspaceFile(name string) {
e.TB.Helper()
if err := e.Sandbox.Workdir.RemoveFile(e.Ctx, name); err != nil {
e.TB.Fatal(err)
}
}
// ReadWorkspaceFile reads a file from the workspace, calling t.Fatal on any
// error.
func (e *Env) ReadWorkspaceFile(name string) string {
e.TB.Helper()
content, err := e.Sandbox.Workdir.ReadFile(name)
if err != nil {
e.TB.Fatal(err)
}
return string(content)
}
// WriteWorkspaceFile writes a file to disk but does nothing in the editor.
// It calls t.Fatal on any error.
func (e *Env) WriteWorkspaceFile(name, content string) {
e.TB.Helper()
if err := e.Sandbox.Workdir.WriteFile(e.Ctx, name, content); err != nil {
e.TB.Fatal(err)
}
}
// WriteWorkspaceFiles deletes a file on disk but does nothing in the
// editor. It calls t.Fatal on any error.
func (e *Env) WriteWorkspaceFiles(files map[string]string) {
e.TB.Helper()
if err := e.Sandbox.Workdir.WriteFiles(e.Ctx, files); err != nil {
e.TB.Fatal(err)
}
}
// ListFiles lists relative paths to files in the given directory.
// It calls t.Fatal on any error.
func (e *Env) ListFiles(dir string) []string {
e.TB.Helper()
paths, err := e.Sandbox.Workdir.ListFiles(dir)
if err != nil {
e.TB.Fatal(err)
}
return paths
}
// OpenFile opens a file in the editor, calling t.Fatal on any error.
func (e *Env) OpenFile(name string) {
e.TB.Helper()
if err := e.Editor.OpenFile(e.Ctx, name); err != nil {
e.TB.Fatal(err)
}
}
// CreateBuffer creates a buffer in the editor, calling t.Fatal on any error.
func (e *Env) CreateBuffer(name string, content string) {
e.TB.Helper()
if err := e.Editor.CreateBuffer(e.Ctx, name, content); err != nil {
e.TB.Fatal(err)
}
}
// BufferText returns the current buffer contents for the file with the given
// relative path, calling t.Fatal if the file is not open in a buffer.
func (e *Env) BufferText(name string) string {
e.TB.Helper()
text, ok := e.Editor.BufferText(name)
if !ok {
e.TB.Fatalf("buffer %q is not open", name)
}
return text
}
// CloseBuffer closes an editor buffer without saving, calling t.Fatal on any
// error.
func (e *Env) CloseBuffer(name string) {
e.TB.Helper()
if err := e.Editor.CloseBuffer(e.Ctx, name); err != nil {
e.TB.Fatal(err)
}
}
// EditBuffer applies edits to an editor buffer, calling t.Fatal on any error.
func (e *Env) EditBuffer(name string, edits ...protocol.TextEdit) {
e.TB.Helper()
if err := e.Editor.EditBuffer(e.Ctx, name, edits); err != nil {
e.TB.Fatal(err)
}
}
func (e *Env) SetBufferContent(name string, content string) {
e.TB.Helper()
if err := e.Editor.SetBufferContent(e.Ctx, name, content); err != nil {
e.TB.Fatal(err)
}
}
// FileContent returns the file content for name that applies to the current
// editing session: it returns the buffer content for an open file, the
// on-disk content for an unopened file, or "" for a non-existent file.
func (e *Env) FileContent(name string) string {
e.TB.Helper()
text, ok := e.Editor.BufferText(name)
if ok {
return text
}
content, err := e.Sandbox.Workdir.ReadFile(name)
if err != nil {
if errors.Is(err, os.ErrNotExist) {
return ""
} else {
e.TB.Fatal(err)
}
}
return string(content)
}
// FileContentAt returns the file content at the given location, using the
// file's mapper.
func (e *Env) FileContentAt(location protocol.Location) string {
e.TB.Helper()
mapper, err := e.Editor.Mapper(location.URI.Path())
if err != nil {
e.TB.Fatal(err)
}
start, end, err := mapper.RangeOffsets(location.Range)
if err != nil {
e.TB.Fatal(err)
}
return string(mapper.Content[start:end])
}
// RegexpSearch returns the starting position of the first match for re in the
// buffer specified by name, calling t.Fatal on any error. It first searches
// for the position in open buffers, then in workspace files.
func (e *Env) RegexpSearch(name, re string) protocol.Location {
e.TB.Helper()
loc, err := e.Editor.RegexpSearch(name, re)
if err == fake.ErrUnknownBuffer {
loc, err = e.Sandbox.Workdir.RegexpSearch(name, re)
}
if err != nil {
e.TB.Fatalf("RegexpSearch: %v, %v for %q", name, err, re)
}
return loc
}
// RegexpReplace replaces the first group in the first match of regexpStr with
// the replace text, calling t.Fatal on any error.
func (e *Env) RegexpReplace(name, regexpStr, replace string) {
e.TB.Helper()
if err := e.Editor.RegexpReplace(e.Ctx, name, regexpStr, replace); err != nil {
e.TB.Fatalf("RegexpReplace: %v", err)
}
}
// SaveBuffer saves an editor buffer, calling t.Fatal on any error.
func (e *Env) SaveBuffer(name string) {
e.TB.Helper()
if err := e.Editor.SaveBuffer(e.Ctx, name); err != nil {
e.TB.Fatal(err)
}
}
func (e *Env) SaveBufferWithoutActions(name string) {
e.TB.Helper()
if err := e.Editor.SaveBufferWithoutActions(e.Ctx, name); err != nil {
e.TB.Fatal(err)
}
}
// FirstDefinition returns the first definition of the symbol at the
// selected location, calling t.Fatal on error.
func (e *Env) FirstDefinition(loc protocol.Location) protocol.Location {
e.TB.Helper()
locs, err := e.Editor.Definitions(e.Ctx, loc)
if err != nil {
e.TB.Fatal(err)
}
if len(locs) == 0 {
e.TB.Fatalf("no definitions")
}
return locs[0]
}
// FirstTypeDefinition returns the first type definition of the symbol
// at the selected location, calling t.Fatal on error.
func (e *Env) FirstTypeDefinition(loc protocol.Location) protocol.Location {
e.TB.Helper()
locs, err := e.Editor.TypeDefinitions(e.Ctx, loc)
if err != nil {
e.TB.Fatal(err)
}
if len(locs) == 0 {
e.TB.Fatalf("no type definitions")
}
return locs[0]
}
// FormatBuffer formats the editor buffer, calling t.Fatal on any error.
func (e *Env) FormatBuffer(name string) {
e.TB.Helper()
if err := e.Editor.FormatBuffer(e.Ctx, name); err != nil {
e.TB.Fatal(err)
}
}
// OrganizeImports processes the source.organizeImports codeAction, calling
// t.Fatal on any error.
func (e *Env) OrganizeImports(name string) {
e.TB.Helper()
if err := e.Editor.OrganizeImports(e.Ctx, name); err != nil {
e.TB.Fatal(err)
}
}
// ApplyQuickFixes processes the quickfix codeAction, calling t.Fatal on any error.
func (e *Env) ApplyQuickFixes(path string, diagnostics []protocol.Diagnostic) {
e.TB.Helper()
loc := e.Sandbox.Workdir.EntireFile(path)
if err := e.Editor.ApplyQuickFixes(e.Ctx, loc, diagnostics); err != nil {
e.TB.Fatal(err)
}
}
// ApplyCodeAction applies the given code action, calling t.Fatal on any error.
func (e *Env) ApplyCodeAction(action protocol.CodeAction) {
e.TB.Helper()
if err := e.Editor.ApplyCodeAction(e.Ctx, action); err != nil {
e.TB.Fatal(err)
}
}
// Diagnostics returns diagnostics for the given file, calling t.Fatal on any
// error.
func (e *Env) Diagnostics(name string) []protocol.Diagnostic {
e.TB.Helper()
diags, err := e.Editor.Diagnostics(e.Ctx, name)
if err != nil {
e.TB.Fatal(err)
}
return diags
}
// GetQuickFixes returns the available quick fix code actions, calling t.Fatal
// on any error.
func (e *Env) GetQuickFixes(path string, diagnostics []protocol.Diagnostic) []protocol.CodeAction {
e.TB.Helper()
loc := e.Sandbox.Workdir.EntireFile(path)
actions, err := e.Editor.GetQuickFixes(e.Ctx, loc, diagnostics)
if err != nil {
e.TB.Fatal(err)
}
return actions
}
// Hover in the editor, calling t.Fatal on any error.
// It may return (nil, zero) even on success.
func (e *Env) Hover(loc protocol.Location) (*protocol.MarkupContent, protocol.Location) {
e.TB.Helper()
c, loc, err := e.Editor.Hover(e.Ctx, loc)
if err != nil {
e.TB.Fatal(err)
}
return c, loc
}
func (e *Env) DocumentLink(name string) []protocol.DocumentLink {
e.TB.Helper()
links, err := e.Editor.DocumentLink(e.Ctx, name)
if err != nil {
e.TB.Fatal(err)
}
return links
}
func (e *Env) DocumentHighlight(loc protocol.Location) []protocol.DocumentHighlight {
e.TB.Helper()
highlights, err := e.Editor.DocumentHighlight(e.Ctx, loc)
if err != nil {
e.TB.Fatal(err)
}
return highlights
}
// RunGenerate runs "go generate" in the given dir, calling t.Fatal on any error.
// It waits for the generate command to complete and checks for file changes
// before returning.
func (e *Env) RunGenerate(dir string) {
e.TB.Helper()
if err := e.Editor.RunGenerate(e.Ctx, dir); err != nil {
e.TB.Fatal(err)
}
e.Await(NoOutstandingWork(IgnoreTelemetryPromptWork))
// Ideally the editor.Workspace would handle all synthetic file watching, but
// we help it out here as we need to wait for the generate command to
// complete before checking the filesystem.
e.CheckForFileChanges()
}
// RunGoCommand runs the given command in the sandbox's default working
// directory.
func (e *Env) RunGoCommand(verb string, args ...string) []byte {
e.TB.Helper()
out, err := e.Sandbox.RunGoCommand(e.Ctx, "", verb, args, nil, true)
if err != nil {
e.TB.Fatal(err)
}
return out
}
// RunGoCommandInDir is like RunGoCommand, but executes in the given
// relative directory of the sandbox.
func (e *Env) RunGoCommandInDir(dir, verb string, args ...string) {
e.TB.Helper()
if _, err := e.Sandbox.RunGoCommand(e.Ctx, dir, verb, args, nil, true); err != nil {
e.TB.Fatal(err)
}
}
// RunGoCommandInDirWithEnv is like RunGoCommand, but executes in the given
// relative directory of the sandbox with the given additional environment variables.
func (e *Env) RunGoCommandInDirWithEnv(dir string, env []string, verb string, args ...string) {
e.TB.Helper()
if _, err := e.Sandbox.RunGoCommand(e.Ctx, dir, verb, args, env, true); err != nil {
e.TB.Fatal(err)
}
}
// GoVersion checks the version of the go command.
// It returns the X in Go 1.X.
func (e *Env) GoVersion() int {
e.TB.Helper()
v, err := e.Sandbox.GoVersion(e.Ctx)
if err != nil {
e.TB.Fatal(err)
}
return v
}
// DumpGoSum prints the correct go.sum contents for dir in txtar format,
// for use in creating integration tests.
func (e *Env) DumpGoSum(dir string) {
e.TB.Helper()
if _, err := e.Sandbox.RunGoCommand(e.Ctx, dir, "list", []string{"-mod=mod", "./..."}, nil, true); err != nil {
e.TB.Fatal(err)
}
sumFile := path.Join(dir, "go.sum")
e.TB.Log("\n\n-- " + sumFile + " --\n" + e.ReadWorkspaceFile(sumFile))
e.TB.Fatal("see contents above")
}
// CheckForFileChanges triggers a manual poll of the workspace for any file
// changes since creation, or since last polling. It is a workaround for the
// lack of true file watching support in the fake workspace.
func (e *Env) CheckForFileChanges() {
e.TB.Helper()
if err := e.Sandbox.Workdir.CheckForFileChanges(e.Ctx); err != nil {
e.TB.Fatal(err)
}
}
// CodeLens calls textDocument/codeLens for the given path, calling t.Fatal on
// any error.
func (e *Env) CodeLens(path string) []protocol.CodeLens {
e.TB.Helper()
lens, err := e.Editor.CodeLens(e.Ctx, path)
if err != nil {
e.TB.Fatal(err)
}
return lens
}
// ExecuteCodeLensCommand executes the command for the code lens matching the
// given command name.
//
// result is a pointer to a variable to be populated by json.Unmarshal.
func (e *Env) ExecuteCodeLensCommand(path string, cmd command.Command, result any) {
e.TB.Helper()
if err := e.Editor.ExecuteCodeLensCommand(e.Ctx, path, cmd, result); err != nil {
e.TB.Fatal(err)
}
}
// ExecuteCommand executes the requested command in the editor, calling t.Fatal
// on any error.
//
// result is a pointer to a variable to be populated by json.Unmarshal.
func (e *Env) ExecuteCommand(params *protocol.ExecuteCommandParams, result any) {
e.TB.Helper()
if err := e.Editor.ExecuteCommand(e.Ctx, params, result); err != nil {
e.TB.Fatal(err)
}
}
// Views returns the server's views.
func (e *Env) Views() []command.View {
var summaries []command.View
cmd := command.NewViewsCommand("")
e.ExecuteCommand(&protocol.ExecuteCommandParams{
Command: cmd.Command,
Arguments: cmd.Arguments,
}, &summaries)
return summaries
}
// StartProfile starts a CPU profile with the given name, using the
// gopls.start_profile custom command. It calls t.Fatal on any error.
//
// The resulting stop function must be called to stop profiling (using the
// gopls.stop_profile custom command).
func (e *Env) StartProfile() (stop func() string) {
// TODO(golang/go#61217): revisit the ergonomics of these command APIs.
//
// This would be a lot simpler if we generated params constructors.
args, err := command.MarshalArgs(command.StartProfileArgs{})
if err != nil {
e.TB.Fatal(err)
}
params := &protocol.ExecuteCommandParams{
Command: command.StartProfile.String(),
Arguments: args,
}
var result command.StartProfileResult
e.ExecuteCommand(params, &result)
return func() string {
stopArgs, err := command.MarshalArgs(command.StopProfileArgs{})
if err != nil {
e.TB.Fatal(err)
}
stopParams := &protocol.ExecuteCommandParams{
Command: command.StopProfile.String(),
Arguments: stopArgs,
}
var result command.StopProfileResult
e.ExecuteCommand(stopParams, &result)
return result.File
}
}
// InlayHints calls textDocument/inlayHints for the given path, calling t.Fatal on
// any error.
func (e *Env) InlayHints(path string) []protocol.InlayHint {
e.TB.Helper()
hints, err := e.Editor.InlayHint(e.Ctx, path)
if err != nil {
e.TB.Fatal(err)
}
return hints
}
// Symbol calls workspace/symbol
func (e *Env) Symbol(query string) []protocol.SymbolInformation {
e.TB.Helper()
ans, err := e.Editor.Symbols(e.Ctx, query)
if err != nil {
e.TB.Fatal(err)
}
return ans
}
// References wraps Editor.References, calling t.Fatal on any error.
func (e *Env) References(loc protocol.Location) []protocol.Location {
e.TB.Helper()
locations, err := e.Editor.References(e.Ctx, loc)
if err != nil {
e.TB.Fatal(err)
}
return locations
}
// Rename wraps Editor.Rename, calling t.Fatal on any error.
func (e *Env) Rename(loc protocol.Location, newName string) {
e.TB.Helper()
if err := e.Editor.Rename(e.Ctx, loc, newName); err != nil {
e.TB.Fatal(err)
}
}
// Implementations wraps Editor.Implementations, calling t.Fatal on any error.
func (e *Env) Implementations(loc protocol.Location) []protocol.Location {
e.TB.Helper()
locations, err := e.Editor.Implementations(e.Ctx, loc)
if err != nil {
e.TB.Fatal(err)
}
return locations
}
// RenameFile wraps Editor.RenameFile, calling t.Fatal on any error.
func (e *Env) RenameFile(oldPath, newPath string) {
e.TB.Helper()
if err := e.Editor.RenameFile(e.Ctx, oldPath, newPath); err != nil {
e.TB.Fatal(err)
}
}
// SignatureHelp wraps Editor.SignatureHelp, calling t.Fatal on error
func (e *Env) SignatureHelp(loc protocol.Location) *protocol.SignatureHelp {
e.TB.Helper()
sighelp, err := e.Editor.SignatureHelp(e.Ctx, loc)
if err != nil {
e.TB.Fatal(err)
}
return sighelp
}
// Completion executes a completion request on the server.
func (e *Env) Completion(loc protocol.Location) *protocol.CompletionList {
e.TB.Helper()
completions, err := e.Editor.Completion(e.Ctx, loc)
if err != nil {
e.TB.Fatal(err)
}
return completions
}
func (e *Env) DidCreateFiles(files ...protocol.DocumentURI) {
e.TB.Helper()
err := e.Editor.DidCreateFiles(e.Ctx, files...)
if err != nil {
e.TB.Fatal(err)
}
}
func (e *Env) SetSuggestionInsertReplaceMode(useReplaceMode bool) {
e.TB.Helper()
e.Editor.SetSuggestionInsertReplaceMode(e.Ctx, useReplaceMode)
}
// AcceptCompletion accepts a completion for the given item at the given
// position.
func (e *Env) AcceptCompletion(loc protocol.Location, item protocol.CompletionItem) {
e.TB.Helper()
if err := e.Editor.AcceptCompletion(e.Ctx, loc, item); err != nil {
e.TB.Fatal(err)
}
}
// CodeActionForFile calls textDocument/codeAction for the entire
// file, and calls t.Fatal if there were errors.
func (e *Env) CodeActionForFile(path string, diagnostics []protocol.Diagnostic) []protocol.CodeAction {
return e.CodeAction(e.Sandbox.Workdir.EntireFile(path), diagnostics, protocol.CodeActionUnknownTrigger)
}
// CodeAction calls textDocument/codeAction for a selection,
// and calls t.Fatal if there were errors.
func (e *Env) CodeAction(loc protocol.Location, diagnostics []protocol.Diagnostic, trigger protocol.CodeActionTriggerKind) []protocol.CodeAction {
e.TB.Helper()
actions, err := e.Editor.CodeAction(e.Ctx, loc, diagnostics, trigger)
if err != nil {
e.TB.Fatal(err)
}
return actions
}
// ChangeConfiguration updates the editor config, calling t.Fatal on any error.
func (e *Env) ChangeConfiguration(newConfig fake.EditorConfig) {
e.TB.Helper()
if err := e.Editor.ChangeConfiguration(e.Ctx, newConfig); err != nil {
e.TB.Fatal(err)
}
}
// ChangeWorkspaceFolders updates the editor workspace folders, calling t.Fatal
// on any error.
func (e *Env) ChangeWorkspaceFolders(newFolders ...string) {
e.TB.Helper()
if err := e.Editor.ChangeWorkspaceFolders(e.Ctx, newFolders); err != nil {
e.TB.Fatal(err)
}
}
// SemanticTokensFull invokes textDocument/semanticTokens/full, calling t.Fatal
// on any error.
func (e *Env) SemanticTokensFull(path string) []fake.SemanticToken {
e.TB.Helper()
toks, err := e.Editor.SemanticTokensFull(e.Ctx, path)
if err != nil {
e.TB.Fatal(err)
}
return toks
}
// SemanticTokensRange invokes textDocument/semanticTokens/range, calling t.Fatal
// on any error.
func (e *Env) SemanticTokensRange(loc protocol.Location) []fake.SemanticToken {
e.TB.Helper()
toks, err := e.Editor.SemanticTokensRange(e.Ctx, loc)
if err != nil {
e.TB.Fatal(err)
}
return toks
}
// Close shuts down resources associated with the environment, calling t.Error
// on any error.
func (e *Env) Close() {
ctx := xcontext.Detach(e.Ctx)
if e.MCPSession != nil {
if err := e.MCPSession.Close(); err != nil {
e.TB.Errorf("closing MCP session: %v", err)
}
}
if e.MCPServer != nil {
e.MCPServer.Close()
}
if err := e.Editor.Close(ctx); err != nil {
e.TB.Errorf("closing editor: %v", err)
}
if err := e.Sandbox.Close(); err != nil {
e.TB.Errorf("cleaning up sandbox: %v", err)
}
}