Skip to content
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

client.needsStop() and client.stop() disagree on whether it's valid to call stop() #1559

Open
DanTup opened this issue Sep 26, 2024 · 5 comments

Comments

@DanTup
Copy link
Contributor

DanTup commented Sep 26, 2024

I'm trying to fix some issues with the LSP client showing errors when I try to dispose it. One issue is that I'm getting errors like this:

rejected promise not handled within 1 second: Error: Client is not running and can't be stopped. It's current state is: starting
extensionHostProcess.js:155
stack trace: Error: Client is not running and can't be stopped. It's current state is: starting
	at LanguageClient.shutdown (d:\Dev\Dart-Code\Dart-Code\out\dist\extension.js:38362:10)
	at LanguageClient.stop (d:\Dev\Dart-Code\Dart-Code\out\dist\extension.js:38332:15)
	at LanguageClient.stop (d:\Dev\Dart-Code\Dart-Code\out\dist\extension.js:47462:22)
	at Object.dispose (d:\Dev\Dart-Code\Dart-Code\out\dist\extension.js:4810:66)
	at disposeAll (d:\Dev\Dart-Code\Dart-Code\out\dist\extension.js:29122:20)
	at LspAnalyzer.dispose (d:\Dev\Dart-Code\Dart-Code\out\dist\extension.js:25645:32)
	at d:\Dev\Dart-Code\Dart-Code\out\dist\extension.js:14511:36
	at tryCleanup (d:\Dev\Dart-Code\Dart-Code\out\dist\extension.js:14544:15)

The problem is that calling stop() is invalid when the state is stopping, however my code looks like this:

if (this.client.needsStop())
	await this.client.stop();

The implementation of needsStop is:

public needsStop(): boolean {
return this.$state === ClientState.Starting || this.$state === ClientState.Running;
}

So needsStop says that stop is required if the state is starting, but calling stop() throws. It's not clear to me what the correct way of shutting down the server cleanly is.

@DanTup DanTup changed the title clietn.needsStop() and client.stop() disagree on whether it's valid to call stop() client.needsStop() and client.stop() disagree on whether it's valid to call stop() Sep 26, 2024
@dbaeumer
Copy link
Member

dbaeumer commented Oct 1, 2024

How do you start the server and which transport are you using? Looking through the code I think we need to allow clients to pass in a terminate function into the client#restart() method in case it is not a process started by the client itself.

@dbaeumer
Copy link
Member

dbaeumer commented Oct 1, 2024

See also #725 (comment)

@DanTup
Copy link
Contributor Author

DanTup commented Oct 1, 2024

I think we need to allow clients to pass in a terminate function into the client#restart() method in case it is not a process started by the client itself.

I didn't realise there was a restart method, but I think this applies to just stop() too. For some cases where I restart the server, things like the active Dart SDK might be changing so I basically restart the entire extension (basically call deactivate() and then activate() internally). I think it should be possible to also do this cleanly (eg. not show the user errors if the server didn't shut down cleanly, and all registered commands must be unregistered so the new server can register them).

@dbaeumer
Copy link
Member

dbaeumer commented Oct 2, 2024

Side note: have you checked that calling deactivate() and activate() on yourself is something that is supported from a VS Code extension side.

@DanTup
Copy link
Contributor Author

DanTup commented Oct 2, 2024

Side note: have you checked that calling deactivate() and activate() on yourself is something that is supported from a VS Code extension side.

They're just functions, so I'm not sure why they wouldn't (I just call them directly). I implemented this many years ago and it hasn't had any issues (except some edge cases for LSP noted here and #725).. It was the simplest way to ensure when something fundamental (like the SDK path) changes, we dispose everything old and create new versions without just implementing our own parallel version of disposing and recreating everything.

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
None yet
Projects
None yet
Development

No branches or pull requests

2 participants