-
-
Notifications
You must be signed in to change notification settings - Fork 30.4k
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
Preexec hook for REPL #122622
Comments
IIUC, the flow of execution would be:
What is the signature of the hook: # Context contains everything the REPL would have seen if there was no hook,
# including the user input, or whatever is available. The return value is
# what would be effectively passed to the REPL as if there was no hook.
def hook(context: Context) -> REPLInput: ...
# Same as above but you cannot modify the REPL input.
def hook(context: Context) -> None: ...
# Only the input is available but you can modify it.
def hook(input: str) -> REPLInput: ...
# Only the input is available but you cannot modify it
def hook(input: str) -> None: ...
# Universal hook without extra information.
def hook() -> None: ... The first version would be the most powerful but I'm not sure what should be exposed. Alternatively we could have:
Could you clarify the flow of execution and explain where/when the hook is supposed to be invoked (and with which inputs) please? |
We have the If we can have input/context that would be fantastic and make it much more reliable what the client (eg. vscode) sees is running. In pwsh we have a command line sequence that gets sent for just this case: This makes it possible to reliably extract the command line even for multi-line or wrapped commands that could be interleaved with line continuations, right prompts, etc. We currently use I'm not sure what else would be on Context but one of these would be ideal for our use case as we don't need to modify the input: # Same as above but you cannot modify the REPL input.
def hook(context: Context) -> None: ...
# Only the input is available but you cannot modify it
def hook(input: str) -> None: ... With the above we would do something like this: def hook(input: str) -> None:
result = "{command_executed}{command_line}".format(
command_executed="\x1b]633;C\x07",
command_line="\x1b]633;E;" + specialEncode(input) + "\x07",
)
result
`` |
Feature or enhancement
Proposal:
Background
In VS Code we override PS1 to enable shell integration, this currently gives VS Code understanding of where the commands are and what they are. This provides a few features:
Command decorations with visualization of exit/error status:
Command navigation (cmd/ctrl+up/down):
Run recent (ctrl+alt+r):
To give an example of what's possible with shell integration, here's our experimental VS Code-native PowerShell intellisense:
And multi-line sticky scroll:
Request
Something we needed to work around for Python was the lack of a pre-execution hook, we would normally print
\x1b[633;C\a
at the end of a command just before it's run such that VS Code knows where that marker is. Currently we have to stick that at the start of the command input though which can degrade the experience, especially when VS Code attempts to figure out that command has been run manually.To visualization of the problem, see the A, B, C, D markers in our python REPL:
Compare this to PowerShell which positions them correctly:
Shell integration script included in VS Code's python extension: https://github.com/microsoft/vscode-python/blob/main/python_files/pythonrc.py
I'm not a Python dev so I'm not sure what such an API would look like, but we essentially need a hook to run arbitrary code immediately before the REPL runs the command, something like this:
cc @anthonykim1 @brettcannon
Has this already been discussed elsewhere?
This is a minor feature, which does not need previous discussion elsewhere
Links to previous discussion of this feature:
No response
The text was updated successfully, but these errors were encountered: