mirror of
https://github.com/LadybirdBrowser/ladybird.git
synced 2024-11-22 23:50:19 +00:00
LibWeb+WebContent: Update and fully implement the JSON clone algorithm
We have the facilities now to fully implement this AO. Do so, and update the AO to match the latest spec.
This commit is contained in:
parent
3d0bbb4bcf
commit
a5ca036d36
Notes:
github-actions[bot]
2024-11-03 17:08:51 +00:00
Author: https://github.com/trflynn89 Commit: https://github.com/LadybirdBrowser/ladybird/commit/a5ca036d366 Pull-request: https://github.com/LadybirdBrowser/ladybird/pull/2137
3 changed files with 99 additions and 59 deletions
|
@ -23,6 +23,7 @@
|
||||||
#include <LibWeb/DOM/Document.h>
|
#include <LibWeb/DOM/Document.h>
|
||||||
#include <LibWeb/DOM/HTMLCollection.h>
|
#include <LibWeb/DOM/HTMLCollection.h>
|
||||||
#include <LibWeb/DOM/NodeList.h>
|
#include <LibWeb/DOM/NodeList.h>
|
||||||
|
#include <LibWeb/DOM/ShadowRoot.h>
|
||||||
#include <LibWeb/FileAPI/FileList.h>
|
#include <LibWeb/FileAPI/FileList.h>
|
||||||
#include <LibWeb/HTML/BrowsingContext.h>
|
#include <LibWeb/HTML/BrowsingContext.h>
|
||||||
#include <LibWeb/HTML/HTMLAllCollection.h>
|
#include <LibWeb/HTML/HTMLAllCollection.h>
|
||||||
|
@ -51,8 +52,10 @@ namespace Web::WebDriver {
|
||||||
_temporary_result.release_value(); \
|
_temporary_result.release_value(); \
|
||||||
})
|
})
|
||||||
|
|
||||||
static ErrorOr<JsonValue, ExecuteScriptResultType> internal_json_clone_algorithm(JS::Realm&, HTML::BrowsingContext const&, JS::Value, HashTable<JS::Object*>& seen);
|
using SeenMap = HashTable<JS::RawGCPtr<JS::Object const>>;
|
||||||
static ErrorOr<JsonValue, ExecuteScriptResultType> clone_an_object(JS::Realm&, HTML::BrowsingContext const&, JS::Object&, HashTable<JS::Object*>& seen, auto const& clone_algorithm);
|
|
||||||
|
static ErrorOr<JsonValue, ExecuteScriptResultType> internal_json_clone(JS::Realm&, HTML::BrowsingContext const&, JS::Value, SeenMap& seen);
|
||||||
|
static ErrorOr<JsonValue, ExecuteScriptResultType> clone_an_object(JS::Realm&, HTML::BrowsingContext const&, JS::Object const&, SeenMap& seen, auto const& clone_algorithm);
|
||||||
|
|
||||||
// https://w3c.github.io/webdriver/#dfn-collection
|
// https://w3c.github.io/webdriver/#dfn-collection
|
||||||
static bool is_collection(JS::Object const& value)
|
static bool is_collection(JS::Object const& value)
|
||||||
|
@ -82,28 +85,30 @@ static bool is_collection(JS::Object const& value)
|
||||||
// https://w3c.github.io/webdriver/#dfn-json-clone
|
// https://w3c.github.io/webdriver/#dfn-json-clone
|
||||||
static ErrorOr<JsonValue, ExecuteScriptResultType> json_clone(JS::Realm& realm, HTML::BrowsingContext const& browsing_context, JS::Value value)
|
static ErrorOr<JsonValue, ExecuteScriptResultType> json_clone(JS::Realm& realm, HTML::BrowsingContext const& browsing_context, JS::Value value)
|
||||||
{
|
{
|
||||||
// To perform a JSON clone return the result of calling the internal JSON clone algorithm with arguments value and an empty List.
|
// To JSON clone given session and value, return the result of internal JSON clone with session, value and an empty List.
|
||||||
auto seen = HashTable<JS::Object*> {};
|
SeenMap seen;
|
||||||
return internal_json_clone_algorithm(realm, browsing_context, value, seen);
|
return internal_json_clone(realm, browsing_context, value, seen);
|
||||||
}
|
}
|
||||||
|
|
||||||
// https://w3c.github.io/webdriver/#dfn-internal-json-clone-algorithm
|
// https://w3c.github.io/webdriver/#dfn-internal-json-clone
|
||||||
static ErrorOr<JsonValue, ExecuteScriptResultType> internal_json_clone_algorithm(JS::Realm& realm, HTML::BrowsingContext const& browsing_context, JS::Value value, HashTable<JS::Object*>& seen)
|
static ErrorOr<JsonValue, ExecuteScriptResultType> internal_json_clone(JS::Realm& realm, HTML::BrowsingContext const& browsing_context, JS::Value value, SeenMap& seen)
|
||||||
{
|
{
|
||||||
auto& vm = realm.vm();
|
auto& vm = realm.vm();
|
||||||
|
|
||||||
// When required to run the internal JSON clone algorithm with arguments value and seen, a remote end must return the value of the first matching statement, matching on value:
|
// To internal JSON clone given session, value and seen, return the value of the first matching statement, matching
|
||||||
|
// on value:
|
||||||
|
|
||||||
// -> undefined
|
// -> undefined
|
||||||
// -> null
|
// -> null
|
||||||
if (value.is_nullish()) {
|
if (value.is_nullish()) {
|
||||||
// Success with data null.
|
// Return success with data null.
|
||||||
return JsonValue {};
|
return JsonValue {};
|
||||||
}
|
}
|
||||||
|
|
||||||
// -> type Boolean
|
// -> type Boolean
|
||||||
// -> type Number
|
// -> type Number
|
||||||
// -> type String
|
// -> type String
|
||||||
// Success with data value.
|
// Return success with data value.
|
||||||
if (value.is_boolean())
|
if (value.is_boolean())
|
||||||
return JsonValue { value.as_bool() };
|
return JsonValue { value.as_bool() };
|
||||||
if (value.is_number())
|
if (value.is_number())
|
||||||
|
@ -111,16 +116,17 @@ static ErrorOr<JsonValue, ExecuteScriptResultType> internal_json_clone_algorithm
|
||||||
if (value.is_string())
|
if (value.is_string())
|
||||||
return JsonValue { value.as_string().byte_string() };
|
return JsonValue { value.as_string().byte_string() };
|
||||||
|
|
||||||
// NOTE: BigInt and Symbol not mentioned anywhere in the WebDriver spec, as it references ES5.
|
// AD-HOC: BigInt and Symbol not mentioned anywhere in the WebDriver spec, as it references ES5.
|
||||||
// It assumes that all primitives are handled above, and the value is an object for the remaining steps.
|
// It assumes that all primitives are handled above, and the value is an object for the remaining steps.
|
||||||
if (value.is_bigint() || value.is_symbol())
|
if (value.is_bigint() || value.is_symbol())
|
||||||
return ExecuteScriptResultType::JavaScriptError;
|
return ExecuteScriptResultType::JavaScriptError;
|
||||||
|
|
||||||
// FIXME: -> a collection
|
VERIFY(value.is_object());
|
||||||
|
auto const& object = static_cast<JS::Object const&>(value.as_object());
|
||||||
|
|
||||||
// -> instance of element
|
// -> instance of Element
|
||||||
if (value.is_object() && is<DOM::Element>(value.as_object())) {
|
if (is<DOM::Element>(object)) {
|
||||||
auto const& element = static_cast<DOM::Element const&>(value.as_object());
|
auto const& element = static_cast<DOM::Element const&>(object);
|
||||||
|
|
||||||
// If the element is stale, return error with error code stale element reference.
|
// If the element is stale, return error with error code stale element reference.
|
||||||
if (is_element_stale(element)) {
|
if (is_element_stale(element)) {
|
||||||
|
@ -136,57 +142,74 @@ static ErrorOr<JsonValue, ExecuteScriptResultType> internal_json_clone_algorithm
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// FIXME: -> instance of shadow root
|
// -> instance of ShadowRoot
|
||||||
|
if (is<DOM::ShadowRoot>(object)) {
|
||||||
|
auto const& shadow_root = static_cast<DOM::ShadowRoot const&>(object);
|
||||||
|
|
||||||
|
// If the shadow root is detached, return error with error code detached shadow root.
|
||||||
|
if (is_shadow_root_detached(shadow_root)) {
|
||||||
|
return ExecuteScriptResultType::DetachedShadowRoot;
|
||||||
|
}
|
||||||
|
// Otherwise:
|
||||||
|
else {
|
||||||
|
// 1. Let reference be the shadow root reference object for session and value.
|
||||||
|
auto reference = shadow_root_reference_object(browsing_context, shadow_root);
|
||||||
|
|
||||||
|
// 2. Return success with data reference.
|
||||||
|
return reference;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
// -> a WindowProxy object
|
// -> a WindowProxy object
|
||||||
if (is<HTML::WindowProxy>(value.as_object())) {
|
if (is<HTML::WindowProxy>(object)) {
|
||||||
auto const& window_proxy = static_cast<HTML::WindowProxy&>(value.as_object());
|
auto const& window_proxy = static_cast<HTML::WindowProxy const&>(object);
|
||||||
|
|
||||||
// If the associated browsing context of the WindowProxy object in value has been destroyed, return error with
|
// If the associated browsing context of the WindowProxy object in value has been destroyed, return error
|
||||||
// error code stale element reference.
|
// with error code stale element reference.
|
||||||
if (window_proxy.associated_browsing_context()->has_navigable_been_destroyed())
|
if (window_proxy.associated_browsing_context()->has_navigable_been_destroyed()) {
|
||||||
return ExecuteScriptResultType::BrowsingContextDiscarded;
|
return ExecuteScriptResultType::BrowsingContextDiscarded;
|
||||||
|
}
|
||||||
|
// Otherwise:
|
||||||
|
else {
|
||||||
|
// 1. Let reference be the WindowProxy reference object for value.
|
||||||
|
auto reference = window_proxy_reference_object(window_proxy);
|
||||||
|
|
||||||
// Otherwise return success with data set to WindowProxy reference object for value.
|
// 2. Return success with data reference.
|
||||||
return window_proxy_reference_object(window_proxy);
|
return reference;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// -> has an own property named "toJSON" that is a Function
|
// -> has an own property named "toJSON" that is a Function
|
||||||
auto to_json = value.as_object().get_without_side_effects(vm.names.toJSON);
|
if (auto to_json = object.get_without_side_effects(vm.names.toJSON); to_json.is_function()) {
|
||||||
if (to_json.is_function()) {
|
|
||||||
// Return success with the value returned by Function.[[Call]](toJSON) with value as the this value.
|
// Return success with the value returned by Function.[[Call]](toJSON) with value as the this value.
|
||||||
auto to_json_result = TRY_OR_JS_ERROR(to_json.as_function().internal_call(value, JS::MarkedVector<JS::Value> { vm.heap() }));
|
auto to_json_result = TRY_OR_JS_ERROR(to_json.as_function().internal_call(value, JS::MarkedVector<JS::Value> { vm.heap() }));
|
||||||
if (!to_json_result.is_string())
|
if (!to_json_result.is_string())
|
||||||
return ExecuteScriptResultType::JavaScriptError;
|
return ExecuteScriptResultType::JavaScriptError;
|
||||||
|
|
||||||
return to_json_result.as_string().byte_string();
|
return to_json_result.as_string().byte_string();
|
||||||
}
|
}
|
||||||
|
|
||||||
// -> Otherwise
|
// -> Otherwise
|
||||||
// 1. If value is in seen, return error with error code javascript error.
|
// 1. Let result be clone an object with session value and seen, and internal JSON clone as the clone algorithm.
|
||||||
if (seen.contains(&value.as_object()))
|
auto result = TRY(clone_an_object(realm, browsing_context, object, seen, internal_json_clone));
|
||||||
return ExecuteScriptResultType::JavaScriptError;
|
|
||||||
|
|
||||||
// 2. Append value to seen.
|
// 2. Return success with data result.
|
||||||
seen.set(&value.as_object());
|
|
||||||
|
|
||||||
ScopeGuard remove_seen { [&] {
|
|
||||||
// 4. Remove the last element of seen.
|
|
||||||
seen.remove(&value.as_object());
|
|
||||||
} };
|
|
||||||
|
|
||||||
// 3. Let result be the value of running the clone an object algorithm with arguments value and seen, and the internal JSON clone algorithm as the clone algorithm.
|
|
||||||
auto result = TRY(clone_an_object(realm, browsing_context, value.as_object(), seen, internal_json_clone_algorithm));
|
|
||||||
|
|
||||||
// 5. Return result.
|
|
||||||
return result;
|
return result;
|
||||||
}
|
}
|
||||||
|
|
||||||
// https://w3c.github.io/webdriver/#dfn-clone-an-object
|
// https://w3c.github.io/webdriver/#dfn-clone-an-object
|
||||||
static ErrorOr<JsonValue, ExecuteScriptResultType> clone_an_object(JS::Realm& realm, HTML::BrowsingContext const& browsing_context, JS::Object& value, HashTable<JS::Object*>& seen, auto const& clone_algorithm)
|
static ErrorOr<JsonValue, ExecuteScriptResultType> clone_an_object(JS::Realm& realm, HTML::BrowsingContext const& browsing_context, JS::Object const& value, SeenMap& seen, auto const& clone_algorithm)
|
||||||
{
|
{
|
||||||
auto& vm = realm.vm();
|
auto& vm = realm.vm();
|
||||||
|
|
||||||
// 1. Let result be the value of the first matching statement, matching on value:
|
// 1. If value is in seen, return error with error code javascript error.
|
||||||
|
if (seen.contains(value))
|
||||||
|
return ExecuteScriptResultType::JavaScriptError;
|
||||||
|
|
||||||
|
// 2. Append value to seen.
|
||||||
|
seen.set(value);
|
||||||
|
|
||||||
|
// 3. Let result be the value of the first matching statement, matching on value:
|
||||||
auto result = TRY(([&]() -> ErrorOr<Variant<JsonArray, JsonObject>, ExecuteScriptResultType> {
|
auto result = TRY(([&]() -> ErrorOr<Variant<JsonArray, JsonObject>, ExecuteScriptResultType> {
|
||||||
// -> a collection
|
// -> a collection
|
||||||
if (is_collection(value)) {
|
if (is_collection(value)) {
|
||||||
|
@ -210,23 +233,27 @@ static ErrorOr<JsonValue, ExecuteScriptResultType> clone_an_object(JS::Realm& re
|
||||||
}
|
}
|
||||||
}()));
|
}()));
|
||||||
|
|
||||||
// 2. For each enumerable own property in value, run the following substeps:
|
Optional<ExecuteScriptResultType> error;
|
||||||
for (auto& key : MUST(value.Object::internal_own_property_keys())) {
|
|
||||||
|
// 4. For each enumerable property in value, run the following substeps:
|
||||||
|
(void)value.enumerate_object_properties([&](auto property) -> Optional<JS::Completion> {
|
||||||
// 1. Let name be the name of the property.
|
// 1. Let name be the name of the property.
|
||||||
auto name = MUST(JS::PropertyKey::from_value(vm, key));
|
auto name = MUST(JS::PropertyKey::from_value(vm, property));
|
||||||
|
|
||||||
if (!value.storage_get(name)->attributes.is_enumerable())
|
// 2. Let source property value be the result of getting a property named name from value. If doing so causes
|
||||||
continue;
|
// script to be run and that script throws an error, return error with error code javascript error.
|
||||||
|
auto source_property_value = value.get(name);
|
||||||
|
if (source_property_value.is_error()) {
|
||||||
|
error = ExecuteScriptResultType::JavaScriptError;
|
||||||
|
return JS::normal_completion({});
|
||||||
|
}
|
||||||
|
|
||||||
// 2. Let source property value be the result of getting a property named name from value. If doing so causes script to be run and that script throws an error, return error with error code javascript error.
|
// 3. Let cloned property result be the result of calling the clone algorithm with session, source property
|
||||||
auto source_property_value = TRY_OR_JS_ERROR(value.internal_get_own_property(name));
|
// value and seen.
|
||||||
if (!source_property_value.has_value() || !source_property_value->value.has_value())
|
auto cloned_property_result = clone_algorithm(realm, browsing_context, source_property_value.value(), seen);
|
||||||
continue;
|
|
||||||
|
|
||||||
// 3. Let cloned property result be the result of calling the clone algorithm with arguments source property value and seen.
|
// 4. If cloned property result is a success, set a property of result with name name and value equal to cloned
|
||||||
auto cloned_property_result = clone_algorithm(realm, browsing_context, *source_property_value->value, seen);
|
// property result's data.
|
||||||
|
|
||||||
// 4. If cloned property result is a success, set a property of result with name name and value equal to cloned property result’s data.
|
|
||||||
if (!cloned_property_result.is_error()) {
|
if (!cloned_property_result.is_error()) {
|
||||||
result.visit(
|
result.visit(
|
||||||
[&](JsonArray& array) {
|
[&](JsonArray& array) {
|
||||||
|
@ -240,11 +267,21 @@ static ErrorOr<JsonValue, ExecuteScriptResultType> clone_an_object(JS::Realm& re
|
||||||
}
|
}
|
||||||
// 5. Otherwise, return cloned property result.
|
// 5. Otherwise, return cloned property result.
|
||||||
else {
|
else {
|
||||||
return cloned_property_result;
|
error = cloned_property_result.release_error();
|
||||||
|
return JS::normal_completion({});
|
||||||
}
|
}
|
||||||
}
|
|
||||||
|
|
||||||
return result.visit([&](auto const& value) -> JsonValue { return value; });
|
return {};
|
||||||
|
});
|
||||||
|
|
||||||
|
if (error.has_value())
|
||||||
|
return error.release_value();
|
||||||
|
|
||||||
|
// 5. Remove the last element of seen.
|
||||||
|
seen.remove(value);
|
||||||
|
|
||||||
|
// 6. Return success with data result.
|
||||||
|
return result.visit([&](auto& value) { return JsonValue { move(value) }; });
|
||||||
}
|
}
|
||||||
|
|
||||||
// https://w3c.github.io/webdriver/#dfn-execute-a-function-body
|
// https://w3c.github.io/webdriver/#dfn-execute-a-function-body
|
||||||
|
|
|
@ -22,6 +22,7 @@ enum class ExecuteScriptResultType {
|
||||||
JavaScriptError,
|
JavaScriptError,
|
||||||
BrowsingContextDiscarded,
|
BrowsingContextDiscarded,
|
||||||
StaleElement,
|
StaleElement,
|
||||||
|
DetachedShadowRoot,
|
||||||
};
|
};
|
||||||
|
|
||||||
struct ExecuteScriptResult {
|
struct ExecuteScriptResult {
|
||||||
|
|
|
@ -1950,6 +1950,8 @@ void WebDriverConnection::handle_script_response(Web::WebDriver::ExecuteScriptRe
|
||||||
return Web::WebDriver::Error::from_code(Web::WebDriver::ErrorCode::StaleElementReference, "Browsing context has been discarded", move(result.value));
|
return Web::WebDriver::Error::from_code(Web::WebDriver::ErrorCode::StaleElementReference, "Browsing context has been discarded", move(result.value));
|
||||||
case Web::WebDriver::ExecuteScriptResultType::StaleElement:
|
case Web::WebDriver::ExecuteScriptResultType::StaleElement:
|
||||||
return Web::WebDriver::Error::from_code(Web::WebDriver::ErrorCode::StaleElementReference, "Referenced element has become stale", move(result.value));
|
return Web::WebDriver::Error::from_code(Web::WebDriver::ErrorCode::StaleElementReference, "Referenced element has become stale", move(result.value));
|
||||||
|
case Web::WebDriver::ExecuteScriptResultType::DetachedShadowRoot:
|
||||||
|
return Web::WebDriver::Error::from_code(Web::WebDriver::ErrorCode::DetachedShadowRoot, "Referenced shadow root has become detached", move(result.value));
|
||||||
}
|
}
|
||||||
|
|
||||||
VERIFY_NOT_REACHED();
|
VERIFY_NOT_REACHED();
|
||||||
|
|
Loading…
Reference in a new issue