mirror of
https://github.com/LadybirdBrowser/ladybird.git
synced 2025-08-11 02:29:21 +00:00
LibWeb: Report exceptions that occur during microtask callback
This commit is contained in:
parent
7a41ab960c
commit
62ecff0b40
Notes:
github-actions[bot]
2024-12-19 15:26:16 +00:00
Author: https://github.com/tcl3
Commit: 62ecff0b40
Pull-request: https://github.com/LadybirdBrowser/ladybird/pull/2739
Reviewed-by: https://github.com/shannonbooth
6 changed files with 74 additions and 7 deletions
|
@ -375,8 +375,8 @@ WebIDL::ExceptionOr<void> HTMLCanvasElement::to_blob(GC::Ref<WebIDL::CallbackTyp
|
||||||
if (file_result.has_value())
|
if (file_result.has_value())
|
||||||
blob_result = FileAPI::Blob::create(realm(), file_result->buffer, TRY_OR_THROW_OOM(vm(), String::from_utf8(file_result->mime_type)));
|
blob_result = FileAPI::Blob::create(realm(), file_result->buffer, TRY_OR_THROW_OOM(vm(), String::from_utf8(file_result->mime_type)));
|
||||||
|
|
||||||
// 2. Invoke callback with « result ».
|
// 2. Invoke callback with « result » and "report".
|
||||||
TRY(WebIDL::invoke_callback(*callback, {}, move(blob_result)));
|
TRY(WebIDL::invoke_callback(*callback, {}, WebIDL::ExceptionBehavior::Report, move(blob_result)));
|
||||||
return {};
|
return {};
|
||||||
});
|
});
|
||||||
if (maybe_error.is_throw_completion())
|
if (maybe_error.is_throw_completion())
|
||||||
|
|
|
@ -84,11 +84,9 @@ void UniversalGlobalScopeMixin::queue_microtask(WebIDL::CallbackType& callback)
|
||||||
if (is<Window>(this_impl()))
|
if (is<Window>(this_impl()))
|
||||||
document = &static_cast<Window&>(this_impl()).associated_document();
|
document = &static_cast<Window&>(this_impl()).associated_document();
|
||||||
|
|
||||||
// The queueMicrotask(callback) method must queue a microtask to invoke callback, and if callback throws an exception, report the exception.
|
// The queueMicrotask(callback) method must queue a microtask to invoke callback with « » and "report".
|
||||||
HTML::queue_a_microtask(document, GC::create_function(realm.heap(), [&callback, &realm] {
|
HTML::queue_a_microtask(document, GC::create_function(realm.heap(), [&callback] {
|
||||||
auto result = WebIDL::invoke_callback(callback, {});
|
(void)WebIDL::invoke_callback(callback, {}, WebIDL::ExceptionBehavior::Report);
|
||||||
if (result.is_error())
|
|
||||||
HTML::report_exception(result, realm);
|
|
||||||
}));
|
}));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -0,0 +1,6 @@
|
||||||
|
Harness status: OK
|
||||||
|
|
||||||
|
Found 1 tests
|
||||||
|
|
||||||
|
1 Pass
|
||||||
|
Pass toBlob() reports the exception from its callback in the callback's global object
|
|
@ -0,0 +1,6 @@
|
||||||
|
Harness status: OK
|
||||||
|
|
||||||
|
Found 1 tests
|
||||||
|
|
||||||
|
1 Pass
|
||||||
|
Pass queueMicrotask() reports the exception from its callback in the callback's global object
|
|
@ -0,0 +1,29 @@
|
||||||
|
<!doctype html>
|
||||||
|
<meta charset=utf-8>
|
||||||
|
<title>toBlob() reports the exception from its callback in the callback's global object</title>
|
||||||
|
<script src=../../../../resources/testharness.js></script>
|
||||||
|
<script src=../../../../resources/testharnessreport.js></script>
|
||||||
|
<iframe srcdoc="<canvas></canvas>"></iframe>
|
||||||
|
<iframe></iframe>
|
||||||
|
<iframe></iframe>
|
||||||
|
<script>
|
||||||
|
setup({ allow_uncaught_exception: true });
|
||||||
|
|
||||||
|
const onerrorCalls = [];
|
||||||
|
window.onerror = () => { onerrorCalls.push("top"); };
|
||||||
|
frames[0].onerror = () => { onerrorCalls.push("frame0"); };
|
||||||
|
frames[1].onerror = () => { onerrorCalls.push("frame1"); };
|
||||||
|
frames[2].onerror = () => { onerrorCalls.push("frame2"); };
|
||||||
|
|
||||||
|
async_test(t => {
|
||||||
|
window.onload = t.step_func(() => {
|
||||||
|
const canvas = frames[0].document.querySelector("canvas");
|
||||||
|
canvas.toBlob(new frames[1].Function(`throw new parent.frames[2].Error("PASS");`));
|
||||||
|
|
||||||
|
t.step_timeout(() => {
|
||||||
|
assert_array_equals(onerrorCalls, ["frame1"]);
|
||||||
|
t.done();
|
||||||
|
}, 25);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
</script>
|
|
@ -0,0 +1,28 @@
|
||||||
|
<!doctype html>
|
||||||
|
<meta charset=utf-8>
|
||||||
|
<title>queueMicrotask() reports the exception from its callback in the callback's global object</title>
|
||||||
|
<script src=../../../resources/testharness.js></script>
|
||||||
|
<script src=../../../resources/testharnessreport.js></script>
|
||||||
|
<iframe></iframe>
|
||||||
|
<iframe></iframe>
|
||||||
|
<iframe></iframe>
|
||||||
|
<script>
|
||||||
|
setup({ allow_uncaught_exception: true });
|
||||||
|
|
||||||
|
const onerrorCalls = [];
|
||||||
|
window.onerror = () => { onerrorCalls.push("top"); };
|
||||||
|
frames[0].onerror = () => { onerrorCalls.push("frame0"); };
|
||||||
|
frames[1].onerror = () => { onerrorCalls.push("frame1"); };
|
||||||
|
frames[2].onerror = () => { onerrorCalls.push("frame2"); };
|
||||||
|
|
||||||
|
async_test(t => {
|
||||||
|
window.onload = t.step_func(() => {
|
||||||
|
frames[0].queueMicrotask(new frames[1].Function(`throw new parent.frames[2].Error("PASS");`));
|
||||||
|
|
||||||
|
t.step_timeout(() => {
|
||||||
|
assert_array_equals(onerrorCalls, ["frame1"]);
|
||||||
|
t.done();
|
||||||
|
}, 4);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
</script>
|
Loading…
Add table
Add a link
Reference in a new issue