1
0
Fork 0
mirror of https://github.com/denoland/deno.git synced 2025-01-05 05:49:20 -05:00

refactor(core): remove unneeded ops for uncaughtException (#15296)

This commit is contained in:
Bartek Iwańczuk 2022-07-25 01:10:56 +02:00 committed by GitHub
parent 7036600be3
commit 2e1d6d3508
No known key found for this signature in database
GPG key ID: 4AEE18F83AFDEB23
3 changed files with 7 additions and 108 deletions

View file

@ -324,8 +324,6 @@ pub extern "C" fn promise_reject_callback(message: v8::PromiseRejectMessage) {
let mut state = state_rc.borrow_mut(); let mut state = state_rc.borrow_mut();
if let Some(js_promise_reject_cb) = state.js_promise_reject_cb.clone() { if let Some(js_promise_reject_cb) = state.js_promise_reject_cb.clone() {
let js_uncaught_exception_cb = state.js_uncaught_exception_cb.clone();
let tc_scope = &mut v8::TryCatch::new(scope); let tc_scope = &mut v8::TryCatch::new(scope);
let undefined: v8::Local<v8::Value> = v8::undefined(tc_scope).into(); let undefined: v8::Local<v8::Value> = v8::undefined(tc_scope).into();
let type_ = v8::Integer::new(tc_scope, message.get_event() as i32); let type_ = v8::Integer::new(tc_scope, message.get_event() as i32);
@ -358,52 +356,10 @@ pub extern "C" fn promise_reject_callback(message: v8::PromiseRejectMessage) {
if !pending_mod_evaluate.has_evaluated { if !pending_mod_evaluate.has_evaluated {
pending_mod_evaluate pending_mod_evaluate
.handled_promise_rejections .handled_promise_rejections
.push(promise_global.clone()); .push(promise_global);
} }
} }
} }
// TODO(bartlomieju): remove this whole block, `js_uncaught_exception_cb` is
// not needed anymore
if let Some(exception) = tc_scope.exception() {
if let Some(js_uncaught_exception_cb) = js_uncaught_exception_cb {
tc_scope.reset(); // Cancel pending exception.
{
let mut state = state_rc.borrow_mut();
if let Some(pending_mod_evaluate) =
state.pending_mod_evaluate.as_mut()
{
if !pending_mod_evaluate.has_evaluated {
pending_mod_evaluate
.handled_promise_rejections
.push(promise_global);
}
}
}
js_uncaught_exception_cb.open(tc_scope).call(
tc_scope,
undefined,
&[exception],
);
}
}
if tc_scope.has_caught() {
// TODO(bartlomieju): ensure that TODO provided below is still valid.
// If we get here, an exception was thrown by the unhandledRejection
// handler and there is ether no uncaughtException handler or the
// handler threw an exception of its own.
//
// TODO(bnoordhuis) Node terminates the process or worker thread
// but we don't really have that option. The exception won't bubble
// up either because V8 cancels it when this function returns.
let exception = tc_scope
.stack_trace()
.or_else(|| tc_scope.exception())
.map(|value| value.to_rust_string_lossy(tc_scope))
.unwrap_or_else(|| "no exception".into());
eprintln!("Unhandled exception: {}", exception);
}
} else { } else {
let promise = message.get_promise(); let promise = message.get_promise();
let promise_global = v8::Global::new(scope, promise); let promise_global = v8::Global::new(scope, promise);

View file

@ -26,7 +26,6 @@ pub(crate) fn init_builtins_v8() -> Vec<OpDecl> {
op_set_macrotask_callback::decl(), op_set_macrotask_callback::decl(),
op_set_next_tick_callback::decl(), op_set_next_tick_callback::decl(),
op_set_promise_reject_callback::decl(), op_set_promise_reject_callback::decl(),
op_set_uncaught_exception_callback::decl(),
op_run_microtasks::decl(), op_run_microtasks::decl(),
op_has_tick_scheduled::decl(), op_has_tick_scheduled::decl(),
op_set_has_tick_scheduled::decl(), op_set_has_tick_scheduled::decl(),
@ -109,18 +108,6 @@ fn op_set_promise_reject_callback<'a>(
Ok(old.map(|v| from_v8(scope, v.into()).unwrap())) Ok(old.map(|v| from_v8(scope, v.into()).unwrap()))
} }
#[op(v8)]
fn op_set_uncaught_exception_callback<'a>(
scope: &mut v8::HandleScope<'a>,
cb: serde_v8::Value,
) -> Result<Option<serde_v8::Value<'a>>, Error> {
let cb = to_v8_fn(scope, cb)?;
let state_rc = JsRuntime::state(scope);
let old = state_rc.borrow_mut().js_uncaught_exception_cb.replace(cb);
let old = old.map(|v| v8::Local::new(scope, v));
Ok(old.map(|v| from_v8(scope, v.into()).unwrap()))
}
#[op(v8)] #[op(v8)]
fn op_run_microtasks(scope: &mut v8::HandleScope) { fn op_run_microtasks(scope: &mut v8::HandleScope) {
scope.perform_microtask_checkpoint(); scope.perform_microtask_checkpoint();

View file

@ -154,7 +154,6 @@ pub(crate) struct JsRuntimeState {
pub(crate) js_macrotask_cbs: Vec<v8::Global<v8::Function>>, pub(crate) js_macrotask_cbs: Vec<v8::Global<v8::Function>>,
pub(crate) js_nexttick_cbs: Vec<v8::Global<v8::Function>>, pub(crate) js_nexttick_cbs: Vec<v8::Global<v8::Function>>,
pub(crate) js_promise_reject_cb: Option<v8::Global<v8::Function>>, pub(crate) js_promise_reject_cb: Option<v8::Global<v8::Function>>,
pub(crate) js_uncaught_exception_cb: Option<v8::Global<v8::Function>>,
pub(crate) js_format_exception_cb: Option<v8::Global<v8::Function>>, pub(crate) js_format_exception_cb: Option<v8::Global<v8::Function>>,
pub(crate) has_tick_scheduled: bool, pub(crate) has_tick_scheduled: bool,
pub(crate) js_wasm_streaming_cb: Option<v8::Global<v8::Function>>, pub(crate) js_wasm_streaming_cb: Option<v8::Global<v8::Function>>,
@ -394,7 +393,6 @@ impl JsRuntime {
js_macrotask_cbs: vec![], js_macrotask_cbs: vec![],
js_nexttick_cbs: vec![], js_nexttick_cbs: vec![],
js_promise_reject_cb: None, js_promise_reject_cb: None,
js_uncaught_exception_cb: None,
js_format_exception_cb: None, js_format_exception_cb: None,
has_tick_scheduled: false, has_tick_scheduled: false,
js_wasm_streaming_cb: None, js_wasm_streaming_cb: None,
@ -3211,7 +3209,6 @@ assertEquals(1, notify_return_value);
#[tokio::test] #[tokio::test]
async fn test_set_promise_reject_callback() { async fn test_set_promise_reject_callback() {
static PROMISE_REJECT: AtomicUsize = AtomicUsize::new(0); static PROMISE_REJECT: AtomicUsize = AtomicUsize::new(0);
static UNCAUGHT_EXCEPTION: AtomicUsize = AtomicUsize::new(0);
#[op] #[op]
fn op_promise_reject() -> Result<(), AnyError> { fn op_promise_reject() -> Result<(), AnyError> {
@ -3219,17 +3216,8 @@ assertEquals(1, notify_return_value);
Ok(()) Ok(())
} }
#[op]
fn op_uncaught_exception() -> Result<(), AnyError> {
UNCAUGHT_EXCEPTION.fetch_add(1, Ordering::Relaxed);
Ok(())
}
let extension = Extension::builder() let extension = Extension::builder()
.ops(vec![ .ops(vec![op_promise_reject::decl()])
op_promise_reject::decl(),
op_uncaught_exception::decl(),
])
.build(); .build();
let mut runtime = JsRuntime::new(RuntimeOptions { let mut runtime = JsRuntime::new(RuntimeOptions {
@ -3249,23 +3237,17 @@ assertEquals(1, notify_return_value);
if (reason.message !== "reject") { if (reason.message !== "reject") {
throw Error("unexpected reason: " + reason); throw Error("unexpected reason: " + reason);
} }
Deno.core.opSync("op_store_pending_promise_exception", promise);
Deno.core.opSync("op_promise_reject"); Deno.core.opSync("op_promise_reject");
throw Error("promiseReject"); // Triggers uncaughtException handler.
});
Deno.core.opSync("op_set_uncaught_exception_callback", (err) => {
if (err.message !== "promiseReject") throw err;
Deno.core.opSync("op_uncaught_exception");
}); });
new Promise((_, reject) => reject(Error("reject"))); new Promise((_, reject) => reject(Error("reject")));
"#, "#,
) )
.unwrap(); .unwrap();
runtime.run_event_loop(false).await.unwrap(); runtime.run_event_loop(false).await.unwrap_err();
assert_eq!(1, PROMISE_REJECT.load(Ordering::Relaxed)); assert_eq!(1, PROMISE_REJECT.load(Ordering::Relaxed));
assert_eq!(1, UNCAUGHT_EXCEPTION.load(Ordering::Relaxed));
runtime runtime
.execute_script( .execute_script(
@ -3277,29 +3259,18 @@ assertEquals(1, notify_return_value);
}); });
} }
{
const prev = Deno.core.opSync("op_set_uncaught_exception_callback", (...args) => {
prev(...args);
throw Error("fail");
});
}
new Promise((_, reject) => reject(Error("reject"))); new Promise((_, reject) => reject(Error("reject")));
"#, "#,
) )
.unwrap(); .unwrap();
// Exception from uncaughtException handler doesn't bubble up but is runtime.run_event_loop(false).await.unwrap_err();
// printed to stderr.
runtime.run_event_loop(false).await.unwrap();
assert_eq!(2, PROMISE_REJECT.load(Ordering::Relaxed)); assert_eq!(2, PROMISE_REJECT.load(Ordering::Relaxed));
assert_eq!(2, UNCAUGHT_EXCEPTION.load(Ordering::Relaxed));
} }
#[tokio::test] #[tokio::test]
async fn test_set_promise_reject_callback_top_level_await() { async fn test_set_promise_reject_callback_top_level_await() {
static PROMISE_REJECT: AtomicUsize = AtomicUsize::new(0); static PROMISE_REJECT: AtomicUsize = AtomicUsize::new(0);
static UNCAUGHT_EXCEPTION: AtomicUsize = AtomicUsize::new(0);
#[op] #[op]
fn op_promise_reject() -> Result<(), AnyError> { fn op_promise_reject() -> Result<(), AnyError> {
@ -3307,17 +3278,8 @@ assertEquals(1, notify_return_value);
Ok(()) Ok(())
} }
#[op]
fn op_uncaught_exception() -> Result<(), AnyError> {
UNCAUGHT_EXCEPTION.fetch_add(1, Ordering::Relaxed);
Ok(())
}
let extension = Extension::builder() let extension = Extension::builder()
.ops(vec![ .ops(vec![op_promise_reject::decl()])
op_promise_reject::decl(),
op_uncaught_exception::decl(),
])
.build(); .build();
#[derive(Default)] #[derive(Default)]
@ -3345,11 +3307,6 @@ assertEquals(1, notify_return_value);
let source = r#" let source = r#"
Deno.core.opSync("op_set_promise_reject_callback", (type, promise, reason) => { Deno.core.opSync("op_set_promise_reject_callback", (type, promise, reason) => {
Deno.core.opSync("op_promise_reject"); Deno.core.opSync("op_promise_reject");
throw reason;
});
Deno.core.opSync("op_set_uncaught_exception_callback", (err) => {
Deno.core.opSync("op_uncaught_exception");
}); });
throw new Error('top level throw'); throw new Error('top level throw');
@ -3379,10 +3336,9 @@ assertEquals(1, notify_return_value);
.unwrap(); .unwrap();
let receiver = runtime.mod_evaluate(id); let receiver = runtime.mod_evaluate(id);
runtime.run_event_loop(false).await.unwrap(); runtime.run_event_loop(false).await.unwrap();
receiver.await.unwrap().unwrap(); receiver.await.unwrap().unwrap_err();
assert_eq!(1, PROMISE_REJECT.load(Ordering::Relaxed)); assert_eq!(1, PROMISE_REJECT.load(Ordering::Relaxed));
assert_eq!(1, UNCAUGHT_EXCEPTION.load(Ordering::Relaxed));
} }
#[test] #[test]