1
0
Fork 0
mirror of https://github.com/denoland/deno.git synced 2025-01-03 04:48:52 -05:00

fix(watch): ignore unload errors on drop (#15782)

This commit is contained in:
Nayeem Rahman 2022-09-06 12:18:23 +01:00 committed by GitHub
parent d20e80fa50
commit 118dd47ad0
No known key found for this signature in database
GPG key ID: 4AEE18F83AFDEB23
3 changed files with 48 additions and 10 deletions

View file

@ -1038,6 +1038,40 @@ fn test_watch_module_graph_error_referrer() {
check_alive_then_kill(child); check_alive_then_kill(child);
} }
// Regression test for https://github.com/denoland/deno/issues/15428.
#[test]
fn test_watch_unload_handler_error_on_drop() {
let t = TempDir::new();
let file_to_watch = t.path().join("file_to_watch.js");
write(
&file_to_watch,
r#"
addEventListener("unload", () => {
throw new Error("foo");
});
setTimeout(() => {
throw new Error("bar");
});
"#,
)
.unwrap();
let mut child = util::deno_cmd()
.current_dir(util::testdata_path())
.arg("run")
.arg("--watch")
.arg(&file_to_watch)
.env("NO_COLOR", "1")
.stdout(std::process::Stdio::piped())
.stderr(std::process::Stdio::piped())
.spawn()
.unwrap();
let (_, mut stderr_lines) = child_lines(&mut child);
wait_contains("Process started", &mut stderr_lines);
wait_contains("Uncaught Error: bar", &mut stderr_lines);
wait_contains("Process finished", &mut stderr_lines);
check_alive_then_kill(child);
}
#[test] #[test]
fn run_watch_dynamic_imports() { fn run_watch_dynamic_imports() {
let t = TempDir::new(); let t = TempDir::new();

View file

@ -125,13 +125,18 @@ impl CliMainWorker {
self.pending_unload = true; self.pending_unload = true;
let result = loop { let result = loop {
let result = self.inner.worker.run_event_loop(false).await; match self.inner.worker.run_event_loop(false).await {
if !self Ok(()) => {}
Err(error) => break Err(error),
}
match self
.inner .inner
.worker .worker
.dispatch_beforeunload_event(&located_script_name!())? .dispatch_beforeunload_event(&located_script_name!())
{ {
break result; Ok(default_prevented) if default_prevented => {} // continue loop
Ok(_) => break Ok(()),
Err(error) => break Err(error),
} }
}; };
self.pending_unload = false; self.pending_unload = false;
@ -152,11 +157,10 @@ impl CliMainWorker {
impl Drop for FileWatcherModuleExecutor { impl Drop for FileWatcherModuleExecutor {
fn drop(&mut self) { fn drop(&mut self) {
if self.pending_unload { if self.pending_unload {
self let _ = self
.inner .inner
.worker .worker
.dispatch_unload_event(&located_script_name!()) .dispatch_unload_event(&located_script_name!());
.unwrap();
} }
} }
} }

View file

@ -1167,11 +1167,11 @@ pub(crate) fn exception_to_err_result<'s, T>(
// If termination is the result of a `op_dispatch_exception` call, we want // If termination is the result of a `op_dispatch_exception` call, we want
// to use the exception that was passed to it rather than the exception that // to use the exception that was passed to it rather than the exception that
// was passed to this function. // was passed to this function.
let mut state = state_rc.borrow_mut(); let state = state_rc.borrow();
exception = state exception = state
.dispatched_exceptions .dispatched_exceptions
.pop_back() .back()
.map(|exception| v8::Local::new(scope, exception)) .map(|exception| v8::Local::new(scope, exception.clone()))
.unwrap_or_else(|| { .unwrap_or_else(|| {
// Maybe make a new exception object. // Maybe make a new exception object.
if was_terminating_execution && exception.is_null_or_undefined() { if was_terminating_execution && exception.is_null_or_undefined() {