Skip to content
This repository has been archived by the owner on Aug 31, 2023. It is now read-only.

feat(rome_lsp): stop the daemon after the last client disconnects #3643

Merged
merged 3 commits into from
Nov 10, 2022

Conversation

leops
Copy link
Contributor

@leops leops commented Nov 10, 2022

Summary

Fixes #3636

This change makes use of the existing session tracking code to schedule a background task once the last session is dropped from the server. After a short timeout (correctly set at 60 seconds), this task will broadcast the shutdown signal to the server if no new session was started in the meantime.

This behavior can be disabled by passing the --no-timeout flag to the __run_server internal command, this is done automatically by the rome start command to ensure manually started instances of the server never times out. Importantly, the timeout task is only scheduled when a client disconnects, meaning a newly started server instance will run indefinitely until at least one client connects event if it wasn't passed the --no-timeout flag (this should rarely happen if ever as the daemon spawning code tries to open a connection immediately afterward to ensure the newly created process is working correctly).

Test Plan

I tested this locally by configuring the VSCode extension to use a locally built binary from this branch, and checking the rome process correctly stops about one minute after exiting the editor.

@leops leops requested a review from ematipico as a code owner November 10, 2022 09:58
@netlify
Copy link

netlify bot commented Nov 10, 2022

Deploy Preview for docs-rometools ready!

Name Link
🔨 Latest commit d2ebdc5
🔍 Latest deploy log https://app.netlify.com/sites/docs-rometools/deploys/636d0c33d657a800096e22bf
😎 Deploy Preview https://deploy-preview-3643--docs-rometools.netlify.app
📱 Preview on mobile
Toggle QR Code...

QR Code

Use your smartphone camera to open QR code link.

To edit notification comments on pull requests, go to your Netlify site settings.

@leops leops had a problem deploying to netlify-playground November 10, 2022 09:58 Failure
@leops leops added A-CLI Area: CLI A-LSP Area: language server protocol labels Nov 10, 2022
Copy link
Contributor

@MichaReiser MichaReiser left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

The code changes look good to me but I recommend terminating Rome immediately because it aligns better with what users expect.

60s is too long to avoid the LSP being outdated after upgrading the extension. I understand that VS Code stops the LSP client when updating the extension but immediately tries to connect as soon as the update is done. This happens in a few milliseconds to low seconds.

The same is the case when the user triggers the "Restart Rome LSP" command from VS Code or does a "Reload Workspace" and probably even is if the user closes VS Code and re-opens it manually to see if the problem is now "fixed".

I see that this works if we ship the fix that ensures that Rome only connects to the same version but believe that not stopping Rome can still result in a frustrating experience if Rome is in an "invalid" state (for whatever reason). The intuition of most users is to restart VS Code if they run into this problem, which won't fix their problem with a timeout of 60s.

@leops
Copy link
Contributor Author

leops commented Nov 10, 2022

The code changes look good to me but I recommend terminating Rome immediately because it aligns better with what users expect.

I could shorten the timeout to maybe 10 seconds, but a grace period is required before terminating the process since as I mentioned in the PR description, the CLI opens and close a connection after spawning the daemon to ensure it works correctly. If the server was to stop immediately then the actual client requesting the name of the socket from the Rome CLI (like the editor extension or JS API) would never be able to connect as the server would have been stopped before the rome __print_socket command exited when the initial test connection got dropped

@MichaReiser
Copy link
Contributor

The code changes look good to me but I recommend terminating Rome immediately because it aligns better with what users expect.

I could shorten the timeout to maybe 10 seconds, but a grace period is required before terminating the process since as I mentioned in the PR description, the CLI opens and close a connection after spawning the daemon to ensure it works correctly. If the server was to stop immediately then the actual client requesting the name of the socket from the Rome CLI (like the editor extension or JS API) would never be able to connect as the server would have been stopped before the rome __print_socket command exited when the initial test connection got dropped

That fair but 10s is still a very long time but I see the risk that comes with reducing it to e.g. 1s or even lower.

the CLI opens and close a connection after spawning the daemon to ensure it works correctly

I can see why this is necessary for an existing socket but it should be fairly unlikely that a newly created socket isn't working (and there are good reasons that re-creating the socket again won't succeed as well).

Could we avoid connecting for newly created sockets instead? That still leaves the risk that the rome server terminates between ensureDeamon running and the LSP actually connecting but that's something that VS code should handle for us.

@leops
Copy link
Contributor Author

leops commented Nov 10, 2022

I can see why this is necessary for an existing socket but it should be fairly unlikely that a newly created socket isn't working (and there are good reasons that re-creating the socket again won't succeed as well).

If I remember correctly the connection test is also important for newly spawned process as the socket will be initially unresponsive and cause IO errors if the server hasn't started yet, so some specific error handling code is required there to ensure the daemon is actually ready before handing control back to the client code.

Copy link
Contributor

@MichaReiser MichaReiser left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I think we should reduce the time out to 10s or even 5s.

Does the lsp-proxy command also shut down the deamon after the timeout?

A more complete solution could be to pass a flag in the initialize request that this is only a probing connection or or can we manually send the exit notification with a custom flag to NOT terminate the server.

@jeysal
Copy link
Contributor

jeysal commented Nov 10, 2022

Not very familiar with this code so could be talking nonsense, but would it help to replace a no_timeout=bool with a timeout=u64|false (default value we can still decide) that can be set differently from CLI vs from an editor extension? Definitely not the cleanest solution if it even helps at all, but either way it seems like a good thing to have and a logical extension of the no_timeout flag, plus it avoids a no_* negated option to avoid confusing !no double negation constructs.

@leops
Copy link
Contributor Author

leops commented Nov 10, 2022

Does the lsp-proxy command also shut down the deamon after the timeout?

The implementation of the command in start_lsp_proxy calls ensure_daemon with no_timeout: false so if the commands does spawn a daemon instance it would get stopped once the proxy disconnects.

A more complete solution could be to pass a flag in the initialize request that this is only a probing connection or or can we manually send the exit notification with a custom flag to NOT terminate the server.

The probing connection only opens a connection on the socket then immediately closes it without event sending the initialize or stop request, on the other hand that could be a mean of differentiating the probe connection by inhibiting the shutdown behavior before a session has been initialized at least once.

Not very familiar with this code so could be talking nonsense, but would it help to replace a no_timeout=bool with a timeout=u64|false (default value we can still decide) that can be set differently from CLI vs from an editor extension? Definitely not the cleanest solution if it even helps at all, but either way it seems like a good thing to have and a logical extension of the no_timeout flag, plus it avoids a no_* negated option to avoid confusing !no double negation constructs.

The --no-timeout flag is internal and only ever injected by the CLI as it spawns a the daemon server, so there wasn't really any use case customizing the timeout duration. I do agree the negation make the semantic more complicated than it needs, hence why I already negated it into has_timeout in the language server itself, but let me see if I can refactor it into something with --with-timeout instead

@leops leops had a problem deploying to netlify-playground November 10, 2022 13:51 Failure
Copy link
Contributor

@MichaReiser MichaReiser left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I find the variable name is_oneshot not very intuitive. How about shutdown_on_disconnect or automatically_started

Can you update the README of the VS Code extension to remove the known limitation of an outdated rome server as this should no longer apply.

@leops leops force-pushed the feature/server-timeout branch from db4be95 to d2ebdc5 Compare November 10, 2022 14:35
@leops leops requested a review from a team November 10, 2022 14:35
@leops leops had a problem deploying to netlify-playground November 10, 2022 14:35 Failure
@leops leops merged commit b96f521 into main Nov 10, 2022
@leops leops deleted the feature/server-timeout branch November 10, 2022 15:57
jeysal added a commit to jeysal/rometools that referenced this pull request Nov 10, 2022
* upstream/main:
  fix(npm/js-api): Import type from @rometools/backend-jsonrpc (rome#3647)
  doc(npm/js-api): Add experimental warning to README
  fix(npm/js-api): Lazy load backend implementations (rome#3652)
  feat(rome_lsp): stop the daemon after the last client disconnects (rome#3643)
  fix(npm/js_api): Ensure JS API build contains `dist` folder (rome#3648)
  fix(rome_cli): ensures the service only connects to compatible versions (rome#3642)
  feat(playground): add settings button and group IR (rome#3559)
  release: v10.0.1 (rome#3649)
  fix(rome_js_analyze): False positives for interactive elements in `useKeyWithClickEvents` (rome#3644)
  fix(rome_js_semantic): support for object and array bindings when exporting (rome#3620)
  doc(editors): Clarify Rome discovery (rome#3639)
  fix(rome_js_analyze): improve the detection of `ReactDOM.render` calls in `noRenderReturnValue` (rome#3626)
  chore(npm): add license (rome#3629)
  Add rustdocs index
  Add environment
  Change rust docs to use Netlify
  fix(rome_js_analyze): useValidAnchor considering all possible expressions (rome#3599)
  fix(rome_js_formatter): Trailing comma inside import rome#3600 (rome#3624)
Sign up for free to subscribe to this conversation on GitHub. Already have an account? Sign in.
Labels
A-CLI Area: CLI A-LSP Area: language server protocol
Projects
None yet
Development

Successfully merging this pull request may close these issues.

📎 LSP: Shutdown after last client disconnects
4 participants