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

Add RuntimeCodeExecutionMode to runtime.executeCode() in the API #5450

Conversation

dhruvisompura
Copy link
Contributor

@dhruvisompura dhruvisompura commented Nov 22, 2024

Description

This partially addresses the ask from #4856.

The ask is to make the positron API executeCode call respect the RuntimeCodeExecutionMode and RuntimeErrorBehavior values being passes in. This PR only handles RuntimeCodeExecutionMode. The work for RuntimeErrorBehavior will be done in a separate PR.

The runtime session execute is already setup to work with RuntimeCodeExecutionMode, see workbench.action.executeCode.silently (#2684) for an example of silent code execution via the session directly.

For code execution to work as expected, some changes were made in the PositronConsoleService to prevent data from being rendered in the console or added to history based off the RuntimeCodeExecutionMode.

QA Notes

Note: These changes will need to be tested in a dev build as we do not have a way to test the API for extensions in a release build, see comment here.

We will want to verify that the three different values for RuntimeCodeExecutionMode are respected when positron.runtime.executeCode() is called and verify there aren't any regressions:

  • RuntimeCodeExecutionMode.Interactive means the input/output is displayed and stored in the runtime's history

    • We should verify that executing code via the console and adds to history.
    • We should verify that positron.runtime.executeCode() without a RuntimeCodeExecutionMode defaults to RuntimeCodeExecutionMode.Interactive
    • Verify the command workbench.action.executeCode.console runs the code interactively
  • RuntimeCodeExecutionMode.Transient means the code should be executed (shown in console) but not stored in history

  • RuntimeCodeExecutionMode.Silent means the code output is displayed or stored in history

    • Verify the command workbench.action.executeCode.silently runs the code silently as described above.

The PositronZedLanguageRuntime has a new command exec silent that can execute a code snippet in the language silently. The exec command now explicitly passes a RuntimeCodeExecutionMode of Interactive to make it clear what is happening.

Screenshot

Silent Code Execution - Print

Screen.Recording.2024-11-21.at.4.55.57.PM.mov

Silent Code Execution - Variable Assignment

Screen.Recording.2024-11-21.at.4.55.24.PM.mov

Interactive Code Execution - Print

Screen.Recording.2024-11-21.at.4.54.40.PM.mov

Interactive Code Execution - Variable Assignment

Screen.Recording.2024-11-21.at.4.53.15.PM.mov

} else if (match = code.match(/^exec ([a-zA-Z]+) (.+)/)) {
// Execute code in another language.
const languageId = match[1];
const codeToExecute = match[2];
this.simulateCodeExecution(id, code, languageId, codeToExecute);
this.simulateCodeExecution(id, code, languageId, codeToExecute, positron.RuntimeCodeExecutionMode.Interactive);
Copy link
Contributor Author

Choose a reason for hiding this comment

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

The default value that code is executed as is RuntimeCodeExecutionMode.Interactive if a RuntimeCodeExecutionMode isn't provided so this is change is a 50/50 on if we want to explicitly provide the value here or not.

// Enter busy state and output the code.
this.simulateBusyState(parentId);
this.simulateInputMessage(parentId, code);

// Let the user know what we're about to do
this.simulateOutputMessage(parentId, `Executing ${languageId} snippet: ${codeToExecute}`);

// Don't focus the console if code should being executed silently
const focus = mode !== positron.RuntimeCodeExecutionMode.Silent;
Copy link
Contributor Author

Choose a reason for hiding this comment

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

I figured it makes sense to update the focus value based off the RuntimeCodeExecutionMode. Since this is all used for dev testing it doesn't really matter but this felt more sane than always having focus set to true.

public executeCode(languageId: string, code: string, focus: boolean, allowIncomplete?: boolean): Promise<boolean> {
return this._proxy.$executeCode(languageId, code, focus, allowIncomplete);
public executeCode(languageId: string, code: string, focus: boolean, allowIncomplete?: boolean, runtimeCodeExecutionMode?: RuntimeCodeExecutionMode): Promise<boolean> {
return this._proxy.$executeCode(languageId, code, focus, allowIncomplete, runtimeCodeExecutionMode);
Copy link
Contributor Author

Choose a reason for hiding this comment

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

This call is going out to the mainThreadLanguageRuntime which doesn't support RuntimeCodeExecutionMode and has been added, the meat of the changes are in the positronConsoleService

*/
private doExecuteCode(code: string) {
private doExecuteCode(code: string, runtimeCodeExecutionMode?: RuntimeCodeExecutionMode) {
Copy link
Contributor Author

Choose a reason for hiding this comment

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

This is where the meat of the changes are that need eyes.

It seems like the mainThreadLanguageRuntime is allowed to call the session directly but extensions aren't and have to go through the mainThreadLanguageRuntime which ends up going through this positronConsoleService which is why we need to handle the different RuntimeCodeExecutionMode consequences here.

My understanding is that the _runtimeItemActivities list contains the information rendered by the console UI, aka the code that will be executed and its output. The list is modified in the service here and we need to conditionally decide when to add things to the list based off the RuntimeCodeExecutionMode value.

The console history list is created on the component side and relies on the executionHistoryService to create the history initially and the onDidExecuteCode event to update the list. The only listeners I found for this event was in the console so it should be fine if we don't fire it when we don't want to add input to the history.

I haven't fully dug into how the executionHistoryService creates the initial history list now that I am reflecting back on these changes so I don't know if I've handled all the scenarios for the console input history case. My testing didn't reveal anything but I'll take another look into this to make sure I didn't miss anything

lionel-
lionel- previously approved these changes Nov 22, 2024
Copy link
Contributor

@lionel- lionel- left a comment

Choose a reason for hiding this comment

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

Looks good!

src/positron-dts/positron.d.ts Outdated Show resolved Hide resolved
Comment on lines 2124 to 2128
// Code that is not executed interactively should not show up in the console history
if (codeExecutionMode === RuntimeCodeExecutionMode.Interactive) {
// Fire the onDidExecuteCode event.
this._onDidExecuteCodeEmitter.fire(code);
}
Copy link
Contributor

Choose a reason for hiding this comment

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

I wonder if we should instead keep firing the event but add mode to the emitted data so that handlers can decide to decline handling (e.g. decline adding to the console history).

I worry that we're creating a footgun for future handlers of this event as it might not be clear it will only be fired for interactive executions.

Copy link
Contributor

Choose a reason for hiding this comment

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

I think this is worth at least taking a quick look at, to see if it is doable.

Copy link
Contributor Author

@dhruvisompura dhruvisompura Nov 22, 2024

Choose a reason for hiding this comment

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

I had a similar thought and was wondering if that is the route to go but didn't initially add it because it seems like this service is mainly used to drive information the user sees (I could totally be wrong about this though 😬 ). I agree with this feeling like a footgun so I'll update this so it includes the mode!

Comment on lines +2111 to +2114
// Add the provisional ActivityItemInput. This provisional ActivityItemInput will be
// replaced with the real ActivityItemInput when the runtime sends it (which can take a
// moment or two to happen).
this.addOrUpdateUpdateRuntimeItemActivity(id, activityItemInput);
Copy link
Contributor

Choose a reason for hiding this comment

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

IIUC, if we don't add this provisional item here, it won't be added back in when the runtime responds? Or does the runtime not send one because execution is silent?

Copy link
Contributor

Choose a reason for hiding this comment

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

why is this updatingupdating twice 😆

Copy link
Contributor

Choose a reason for hiding this comment

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

oh I think I see what's going on: the Jupyter protocol advises kernels to rebroadcast an execution input: https://jupyter-client.readthedocs.io/en/stable/messaging.html#code-inputs. If the execute_request is silent, we don't rebroadcast (see https://github.com/posit-dev/ark/blob/cab11b9d0af80ae27d589985ab69bde679d06d5f/crates/ark/src/interface.rs#L610-L614 in Ark). So the input item will not be added back in.

I think it might be worth adding notes about these behaviour dependencies here?

Copy link
Contributor Author

Choose a reason for hiding this comment

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

I can add a note about this! One thing that I don't understand is the code path for what happens to the rebroadcast execution input. It sounds like what you are saying is that the frontend is provided the execution input again from the kernel and somewhere in the code the runtimeItems list is updated to have that information if it doesn't?

Copy link
Contributor

Choose a reason for hiding this comment

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

Here is my understanding of things:

We rebroadcast the input from the client in Ark here: https://github.com/posit-dev/ark/blob/586df407ef05dd99e83770825f282fe4b47ecbab/crates/ark/src/interface.rs#L608-L621

The kernel supervisor forwards that message as positron.LanguageRuntimeInput here:

onExecuteInput(message: JupyterMessage, data: JupyterExecuteInput) {

On the main process this type is known as ILanguageRuntimeMessageInput:

export interface ILanguageRuntimeMessageInput extends ILanguageRuntimeMessage {

The console service handles this message to add a runtime activity item here:

this._runtimeDisposableStore.add(this._session.onDidReceiveRuntimeMessageInput(languageRuntimeMessageInput => {
// If trace is enabled, add a trace runtime item.
if (this._trace) {
this.addRuntimeItemTrace(
formatCallbackTrace('onDidReceiveRuntimeMessageInput', languageRuntimeMessageInput) +
'\nCode:\n' +
languageRuntimeMessageInput.code
);
}
// Add or update the runtime item activity.
this.addOrUpdateUpdateRuntimeItemActivity(
languageRuntimeMessageInput.parent_id,
new ActivityItemInput(
ActivityItemInputState.Executing,
languageRuntimeMessageInput.id,
languageRuntimeMessageInput.parent_id,
new Date(languageRuntimeMessageInput.when),
this._session.dynState.inputPrompt,
this._session.dynState.continuationPrompt,
languageRuntimeMessageInput.code
)
);
}));

Copy link
Contributor Author

Choose a reason for hiding this comment

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

This was extremely helpful! Thank you for taking the time to explain this. I will add a note about how the ark kernel doesn't rebroadcast silent input. This begs the question if the Python kernel has the same behavior. I assume it does but I'll double check.

juliasilge
juliasilge previously approved these changes Nov 22, 2024
Copy link
Contributor

@juliasilge juliasilge left a comment

Choose a reason for hiding this comment

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

This is looking great, pending the couple of minor changes!

I want to highlight for QA verification that we really can only verify this in a dev build currently, via these two dev-build-only methods:

  • The Zed extension exec and exec silent
  • The "JavaScript: Connect Extension Host Runtime" command that lets you run Positron API commands

As a reminder (since I myself had forgotten), for that JavaScript console, you would do something like:

var p = acquirePositronApi()
p.runtime.executeCode("r", "1 + 1", false, false, p.RuntimeCodeExecutionMode.Silent)

You'll see some unrelated errors if you do this, related to how this fake-ish runtime is not hooked up to the Variables pane and similar.

If we wanted to write automated tests for the Positron API, that would be something fairly new. The closest we have right now is here:
https://github.com/posit-dev/positron/tree/main/extensions/vscode-api-tests/src/singlefolder-tests/positron

Comment on lines 2124 to 2128
// Code that is not executed interactively should not show up in the console history
if (codeExecutionMode === RuntimeCodeExecutionMode.Interactive) {
// Fire the onDidExecuteCode event.
this._onDidExecuteCodeEmitter.fire(code);
}
Copy link
Contributor

Choose a reason for hiding this comment

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

I think this is worth at least taking a quick look at, to see if it is doable.

@dhruvisompura dhruvisompura dismissed stale reviews from juliasilge and lionel- via b2e9c72 November 23, 2024 00:49
@dhruvisompura
Copy link
Contributor Author

dhruvisompura commented Nov 23, 2024

FYI, looks like there's a gap in this solution when it comes to handling execution of pending code. The execution explicitly uses Interactive as a mode which ay no longer be correct?

I may just do the work in a follow up PR if that's alright with folks. I can hold off on merging this PR until the pending code execution scenarios are handled so we don't introduce changes that aren't fully functional.

Edit: On further inspection, I don't think there's any work to be done in regards to pending code input. I don't think there's a real life use case where a user would be executing incomplete code fragments non-interactively.

Copy link
Contributor

@juliasilge juliasilge left a comment

Choose a reason for hiding this comment

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

I don't think there's a real life use case where a user would be executing incomplete code fragments non-interactively.

I totally agree, yep. 👍

Working great for me! I tested out both via Zed and the JavaScript runtime.

@dhruvisompura dhruvisompura merged commit e495864 into main Dec 3, 2024
5 checks passed
@dhruvisompura dhruvisompura deleted the feature/add-runtimecodeexecutionmode-to-runtimeexecutecode-in-api branch December 3, 2024 02:08
@github-actions github-actions bot locked and limited conversation to collaborators Dec 3, 2024
Sign up for free to subscribe to this conversation on GitHub. Already have an account? Sign in.
Labels
None yet
Projects
None yet
Development

Successfully merging this pull request may close these issues.

3 participants