mirror of
https://github.com/LadybirdBrowser/ladybird.git
synced 2025-01-22 09:12:13 -05:00
LibWeb/HTML: Include better information in 'report an exception' event
Instead of always reporting a colno and lineno of zero try and use the values from the Error object that may be provided, falling back to the source location of the invocation if not provided. We can definitely improve the reporting even more, but this is a start! Also update this function to latest spec while we're in the area.
This commit is contained in:
parent
f388d3c88c
commit
57479c2d4b
Notes:
github-actions[bot]
2025-01-12 18:50:52 +00:00
Author: https://github.com/shannonbooth Commit: https://github.com/LadybirdBrowser/ladybird/commit/57479c2d4b9 Pull-request: https://github.com/LadybirdBrowser/ladybird/pull/3232 Reviewed-by: https://github.com/tcl3 ✅
6 changed files with 204 additions and 72 deletions
|
@ -2,6 +2,7 @@
|
||||||
* Copyright (c) 2022, Andrew Kaster <akaster@serenityos.org>
|
* Copyright (c) 2022, Andrew Kaster <akaster@serenityos.org>
|
||||||
* Copyright (c) 2023, Linus Groh <linusg@serenityos.org>
|
* Copyright (c) 2023, Linus Groh <linusg@serenityos.org>
|
||||||
* Copyright (c) 2023, Luke Wilde <lukew@serenityos.org>
|
* Copyright (c) 2023, Luke Wilde <lukew@serenityos.org>
|
||||||
|
* Copyright (c) 2025, Shannon Booth <shannon@serenityos.org>
|
||||||
*
|
*
|
||||||
* SPDX-License-Identifier: BSD-2-Clause
|
* SPDX-License-Identifier: BSD-2-Clause
|
||||||
*/
|
*/
|
||||||
|
@ -718,99 +719,151 @@ void WindowOrWorkerGlobalScopeMixin::report_error(JS::Value e)
|
||||||
report_an_exception(e);
|
report_an_exception(e);
|
||||||
}
|
}
|
||||||
|
|
||||||
// https://html.spec.whatwg.org/multipage/webappapis.html#report-an-exception
|
// https://html.spec.whatwg.org/multipage/webappapis.html#extract-error
|
||||||
void WindowOrWorkerGlobalScopeMixin::report_an_exception(JS::Value const& e)
|
struct ErrorInformation {
|
||||||
|
String message;
|
||||||
|
String filename;
|
||||||
|
JS::Value error;
|
||||||
|
size_t lineno { 0 };
|
||||||
|
size_t colno { 0 };
|
||||||
|
};
|
||||||
|
|
||||||
|
// https://html.spec.whatwg.org/multipage/webappapis.html#extract-error
|
||||||
|
static ErrorInformation extract_error_information(JS::VM& vm, JS::Value exception)
|
||||||
{
|
{
|
||||||
auto& target = static_cast<DOM::EventTarget&>(this_impl());
|
// 1. Let attributes be an empty map keyed by IDL attributes.
|
||||||
auto& realm = relevant_realm(target);
|
ErrorInformation attributes;
|
||||||
auto& vm = realm.vm();
|
|
||||||
auto script_or_module = vm.get_active_script_or_module();
|
|
||||||
|
|
||||||
// FIXME: Get the current position in the script.
|
// 2. Set attributes[error] to exception.
|
||||||
auto line = 0;
|
attributes.error = exception;
|
||||||
auto col = 0;
|
|
||||||
|
|
||||||
// 1. If target is in error reporting mode, then return; the error is not handled.
|
// 3. Set attributes[message], attributes[filename], attributes[lineno], and attributes[colno] to
|
||||||
if (m_error_reporting_mode) {
|
// implementation-defined values derived from exception.
|
||||||
report_exception_to_console(e, realm, ErrorInPromise::No);
|
attributes.message = [&] {
|
||||||
return;
|
if (exception.is_object()) {
|
||||||
}
|
auto& object = exception.as_object();
|
||||||
|
|
||||||
// 2. Let target be in error reporting mode.
|
|
||||||
m_error_reporting_mode = true;
|
|
||||||
|
|
||||||
// 3. Let message be an implementation-defined string describing the error in a helpful manner.
|
|
||||||
auto message = [&] {
|
|
||||||
if (e.is_object()) {
|
|
||||||
auto& object = e.as_object();
|
|
||||||
if (MUST(object.has_own_property(vm.names.message))) {
|
if (MUST(object.has_own_property(vm.names.message))) {
|
||||||
auto message = object.get_without_side_effects(vm.names.message);
|
auto message = object.get_without_side_effects(vm.names.message);
|
||||||
return message.to_string_without_side_effects();
|
return message.to_string_without_side_effects();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
return MUST(String::formatted("Uncaught exception: {}", e.to_string_without_side_effects()));
|
return MUST(String::formatted("Uncaught exception: {}", exception.to_string_without_side_effects()));
|
||||||
}();
|
}();
|
||||||
|
|
||||||
// 4. Let errorValue be the value that represents the error: in the case of an uncaught exception,
|
// FIXME: This offset is relative to the javascript source. Other browsers appear to do it relative
|
||||||
// that would be the value that was thrown; in the case of a JavaScript error that would be an Error object
|
// to the entire source document! Calculate that somehow.
|
||||||
// If there is no corresponding value, then the null value must be used instead.
|
|
||||||
auto error_value = e;
|
|
||||||
|
|
||||||
// 5. Let urlString be the result of applying the URL serializer to the URL record that corresponds to the resource from which script was obtained.
|
// If we got an Error object, then try and extract the information from the location the object was made.
|
||||||
// NOTE: urlString is set below once we have determined whether we are dealing with a script or a module.
|
if (exception.is_object() && is<JS::Error>(exception.as_object())) {
|
||||||
String url_string;
|
auto const& error = static_cast<JS::Error&>(exception.as_object());
|
||||||
auto script_or_module_filename = [](auto const& script_or_module) {
|
for (auto const& frame : error.traceback()) {
|
||||||
return MUST(String::from_utf8(script_or_module->filename()));
|
auto source_range = frame.source_range();
|
||||||
};
|
if (source_range.start.line != 0 || source_range.start.column != 0) {
|
||||||
|
attributes.filename = MUST(String::from_byte_string(source_range.filename()));
|
||||||
|
attributes.lineno = source_range.start.line;
|
||||||
|
attributes.colno = source_range.start.column;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
// Otherwise, we fall back to try and find the location of the invocation of the function itself.
|
||||||
|
else {
|
||||||
|
for (ssize_t i = vm.execution_context_stack().size() - 1; i >= 0; --i) {
|
||||||
|
auto& frame = vm.execution_context_stack()[i];
|
||||||
|
if (frame->executable && frame->program_counter.has_value()) {
|
||||||
|
auto source_range = frame->executable->source_range_at(frame->program_counter.value()).realize();
|
||||||
|
attributes.filename = MUST(String::from_byte_string(source_range.filename()));
|
||||||
|
attributes.lineno = source_range.start.line;
|
||||||
|
attributes.colno = source_range.start.column;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
// 6. If script is a classic script and script's muted errors is true, then set message to "Script error.",
|
// 4. Return attributes.
|
||||||
// urlString to the empty string, line and col to 0, and errorValue to null.
|
return attributes;
|
||||||
|
}
|
||||||
|
|
||||||
|
// https://html.spec.whatwg.org/multipage/webappapis.html#report-an-exception
|
||||||
|
void WindowOrWorkerGlobalScopeMixin::report_an_exception(JS::Value exception, OmitError omit_error)
|
||||||
|
{
|
||||||
|
auto& target = static_cast<DOM::EventTarget&>(this_impl());
|
||||||
|
auto& realm = relevant_realm(target);
|
||||||
|
auto& vm = realm.vm();
|
||||||
|
|
||||||
|
// 1. Let notHandled be true.
|
||||||
|
bool not_handled = true;
|
||||||
|
|
||||||
|
// 2. Let errorInfo be the result of extracting error information from exception.
|
||||||
|
auto error_info = extract_error_information(vm, exception);
|
||||||
|
|
||||||
|
// 3. Let script be a script found in an implementation-defined way, or null. This should usually be the
|
||||||
|
// running script (most notably during run a classic script).
|
||||||
|
auto script_or_module = vm.get_active_script_or_module();
|
||||||
|
|
||||||
|
// 4. If script is a classic script and script's muted errors is true, then set errorInfo[error] to null,
|
||||||
|
// errorInfo[message] to "Script error.", errorInfo[filename] to the empty string, errorInfo[lineno] to
|
||||||
|
// 0, and errorInfo[colno] to 0.
|
||||||
script_or_module.visit(
|
script_or_module.visit(
|
||||||
[&](GC::Ref<JS::Script> const& js_script) {
|
[&](GC::Ref<JS::Script> const& js_script) {
|
||||||
if (verify_cast<ClassicScript>(js_script->host_defined())->muted_errors() == ClassicScript::MutedErrors::Yes) {
|
if (verify_cast<ClassicScript>(js_script->host_defined())->muted_errors() == ClassicScript::MutedErrors::Yes) {
|
||||||
message = "Script error."_string;
|
error_info.error = JS::js_null();
|
||||||
url_string = String {};
|
error_info.message = "Script error."_string;
|
||||||
line = 0;
|
error_info.filename = String {};
|
||||||
col = 0;
|
error_info.lineno = 0;
|
||||||
error_value = JS::js_null();
|
error_info.colno = 0;
|
||||||
} else {
|
|
||||||
url_string = script_or_module_filename(js_script);
|
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
[&](GC::Ref<JS::Module> const& js_module) {
|
[](auto const&) {});
|
||||||
url_string = script_or_module_filename(js_module);
|
|
||||||
},
|
|
||||||
[](Empty) {});
|
|
||||||
|
|
||||||
// 7. Let notHandled be true.
|
// 5. If omitError is true, then set errorInfo[error] to null.
|
||||||
auto not_handled = true;
|
if (omit_error == OmitError::Yes)
|
||||||
|
error_info.error = JS::js_null();
|
||||||
|
|
||||||
// 8. If target implements EventTarget, then set notHandled to the result of firing an event named error at target,
|
// 6. If global is not in error reporting mode, then:
|
||||||
// using ErrorEvent, with the cancelable attribute initialized to true, the message attribute initialized to message,
|
if (!m_error_reporting_mode) {
|
||||||
// the filename attribute initialized to urlString, the lineno attribute initialized to line, the colno attribute initialized to col,
|
// 1. Set global's in error reporting mode to true.
|
||||||
// and the error attribute initialized to errorValue.
|
m_error_reporting_mode = true;
|
||||||
ErrorEventInit event_init = {};
|
|
||||||
event_init.cancelable = true;
|
|
||||||
event_init.message = message;
|
|
||||||
event_init.filename = url_string;
|
|
||||||
event_init.lineno = line;
|
|
||||||
event_init.colno = col;
|
|
||||||
event_init.error = error_value;
|
|
||||||
|
|
||||||
not_handled = target.dispatch_event(ErrorEvent::create(realm, EventNames::error, event_init));
|
// 2. If global implements EventTarget, then set notHandled to the result of firing an event named
|
||||||
|
// error at global, using ErrorEvent, with the cancelable attribute initialized to true, and
|
||||||
|
// additional attributes initialized according to errorInfo.
|
||||||
|
ErrorEventInit event_init = {};
|
||||||
|
event_init.cancelable = true;
|
||||||
|
event_init.message = error_info.message;
|
||||||
|
event_init.filename = error_info.filename;
|
||||||
|
event_init.lineno = error_info.lineno;
|
||||||
|
event_init.colno = error_info.colno;
|
||||||
|
event_init.error = error_info.error;
|
||||||
|
|
||||||
// 9. Let target no longer be in error reporting mode.
|
not_handled = target.dispatch_event(ErrorEvent::create(realm, EventNames::error, event_init));
|
||||||
m_error_reporting_mode = false;
|
|
||||||
|
|
||||||
// 10. If notHandled is false, then the error is handled. Otherwise, the error is not handled.
|
// 3. Set global's in error reporting mode to false.
|
||||||
|
m_error_reporting_mode = false;
|
||||||
|
}
|
||||||
|
|
||||||
|
// 7. If notHandled is true, then:
|
||||||
if (not_handled) {
|
if (not_handled) {
|
||||||
// When the user agent is to report an exception E, the user agent must report the error for the relevant script,
|
// 1. Set errorInfo[error] to null.
|
||||||
// with the problematic position (line number and column number) in the resource containing the script,
|
error_info.error = JS::js_null();
|
||||||
// using the global object specified by the script's settings object as the target.
|
|
||||||
// If the error is still not handled after this, then the error may be reported to a developer console.
|
// FIXME: 2. If global implements DedicatedWorkerGlobalScope, queue a global task on the DOM manipulation
|
||||||
// https://html.spec.whatwg.org/multipage/webappapis.html#report-the-exception
|
// task source with the global's associated Worker's relevant global object to run these steps:
|
||||||
report_exception_to_console(e, realm, ErrorInPromise::No);
|
if (false) {
|
||||||
|
// FIXME: 1. Let workerObject be the Worker object associated with global.
|
||||||
|
|
||||||
|
// FIXME: 2. Set notHandled be the result of firing an event named error at workerObject, using ErrorEvent,
|
||||||
|
// with the cancelable attribute initialized to true, and additional attributes initialized
|
||||||
|
// according to errorInfo.
|
||||||
|
|
||||||
|
// FIXME: 3. If notHandled is true, then report exception for workerObject's relevant global object with
|
||||||
|
// omitError set to true.
|
||||||
|
}
|
||||||
|
}
|
||||||
|
// 8. Otherwise, the user agent may report exception to a developer console.
|
||||||
|
else {
|
||||||
|
report_exception_to_console(exception, realm, ErrorInPromise::No);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -72,7 +72,12 @@ public:
|
||||||
GC::Ref<IndexedDB::IDBFactory> indexed_db();
|
GC::Ref<IndexedDB::IDBFactory> indexed_db();
|
||||||
|
|
||||||
void report_error(JS::Value e);
|
void report_error(JS::Value e);
|
||||||
void report_an_exception(JS::Value const& e);
|
|
||||||
|
enum class OmitError {
|
||||||
|
Yes,
|
||||||
|
No,
|
||||||
|
};
|
||||||
|
void report_an_exception(JS::Value exception, OmitError = OmitError::No);
|
||||||
|
|
||||||
[[nodiscard]] GC::Ref<Crypto::Crypto> crypto();
|
[[nodiscard]] GC::Ref<Crypto::Crypto> crypto();
|
||||||
|
|
||||||
|
|
|
@ -1,6 +1,6 @@
|
||||||
message = Reporting an Error!
|
message = Reporting an Error!
|
||||||
lineno = 0
|
lineno = 17
|
||||||
colno = 0
|
colno = 28
|
||||||
error = Error: Reporting an Error!
|
error = Error: Reporting an Error!
|
||||||
filename URL scheme = file:
|
filename URL scheme = file:
|
||||||
filename URL final path segment = WindowOrWorkerGlobalScope-reportError.html
|
filename URL final path segment = WindowOrWorkerGlobalScope-reportError.html
|
||||||
|
|
|
@ -0,0 +1,10 @@
|
||||||
|
Harness status: OK
|
||||||
|
|
||||||
|
Found 5 tests
|
||||||
|
|
||||||
|
5 Pass
|
||||||
|
Pass self.reportError(1)
|
||||||
|
Pass self.reportError(TypeError)
|
||||||
|
Pass self.reportError(undefined)
|
||||||
|
Pass self.reportError() (without arguments) throws
|
||||||
|
Pass self.reportError() doesn't invoke getters
|
|
@ -0,0 +1,15 @@
|
||||||
|
<!doctype html>
|
||||||
|
<meta charset=utf-8>
|
||||||
|
|
||||||
|
<script>
|
||||||
|
self.GLOBAL = {
|
||||||
|
isWindow: function() { return true; },
|
||||||
|
isWorker: function() { return false; },
|
||||||
|
isShadowRealm: function() { return false; },
|
||||||
|
};
|
||||||
|
</script>
|
||||||
|
<script src="../../../resources/testharness.js"></script>
|
||||||
|
<script src="../../../resources/testharnessreport.js"></script>
|
||||||
|
|
||||||
|
<div id=log></div>
|
||||||
|
<script src="../../../html/webappapis/scripting/reporterror.any.js"></script>
|
|
@ -0,0 +1,49 @@
|
||||||
|
setup({ allow_uncaught_exception:true });
|
||||||
|
|
||||||
|
[
|
||||||
|
1,
|
||||||
|
new TypeError(),
|
||||||
|
undefined
|
||||||
|
].forEach(throwable => {
|
||||||
|
test(t => {
|
||||||
|
let happened = false;
|
||||||
|
self.addEventListener("error", t.step_func(e => {
|
||||||
|
assert_true(e.message !== "");
|
||||||
|
assert_equals(e.filename, new URL("reporterror.any.js", location.href).href);
|
||||||
|
assert_greater_than(e.lineno, 0);
|
||||||
|
assert_greater_than(e.colno, 0);
|
||||||
|
assert_equals(e.error, throwable);
|
||||||
|
happened = true;
|
||||||
|
}), { once:true });
|
||||||
|
self.reportError(throwable);
|
||||||
|
assert_true(happened);
|
||||||
|
}, `self.reportError(${throwable})`);
|
||||||
|
});
|
||||||
|
|
||||||
|
test(() => {
|
||||||
|
assert_throws_js(TypeError, () => self.reportError());
|
||||||
|
}, `self.reportError() (without arguments) throws`);
|
||||||
|
|
||||||
|
test(() => {
|
||||||
|
// Workaround for https://github.com/web-platform-tests/wpt/issues/32105
|
||||||
|
let invoked = false;
|
||||||
|
self.reportError({
|
||||||
|
get name() {
|
||||||
|
invoked = true;
|
||||||
|
assert_unreached('get name')
|
||||||
|
},
|
||||||
|
get message() {
|
||||||
|
invoked = true;
|
||||||
|
assert_unreached('get message');
|
||||||
|
},
|
||||||
|
get fileName() {
|
||||||
|
invoked = true;
|
||||||
|
assert_unreached('get fileName');
|
||||||
|
},
|
||||||
|
get lineNumber() {
|
||||||
|
invoked = true;
|
||||||
|
assert_unreached('get lineNumber');
|
||||||
|
}
|
||||||
|
});
|
||||||
|
assert_false(invoked);
|
||||||
|
}, `self.reportError() doesn't invoke getters`);
|
Loading…
Reference in a new issue