mirror of
https://github.com/LadybirdBrowser/ladybird.git
synced 2025-07-24 09:52:31 +00:00
Lots of editorial spec bugs here, but these changes largely affect how the unhandledPromptBehavior capability is handled. We also now set an additional capability for the default User Agent string.
285 lines
13 KiB
C++
285 lines
13 KiB
C++
/*
|
||
* Copyright (c) 2022, Florent Castelli <florent.castelli@gmail.com>
|
||
* Copyright (c) 2022, Sam Atkins <atkinssj@serenityos.org>
|
||
* Copyright (c) 2022, Tobias Christiansen <tobyase@serenityos.org>
|
||
* Copyright (c) 2022, Linus Groh <linusg@serenityos.org>
|
||
* Copyright (c) 2022-2025, Tim Flynn <trflynn89@ladybird.org>
|
||
*
|
||
* SPDX-License-Identifier: BSD-2-Clause
|
||
*/
|
||
|
||
#include "Session.h"
|
||
#include "Client.h"
|
||
#include <AK/JsonObject.h>
|
||
#include <LibCore/LocalServer.h>
|
||
#include <LibCore/StandardPaths.h>
|
||
#include <LibCore/System.h>
|
||
#include <LibWeb/Crypto/Crypto.h>
|
||
#include <LibWeb/WebDriver/TimeoutsConfiguration.h>
|
||
#include <LibWeb/WebDriver/UserPrompt.h>
|
||
#include <unistd.h>
|
||
|
||
namespace WebDriver {
|
||
|
||
// https://w3c.github.io/webdriver/#dfn-create-a-session
|
||
ErrorOr<NonnullRefPtr<Session>> Session::create(NonnullRefPtr<Client> client, JsonObject& capabilities, ReadonlySpan<StringView> flags)
|
||
{
|
||
// 1. Let session id be the result of generating a UUID.
|
||
auto session_id = MUST(Web::Crypto::generate_random_uuid());
|
||
|
||
// 2. Let session be a new session with session ID session id, and HTTP flag flags contains "http".
|
||
auto session = adopt_ref(*new Session(client, capabilities, move(session_id), flags.contains_slow("http"sv)));
|
||
TRY(session->start(client->launch_browser_callbacks()));
|
||
|
||
// FIXME: 3. Let proxy be the result of getting property "proxy" from capabilities and run the substeps of the first matching statement:
|
||
// -> proxy is a proxy configuration object
|
||
// Take implementation-defined steps to set the user agent proxy using the extracted proxy configuration. If the defined proxy cannot be configured return error with error code session not created. Otherwise set the has proxy configuration flag to true.
|
||
// -> Otherwise
|
||
// Set a property of capabilities with name "proxy" and a value that is a new JSON Object.
|
||
|
||
// FIXME: 4. If capabilites has a property named "acceptInsecureCerts", set the endpoint node's accept insecure TLS flag
|
||
// to the result of getting a property named "acceptInsecureCerts" from capabilities.
|
||
|
||
// 5. Let user prompt handler capability be the result of getting property "unhandledPromptBehavior" from capabilities.
|
||
auto user_prompt_handler_capability = capabilities.get_object("unhandledPromptBehavior"sv);
|
||
|
||
// 6. If user prompt handler capability is not undefined, update the user prompt handler with user prompt handler capability.
|
||
if (user_prompt_handler_capability.has_value())
|
||
Web::WebDriver::update_the_user_prompt_handler(*user_prompt_handler_capability);
|
||
|
||
session->web_content_connection().async_set_user_prompt_handler(Web::WebDriver::user_prompt_handler());
|
||
|
||
// 7. Let serialized user prompt handler be serialize the user prompt handler.
|
||
auto serialized_user_prompt_handler = Web::WebDriver::serialize_the_user_prompt_handler();
|
||
|
||
// 8. Set a property on capabilities with the name "unhandledPromptBehavior", and the value serialized user prompt handler.
|
||
capabilities.set("unhandledPromptBehavior"sv, move(serialized_user_prompt_handler));
|
||
|
||
// 9. If flags contains "http":
|
||
if (flags.contains_slow("http"sv)) {
|
||
// 1. Let strategy be the result of getting property "pageLoadStrategy" from capabilities. If strategy is a
|
||
// string, set the session's page loading strategy to strategy. Otherwise, set the page loading strategy to
|
||
// normal and set a property of capabilities with name "pageLoadStrategy" and value "normal".
|
||
if (auto strategy = capabilities.get_byte_string("pageLoadStrategy"sv); strategy.has_value()) {
|
||
session->m_page_load_strategy = Web::WebDriver::page_load_strategy_from_string(*strategy);
|
||
session->web_content_connection().async_set_page_load_strategy(session->m_page_load_strategy);
|
||
} else {
|
||
capabilities.set("pageLoadStrategy"sv, "normal"sv);
|
||
}
|
||
|
||
// 3. Let strictFileInteractability be the result of getting property "strictFileInteractability" from .
|
||
// capabilities. If strictFileInteractability is a boolean, set session's strict file interactability to
|
||
// strictFileInteractability.
|
||
if (auto strict_file_interactiblity = capabilities.get_bool("strictFileInteractability"sv); strict_file_interactiblity.has_value()) {
|
||
session->m_strict_file_interactiblity = *strict_file_interactiblity;
|
||
session->web_content_connection().async_set_strict_file_interactability(session->m_strict_file_interactiblity);
|
||
}
|
||
|
||
// 4. Let timeouts be the result of getting a property "timeouts" from capabilities. If timeouts is not
|
||
// undefined, set session's session timeouts to timeouts.
|
||
if (auto timeouts = capabilities.get_object("timeouts"sv); timeouts.has_value()) {
|
||
MUST(session->set_timeouts(*timeouts));
|
||
}
|
||
|
||
// 5. Set a property on capabilities with name "timeouts" and value serialize the timeouts configuration with
|
||
// session's session timeouts.
|
||
capabilities.set("timeouts"sv, session->m_timeouts_configuration.value_or_lazy_evaluated([]() {
|
||
return Web::WebDriver::timeouts_object({});
|
||
}));
|
||
}
|
||
|
||
// FIXME: 10. Process any extension capabilities in capabilities in an implementation-defined manner.
|
||
|
||
// FIXME: 11. Run any WebDriver new session algorithm defined in external specifications, with arguments session, capabilities, and flags.
|
||
|
||
// 12. Append session to active sessions.
|
||
// 13. If flags contains "http", append session to active HTTP sessions.
|
||
// NOTE: These steps are handled by WebDriver::Client.
|
||
|
||
// 14. Set the webdriver-active flag to true.
|
||
session->web_content_connection().async_set_is_webdriver_active(true);
|
||
|
||
return session;
|
||
}
|
||
|
||
Session::Session(NonnullRefPtr<Client> client, JsonObject const& capabilities, String session_id, bool http)
|
||
: m_client(move(client))
|
||
, m_options(capabilities)
|
||
, m_id(move(session_id))
|
||
, m_http(http)
|
||
{
|
||
}
|
||
|
||
// https://w3c.github.io/webdriver/#dfn-close-the-session
|
||
Session::~Session()
|
||
{
|
||
if (!m_started)
|
||
return;
|
||
|
||
// 1. Perform the following substeps based on the remote end’s type:
|
||
// NOTE: We perform the "Remote end is an endpoint node" steps in the WebContent process.
|
||
for (auto& it : m_windows) {
|
||
it.value.web_content_connection->close_session();
|
||
}
|
||
|
||
// 2. Remove the current session from active sessions.
|
||
// NOTE: We are in a session destruction which means it is already removed
|
||
// from active sessions
|
||
|
||
// 3. Perform any implementation-specific cleanup steps.
|
||
if (m_browser_process.has_value())
|
||
MUST(Core::System::kill(m_browser_process->pid(), SIGTERM));
|
||
|
||
if (m_web_content_socket_path.has_value()) {
|
||
MUST(Core::System::unlink(*m_web_content_socket_path));
|
||
m_web_content_socket_path = {};
|
||
}
|
||
}
|
||
|
||
ErrorOr<NonnullRefPtr<Core::LocalServer>> Session::create_server(NonnullRefPtr<ServerPromise> promise)
|
||
{
|
||
static_assert(IsSame<IPC::Transport, IPC::TransportSocket>, "Need to handle other IPC transports here");
|
||
|
||
dbgln("Listening for WebDriver connection on {}", *m_web_content_socket_path);
|
||
|
||
(void)Core::System::unlink(*m_web_content_socket_path);
|
||
|
||
auto server = TRY(Core::LocalServer::try_create());
|
||
server->listen(*m_web_content_socket_path);
|
||
|
||
server->on_accept = [this, promise](auto client_socket) {
|
||
auto maybe_connection = adopt_nonnull_ref_or_enomem(new (nothrow) WebContentConnection(IPC::Transport(move(client_socket))));
|
||
if (maybe_connection.is_error()) {
|
||
promise->resolve(maybe_connection.release_error());
|
||
return;
|
||
}
|
||
|
||
dbgln("WebDriver is connected to WebContent socket");
|
||
auto web_content_connection = maybe_connection.release_value();
|
||
|
||
auto maybe_window_handle = web_content_connection->get_window_handle();
|
||
if (maybe_window_handle.is_error()) {
|
||
promise->reject(Error::from_string_literal("Window was closed immediately"));
|
||
return;
|
||
}
|
||
|
||
auto window_handle = MUST(String::from_byte_string(maybe_window_handle.value().as_string()));
|
||
|
||
web_content_connection->on_close = [this, window_handle]() {
|
||
dbgln_if(WEBDRIVER_DEBUG, "Window {} was closed remotely.", window_handle);
|
||
m_windows.remove(window_handle);
|
||
if (m_windows.is_empty())
|
||
m_client->close_session(session_id());
|
||
};
|
||
|
||
web_content_connection->async_set_page_load_strategy(m_page_load_strategy);
|
||
web_content_connection->async_set_strict_file_interactability(m_strict_file_interactiblity);
|
||
web_content_connection->async_set_user_prompt_handler(Web::WebDriver::user_prompt_handler());
|
||
if (m_timeouts_configuration.has_value())
|
||
web_content_connection->async_set_timeouts(*m_timeouts_configuration);
|
||
|
||
m_windows.set(window_handle, Session::Window { window_handle, move(web_content_connection) });
|
||
|
||
if (m_current_window_handle.is_empty())
|
||
m_current_window_handle = window_handle;
|
||
|
||
promise->resolve({});
|
||
};
|
||
|
||
server->on_accept_error = [promise](auto error) {
|
||
promise->resolve(move(error));
|
||
};
|
||
|
||
return server;
|
||
}
|
||
|
||
ErrorOr<void> Session::start(LaunchBrowserCallbacks const& callbacks)
|
||
{
|
||
auto promise = TRY(ServerPromise::try_create());
|
||
|
||
m_web_content_socket_path = ByteString::formatted("{}/webdriver/session_{}_{}", TRY(Core::StandardPaths::runtime_directory()), getpid(), m_id);
|
||
m_web_content_server = TRY(create_server(promise));
|
||
|
||
if (m_options.headless)
|
||
m_browser_process = TRY(callbacks.launch_headless_browser(*m_web_content_socket_path));
|
||
else
|
||
m_browser_process = TRY(callbacks.launch_browser(*m_web_content_socket_path));
|
||
|
||
// FIXME: Allow this to be more asynchronous. For now, this at least allows us to propagate
|
||
// errors received while accepting the Browser and WebContent sockets.
|
||
TRY(TRY(promise->await()));
|
||
|
||
m_started = true;
|
||
return {};
|
||
}
|
||
|
||
Web::WebDriver::Response Session::set_timeouts(JsonValue payload)
|
||
{
|
||
m_timeouts_configuration = TRY(web_content_connection().set_timeouts(move(payload)));
|
||
return JsonValue {};
|
||
}
|
||
|
||
// 11.2 Close Window, https://w3c.github.io/webdriver/#dfn-close-window
|
||
Web::WebDriver::Response Session::close_window()
|
||
{
|
||
// 3. Close the current top-level browsing context.
|
||
TRY(perform_async_action([&](auto& connection) {
|
||
return connection.close_window();
|
||
}));
|
||
|
||
{
|
||
// Defer removing the window handle from this session until after we know we are done with its connection.
|
||
ScopeGuard guard { [this] { m_windows.remove(m_current_window_handle); m_current_window_handle = "NoSuchWindowPleaseSelectANewOne"_string; } };
|
||
|
||
// 4. If there are no more open top-level browsing contexts, then close the session.
|
||
if (m_windows.size() == 1)
|
||
m_client->close_session(session_id());
|
||
}
|
||
|
||
// 5. Return the result of running the remote end steps for the Get Window Handles command.
|
||
return get_window_handles();
|
||
}
|
||
|
||
// 11.3 Switch to Window, https://w3c.github.io/webdriver/#dfn-switch-to-window
|
||
Web::WebDriver::Response Session::switch_to_window(StringView handle)
|
||
{
|
||
// 4. If handle is equal to the associated window handle for some top-level browsing context in the
|
||
// current session, let context be the that browsing context, and set the current top-level
|
||
// browsing context with context.
|
||
// Otherwise, return error with error code no such window.
|
||
if (auto it = m_windows.find(handle); it != m_windows.end())
|
||
m_current_window_handle = it->key;
|
||
else
|
||
return Web::WebDriver::Error::from_code(Web::WebDriver::ErrorCode::NoSuchWindow, "Window not found");
|
||
|
||
// 5. Update any implementation-specific state that would result from the user selecting the current
|
||
// browsing context for interaction, without altering OS-level focus.
|
||
TRY(web_content_connection().switch_to_window(m_current_window_handle));
|
||
|
||
// 6. Return success with data null.
|
||
return JsonValue {};
|
||
}
|
||
|
||
// 11.4 Get Window Handles, https://w3c.github.io/webdriver/#dfn-get-window-handles
|
||
Web::WebDriver::Response Session::get_window_handles() const
|
||
{
|
||
// 1. Let handles be a JSON List.
|
||
JsonArray handles {};
|
||
|
||
// 2. For each top-level browsing context in the remote end, push the associated window handle onto handles.
|
||
for (auto const& window_handle : m_windows.keys()) {
|
||
handles.must_append(JsonValue(window_handle));
|
||
}
|
||
|
||
// 3. Return success with data handles.
|
||
return JsonValue { move(handles) };
|
||
}
|
||
|
||
ErrorOr<void, Web::WebDriver::Error> Session::ensure_current_window_handle_is_valid() const
|
||
{
|
||
if (auto current_window = m_windows.get(m_current_window_handle); current_window.has_value())
|
||
return {};
|
||
return Web::WebDriver::Error::from_code(Web::WebDriver::ErrorCode::NoSuchWindow, "Window not found"sv);
|
||
}
|
||
|
||
}
|