-
Notifications
You must be signed in to change notification settings - Fork 17.8k
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
x/tools/gopls: better "run this test" UX #67400
Comments
@adonovan Do you think it would be appropriate for VSCode to utilize command.Test? That may require updates to gopls to enable various flags, and it would definitely need streaming support. Currently I'm using -json, -fullpath, -run, -bench, and -skip though once my work is merged into vscode-go, I plan on adding/migrating support for the I also have support for executing a test via the debugger, but that's a lot more complex and probably not suitable to move to gopls. Because of the way vscode works, to get the output from the debug session I have to use a "debug adapter tracker" and because |
In principle it's fine for VS Code to use LSP commands (which since https://go.dev/cl/597276 are officially undocumented) since both sides are maintained by us. But I thought you were working on a richer VS Code-only testing UX: does it still need the server to run the It would be good to discuss any changes you need to make to the protocol before spending too much time implementing them. |
It does not need the server to run |
VSCode Go extension experimented with the gopls's test command some years ago - part of the legacy Test implementation in gopls was made during the experiment - but dropped the idea. Lack of good output streaming is the main blocker. As @adonovan described, the current implementation combines and sends both stderr and stdout as progress report messages to simulate streaming. Progress message is hard to work with from vscode side. Later we encountered the exact same issue when adding The current implementation also sends back the failure message using showMessage at the end, which is translated as a big prompt in vscode. That is not great either. Some paths that I hope to explore eventually
|
Currently, gopls offers many ways to run the test under the cursor or selected tests:
CodeLenses, in a test.go file, returns a "run test" CodeLens (command + range) for each Test function in the entire file. The command (command.Test) runs the test with streaming output via LSP progress tokens followed by a showMessage. This feature is disabled by default, unless the client specifies because
codelenses: ["test"]
, presumably because it is redundant with VS Code's superior client code (see below). I suspect it is almost entirely unused; telemetry would tell us for sure.CodeActions, in a test.go file, returns a GoTest ("Run tests and benchmarks") CodeAction for each test or benchmark in the selection. The actions commands are command.Test, as before. However, the GoTest code action kind is disabled in DefaultOptions, ~impossible to find, and again I guess that it is largely unused. Its title should be more specific (e.g. "Run TestFoo" or "Run 3 selected tests").
VS Code Go uses VS Code's custom Test Explorer UI, and is migrating to the Testing API. See for example:
VS Code is clearly going the path of custom client side logic, but for other editors, I think we can improve the experience.
The first challenge is to enable either the code lens or the code action, which is not trivial. Documentation may help. One possibility is to sniff the client type and enable the lens or action by default for all clients other than VS Code.
The second is the UX around the test output. Each client displays the streamed output and final showMessage in a different UI element: VS Code feeds it into a sticky dialog box; Vim accumulates the stream into a buffer which is displayed at the end. Emacs+eglot displays the output in the little echo area (bottom line), which is ephemeral and not suitable for large amounts of text.
Personally I use a custom Emacs Lisp function to fork+exec 'gopls codelens' through Emacs' compile package, so that its output is immediately streamed into a first-class editor buffer. I like this because it is immediate, real-time, editable, searchable, and durable, and because it displays the test logs even when the test passes, which is sometimes helpful. Also, Emacs treats the buffer just like a compiler output, so that locations become links, etc. The UX here is pretty close to ideal for me, but the mechanism is very clunky, and of course no-one but me benefits from it.
Yet another possibility would be to stream the output into a file and immediately open the file (using showDocument) in the client. I don't know how well different clients handle a file that is growing. (For example, Emacs has non-file buffers for things like shell sessions, which grow in real time, but for actual files, it only reloads the content periodically after polling.)
Another possibility is to stream the test output using gopls's new web server. I mention it for completeness, but it's a very loose integration and probably not a good idea.
I feel like the LSP ought to provide some way for a command to indicate to the client that its output is best handled like a terminal stream: large, continuously growing, durable, searchable. Perhaps we need to come up with a coherent plan for that and lobby our LSP overlords.
The text was updated successfully, but these errors were encountered: