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

Factor out handle_active_request() to correct multiline input request issue #568

Merged
merged 2 commits into from
Oct 10, 2024
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
173 changes: 102 additions & 71 deletions crates/ark/src/interface.rs
Original file line number Diff line number Diff line change
Expand Up @@ -640,20 +640,34 @@ impl RMain {
buflen: c_int,
_hist: c_int,
) -> ConsoleResult {
let info = Self::prompt_info(prompt);
log::trace!("R prompt: {}", info.input_prompt);

// Upon entering read-console, finalize any debug call text that we were capturing.
// At this point, the user can either advance the debugger, causing us to capture
// a new expression, or execute arbitrary code, where we will reuse a finalized
// debug call text to maintain the debug state.
self.dap.finalize_call_text();
Comment on lines +646 to +650
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'm not entirely sure if it matters that this now runs before pending lines are handled 🤔

Copy link
Contributor

Choose a reason for hiding this comment

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

hmm, I think it's alright but we should only finalize capture if in a browser prompt? The only edge case I can think of is with adversarial readline prompts so this does not matter much I guess.


// We get called here everytime R needs more input. This handler
// represents the driving event of a small state machine that manages
// communication between R and the frontend:
// communication between R and the frontend. In the following order:
//
// - If the vector of pending lines is empty, and if the prompt is for
// new R code, we close the active ExecuteRequest and send a response to
// the frontend.
// - If we detect an input request prompt, then we forward the request
// on to the frontend and then fall through to the event loop to wait
// on the input reply.
//
// - If the vector of pending lines is not empty, R might be waiting for
// us to complete an incomplete expression, or we might just have
// completed an intermediate expression (e.g. from an ExecuteRequest
// like `foo\nbar` where `foo` is intermediate and `bar` is final).
// Send the next line to R.
//
// - If the vector of pending lines is empty, and if the prompt is for
// new R code, we close the active ExecuteRequest and send an
// ExecuteReply to the frontend. We then fall through to the event
// loop to wait for more input.
//
// This state machine depends on being able to reliably distinguish
// between readline prompts (from `readline()`, `scan()`, or `menu()`),
// and actual R code prompts (either top-level or from a nested debug
Expand All @@ -671,74 +685,9 @@ impl RMain {
// prompt, this is a panic. We check ahead of time for complete
// expressions before breaking up an ExecuteRequest in multiple lines,
// so this should not happen.

if let Some(console_result) = self.handle_pending_line(buf, buflen) {
if let Some(console_result) = self.handle_active_request(&info, buf, buflen) {
return console_result;
}

let info = Self::prompt_info(prompt);
log::trace!("R prompt: {}", info.input_prompt);

// An incomplete prompt when we no longer have any inputs to send should
// never happen because we check for incomplete inputs ahead of time and
// respond to the frontend with an error.
if info.incomplete && self.pending_lines.is_empty() {
unreachable!("Incomplete input in `ReadConsole` handler");
}

// Upon entering read-console, finalize any debug call text that we were capturing.
// At this point, the user can either advance the debugger, causing us to capture
// a new expression, or execute arbitrary code, where we will reuse a finalized
// debug call text to maintain the debug state.
self.dap.finalize_call_text();

// TODO: Can we remove this below code?
// If the prompt begins with "Save workspace", respond with (n)
//
// NOTE: Should be able to overwrite the `Cleanup` frontend method.
// This would also help with detecting normal exits versus crashes.
if info.input_prompt.starts_with("Save workspace") {
match Self::on_console_input(buf, buflen, String::from("n")) {
Ok(()) => return ConsoleResult::NewInput,
Err(err) => return ConsoleResult::Error(err),
}
}

if info.input_request {
if let Some(req) = &self.active_request {
// Send request to frontend. We'll wait for an `input_reply`
// from the frontend in the event loop below. The active request
// remains active.
self.request_input(req.orig.clone(), info.input_prompt.to_string());
} else {
// Invalid input request, propagate error to R
return self.handle_invalid_input_request(buf, buflen);
}
} else if let Some(req) = std::mem::take(&mut self.active_request) {
// We got a prompt request marking the end of the previous
// execution. We took and cleared the active request as we're about
// to complete it and send a reply to unblock the active Shell
// request.

// FIXME: Race condition between the comm and shell socket threads.
//
// Send info for the next prompt to frontend. This handles
// custom prompts set by users, e.g. `options(prompt = ,
// continue = )`, as well as debugging prompts, e.g. after a
// call to `browser()`.
let event = UiFrontendEvent::PromptState(PromptStateParams {
input_prompt: info.input_prompt.clone(),
continuation_prompt: info.continuation_prompt.clone(),
});
{
let kernel = self.kernel.lock().unwrap();
kernel.send_ui_event(event);
}

// Let frontend know the last request is complete. This turns us
// back to Idle.
self.reply_execute_request(req, &info);
}
};

// In the future we'll also send browser information, see
// https://github.com/posit-dev/positron/issues/3001. Currently this is
Expand Down Expand Up @@ -880,6 +829,88 @@ impl RMain {
};
}

/// Returns:
/// - `None` if we should fall through to the event loop to wait for more user input
/// - `Some(ConsoleResult)` if we should immediately exit `read_console()`
fn handle_active_request(
&mut self,
info: &PromptInfo,
buf: *mut c_uchar,
buflen: c_int,
) -> Option<ConsoleResult> {
// TODO: Can we remove this below code?
// If the prompt begins with "Save workspace", respond with (n)
// and allow R to immediately exit.
//
// NOTE: Should be able to overwrite the `Cleanup` frontend method.
// This would also help with detecting normal exits versus crashes.
if info.input_prompt.starts_with("Save workspace") {
match Self::on_console_input(buf, buflen, String::from("n")) {
Ok(()) => return Some(ConsoleResult::NewInput),
Err(err) => return Some(ConsoleResult::Error(err)),
}
}

// First check if we are inside request for user input, like a `readline()` or `menu()`.
// It's entirely possible that we still have more pending lines, but an intermediate line
// put us into an `input_request` state. We must respond to that request before processing
// the rest of the pending lines.
if info.input_request {
if let Some(req) = &self.active_request {
// Send request to frontend. We'll wait for an `input_reply`
// from the frontend in the event loop in `read_console()`.
// The active request remains active.
self.request_input(req.orig.clone(), info.input_prompt.to_string());
return None;
} else {
// Invalid input request, propagate error to R
return Some(self.handle_invalid_input_request(buf, buflen));
}
}

// An incomplete prompt when we no longer have any inputs to send should
// never happen because we check for incomplete inputs ahead of time and
// respond to the frontend with an error.
if info.incomplete && self.pending_lines.is_empty() {
unreachable!("Incomplete input in `ReadConsole` handler");
}

// Next check if we have any pending lines. If we do, we are in the middle of
// evaluating a multi line selection, so immediately write the next line into R's buffer.
// The active request remains active.
if let Some(console_result) = self.handle_pending_line(buf, buflen) {
return Some(console_result);
}

// Finally, check if we have an active request from a previous `read_console()`
// iteration. If so, we `take()` and clear the `active_request` as we're about
// to complete it and send a reply to unblock the active Shell
// request.
if let Some(req) = std::mem::take(&mut self.active_request) {
// FIXME: Race condition between the comm and shell socket threads.
//
// Send info for the next prompt to frontend. This handles
// custom prompts set by users, e.g. `options(prompt = ,
// continue = )`, as well as debugging prompts, e.g. after a
// call to `browser()`.
let event = UiFrontendEvent::PromptState(PromptStateParams {
input_prompt: info.input_prompt.clone(),
continuation_prompt: info.continuation_prompt.clone(),
});
{
let kernel = self.kernel.lock().unwrap();
kernel.send_ui_event(event);
}

// Let frontend know the last request is complete. This turns us
// back to Idle.
self.reply_execute_request(req, &info);
}

// Prepare for the next user input
None
}

fn handle_execute_request(
&mut self,
req: RRequest,
Expand Down
76 changes: 76 additions & 0 deletions crates/ark/tests/kernel-notebook.rs
Original file line number Diff line number Diff line change
Expand Up @@ -104,3 +104,79 @@ fn test_notebook_execute_request_incomplete_multiple_lines() {
input.execution_count
)
}

#[test]
fn test_notebook_stdin_basic_prompt() {
let frontend = DummyArkFrontendNotebook::lock();

let options = ExecuteRequestOptions { allow_stdin: true };

let code = "readline('prompt>')";
frontend.send_execute_request(code, options);
frontend.recv_iopub_busy();

let input = frontend.recv_iopub_execute_input();
assert_eq!(input.code, code);

let prompt = frontend.recv_stdin_input_request();
assert_eq!(prompt, String::from("prompt>"));

frontend.send_stdin_input_reply(String::from("hi"));

assert_eq!(frontend.recv_iopub_execute_result(), "[1] \"hi\"");

frontend.recv_iopub_idle();

assert_eq!(frontend.recv_shell_execute_reply(), input.execution_count);
}

#[test]
fn test_notebook_stdin_followed_by_an_expression_on_the_same_line() {
let frontend = DummyArkFrontendNotebook::lock();

let options = ExecuteRequestOptions { allow_stdin: true };

let code = "val <- readline('prompt>'); paste0(val,'-there')";
frontend.send_execute_request(code, options);
frontend.recv_iopub_busy();

let input = frontend.recv_iopub_execute_input();
assert_eq!(input.code, code);

let prompt = frontend.recv_stdin_input_request();
assert_eq!(prompt, String::from("prompt>"));

frontend.send_stdin_input_reply(String::from("hi"));

assert_eq!(frontend.recv_iopub_execute_result(), "[1] \"hi-there\"");

frontend.recv_iopub_idle();

assert_eq!(frontend.recv_shell_execute_reply(), input.execution_count);
}

#[test]
fn test_notebook_stdin_followed_by_an_expression_on_the_next_line() {
let frontend = DummyArkFrontendNotebook::lock();

let options = ExecuteRequestOptions { allow_stdin: true };

// Note, `1` is an intermediate output and is not emitted in notebooks
let code = "1\nval <- readline('prompt>')\npaste0(val,'-there')";
frontend.send_execute_request(code, options);
frontend.recv_iopub_busy();

let input = frontend.recv_iopub_execute_input();
assert_eq!(input.code, code);

let prompt = frontend.recv_stdin_input_request();
assert_eq!(prompt, String::from("prompt>"));

frontend.send_stdin_input_reply(String::from("hi"));

assert_eq!(frontend.recv_iopub_execute_result(), "[1] \"hi-there\"");

frontend.recv_iopub_idle();

assert_eq!(frontend.recv_shell_execute_reply(), input.execution_count);
}
89 changes: 89 additions & 0 deletions crates/ark/tests/kernel.rs
Original file line number Diff line number Diff line change
Expand Up @@ -415,6 +415,58 @@ fn test_stdin_basic_prompt() {
assert_eq!(frontend.recv_shell_execute_reply(), input.execution_count);
}

#[test]
fn test_stdin_followed_by_an_expression_on_the_same_line() {
let frontend = DummyArkFrontend::lock();

let options = ExecuteRequestOptions { allow_stdin: true };

let code = "val <- readline('prompt>'); paste0(val,'-there')";
frontend.send_execute_request(code, options);
frontend.recv_iopub_busy();

let input = frontend.recv_iopub_execute_input();
assert_eq!(input.code, code);

let prompt = frontend.recv_stdin_input_request();
assert_eq!(prompt, String::from("prompt>"));

frontend.send_stdin_input_reply(String::from("hi"));

assert_eq!(frontend.recv_iopub_execute_result(), "[1] \"hi-there\"");

frontend.recv_iopub_idle();

assert_eq!(frontend.recv_shell_execute_reply(), input.execution_count);
}

#[test]
fn test_stdin_followed_by_an_expression_on_the_next_line() {
let frontend = DummyArkFrontend::lock();

let options = ExecuteRequestOptions { allow_stdin: true };

let code = "1\nval <- readline('prompt>')\npaste0(val,'-there')";
frontend.send_execute_request(code, options);
frontend.recv_iopub_busy();

let input = frontend.recv_iopub_execute_input();
assert_eq!(input.code, code);

frontend.recv_iopub_stream_stdout("[1] 1\n");

let prompt = frontend.recv_stdin_input_request();
assert_eq!(prompt, String::from("prompt>"));

frontend.send_stdin_input_reply(String::from("hi"));

assert_eq!(frontend.recv_iopub_execute_result(), "[1] \"hi-there\"");

frontend.recv_iopub_idle();

assert_eq!(frontend.recv_shell_execute_reply(), input.execution_count);
}

#[test]
fn test_stdin_single_line_buffer_overflow() {
let frontend = DummyArkFrontend::lock();
Expand Down Expand Up @@ -448,3 +500,40 @@ fn test_stdin_single_line_buffer_overflow() {
input.execution_count
);
}

#[test]
fn test_stdin_from_menu() {
let frontend = DummyArkFrontend::lock();

let options = ExecuteRequestOptions { allow_stdin: true };

let code = "menu(c('a', 'b'))\n3";
frontend.send_execute_request(code, options);
frontend.recv_iopub_busy();

let input = frontend.recv_iopub_execute_input();
assert_eq!(input.code, code);

// R emits this before asking for your selection
frontend.recv_iopub_stream_stdout(
"
1: a
2: b

",
);

let prompt = frontend.recv_stdin_input_request();
assert_eq!(prompt, String::from("Selection: "));

frontend.send_stdin_input_reply(String::from("b"));

// Position of selection is returned
frontend.recv_iopub_stream_stdout("[1] 2\n");

assert_eq!(frontend.recv_iopub_execute_result(), "[1] 3");

frontend.recv_iopub_idle();

assert_eq!(frontend.recv_shell_execute_reply(), input.execution_count);
}
Loading