/* * Copyright (c) 2025, Tim Flynn * * SPDX-License-Identifier: BSD-2-Clause */ #include #include #include #include #include #include #include #include #include #include #include #include #include namespace DevTools { NonnullRefPtr FrameActor::create(DevToolsServer& devtools, String name, WeakPtr tab, WeakPtr css_properties, WeakPtr console, WeakPtr inspector, WeakPtr style_sheets, WeakPtr thread) { return adopt_ref(*new FrameActor(devtools, move(name), move(tab), move(css_properties), move(console), move(inspector), move(style_sheets), move(thread))); } FrameActor::FrameActor(DevToolsServer& devtools, String name, WeakPtr tab, WeakPtr css_properties, WeakPtr console, WeakPtr inspector, WeakPtr style_sheets, WeakPtr thread) : Actor(devtools, move(name)) , m_tab(move(tab)) , m_css_properties(move(css_properties)) , m_console(move(console)) , m_inspector(move(inspector)) , m_style_sheets(move(style_sheets)) , m_thread(move(thread)) { if (auto tab = m_tab.strong_ref()) { devtools.delegate().listen_for_console_messages( tab->description(), [weak_self = make_weak_ptr()](i32 message_index) { if (auto self = weak_self.strong_ref()) self->console_message_available(message_index); }, [weak_self = make_weak_ptr()](i32 start_index, Vector console_output) { if (auto self = weak_self.strong_ref()) self->console_messages_received(start_index, move(console_output)); }); // FIXME: We should adopt WebContent to inform us when style sheets are available or removed. devtools.delegate().retrieve_style_sheets(tab->description(), async_handler({}, [](auto& self, auto style_sheets, auto& response) { self.style_sheets_available(response, move(style_sheets)); })); } } FrameActor::~FrameActor() { if (auto tab = m_tab.strong_ref()) devtools().delegate().stop_listening_for_console_messages(tab->description()); } void FrameActor::handle_message(Message const& message) { JsonObject response; if (message.type == "detach"sv) { if (auto tab = m_tab.strong_ref()) { devtools().delegate().stop_listening_for_dom_properties(tab->description()); devtools().delegate().stop_listening_for_dom_mutations(tab->description()); devtools().delegate().stop_listening_for_console_messages(tab->description()); devtools().delegate().stop_listening_for_style_sheet_sources(tab->description()); tab->reset_selected_node(); } send_response(message, move(response)); return; } if (message.type == "listFrames"sv) { send_response(message, move(response)); return; } send_unrecognized_packet_type_error(message); } void FrameActor::send_frame_update_message() { JsonArray frames; if (auto tab_actor = m_tab.strong_ref()) { JsonObject frame; frame.set("id"sv, tab_actor->description().id); frame.set("title"sv, tab_actor->description().title); frame.set("url"sv, tab_actor->description().url); frames.must_append(move(frame)); } JsonObject message; message.set("type"sv, "frameUpdate"sv); message.set("frames"sv, move(frames)); send_message(move(message)); } JsonObject FrameActor::serialize_target() const { JsonObject traits; traits.set("frames"sv, true); traits.set("isBrowsingContext"sv, true); traits.set("logInPage"sv, false); traits.set("navigation"sv, true); traits.set("supportsTopLevelTargetFlag"sv, true); traits.set("watchpoints"sv, true); JsonObject target; target.set("actor"sv, name()); target.set("targetType"sv, "frame"sv); if (auto tab_actor = m_tab.strong_ref()) { target.set("title"sv, tab_actor->description().title); target.set("url"sv, tab_actor->description().url); target.set("browsingContextID"sv, tab_actor->description().id); target.set("outerWindowID"sv, tab_actor->description().id); target.set("isTopLevelTarget"sv, true); } target.set("traits"sv, move(traits)); if (auto css_properties = m_css_properties.strong_ref()) target.set("cssPropertiesActor"sv, css_properties->name()); if (auto console = m_console.strong_ref()) target.set("consoleActor"sv, console->name()); if (auto inspector = m_inspector.strong_ref()) target.set("inspectorActor"sv, inspector->name()); if (auto style_sheets = m_style_sheets.strong_ref()) target.set("styleSheetsActor"sv, style_sheets->name()); if (auto thread = m_thread.strong_ref()) target.set("threadActor"sv, thread->name()); return target; } void FrameActor::style_sheets_available(JsonObject& response, Vector style_sheets) { JsonArray sheets; String tab_url; if (auto tab_actor = m_tab.strong_ref()) tab_url = tab_actor->description().url; auto style_sheets_actor = m_style_sheets.strong_ref(); if (!style_sheets_actor) return; for (auto const& [i, style_sheet] : enumerate(style_sheets)) { auto resource_id = MUST(String::formatted("{}-stylesheet:{}", style_sheets_actor->name(), i)); JsonValue href; JsonValue source_map_base_url; JsonValue title; if (style_sheet.url.has_value()) { // LibWeb sets the URL to a style sheet name for UA style sheets. DevTools would reject these invalid URLs. if (style_sheet.type == Web::CSS::StyleSheetIdentifier::Type::UserAgent) { title = *style_sheet.url; source_map_base_url = tab_url; } else { href = *style_sheet.url; source_map_base_url = *style_sheet.url; } } else { source_map_base_url = tab_url; } JsonObject sheet; sheet.set("atRules"sv, JsonArray {}); sheet.set("constructed"sv, false); sheet.set("disabled"sv, false); sheet.set("fileName"sv, JsonValue {}); sheet.set("href"sv, move(href)); sheet.set("isNew"sv, false); sheet.set("nodeHref"sv, tab_url); sheet.set("resourceId"sv, move(resource_id)); sheet.set("ruleCount"sv, style_sheet.rule_count); sheet.set("sourceMapBaseURL"sv, move(source_map_base_url)); sheet.set("sourceMapURL"sv, ""sv); sheet.set("styleSheetIndex"sv, i); sheet.set("system"sv, false); sheet.set("title"sv, move(title)); sheets.must_append(move(sheet)); } JsonArray stylesheets; stylesheets.must_append("stylesheet"sv); stylesheets.must_append(move(sheets)); JsonArray array; array.must_append(move(stylesheets)); response.set("type"sv, "resources-available-array"sv); response.set("array"sv, move(array)); style_sheets_actor->set_style_sheets(move(style_sheets)); } void FrameActor::console_message_available(i32 message_index) { if (message_index <= m_highest_received_message_index) { dbgln("Notified about console message we already have"); return; } if (message_index <= m_highest_notified_message_index) { dbgln("Notified about console message we're already aware of"); return; } m_highest_notified_message_index = message_index; if (!m_waiting_for_messages) request_console_messages(); } void FrameActor::console_messages_received(i32 start_index, Vector console_output) { auto end_index = start_index + static_cast(console_output.size()) - 1; if (end_index <= m_highest_received_message_index) { dbgln("Received old console messages"); return; } JsonArray console_messages; JsonArray error_messages; for (auto& output : console_output) { JsonObject message; output.output.visit( [&](WebView::ConsoleLog& log) { switch (log.level) { case JS::Console::LogLevel::Debug: message.set("level"sv, "debug"sv); break; case JS::Console::LogLevel::Error: message.set("level"sv, "error"sv); break; case JS::Console::LogLevel::Info: message.set("level"sv, "info"sv); break; case JS::Console::LogLevel::Log: message.set("level"sv, "log"sv); break; case JS::Console::LogLevel::Warn: message.set("level"sv, "warn"sv); break; default: // FIXME: Implement remaining console levels. return; } message.set("filename"sv, ""sv); message.set("lineNumber"sv, 1); message.set("columnNumber"sv, 1); message.set("timeStamp"sv, output.timestamp.milliseconds_since_epoch()); message.set("arguments"sv, JsonArray { move(log.arguments) }); console_messages.must_append(move(message)); }, [&](WebView::ConsoleError const& error) { StringBuilder stack; for (auto const& frame : error.trace) { if (frame.function.has_value()) stack.append(*frame.function); stack.append('@'); stack.append(frame.file.map([](auto const& file) -> StringView { return file; }).value_or("unknown"sv)); stack.appendff(":{}:{}\n", frame.line.value_or(0), frame.column.value_or(0)); } JsonObject preview; preview.set("kind"sv, "Error"sv); preview.set("message"sv, error.message); preview.set("name"sv, error.name); if (!stack.is_empty()) preview.set("stack"sv, MUST(stack.to_string())); JsonObject exception; exception.set("class"sv, error.name); exception.set("isError"sv, true); exception.set("preview"sv, move(preview)); JsonObject page_error; page_error.set("error"sv, true); page_error.set("exception"sv, move(exception)); page_error.set("hasException"sv, !error.trace.is_empty()); page_error.set("isPromiseRejection"sv, error.inside_promise); page_error.set("timeStamp"sv, output.timestamp.milliseconds_since_epoch()); message.set("pageError"sv, move(page_error)); error_messages.must_append(move(message)); }); } JsonArray array; if (!console_messages.is_empty()) { JsonArray console_message; console_message.must_append("console-message"sv); console_message.must_append(move(console_messages)); array.must_append(move(console_message)); } if (!error_messages.is_empty()) { JsonArray error_message; error_message.must_append("error-message"sv); error_message.must_append(move(error_messages)); array.must_append(move(error_message)); } JsonObject message; message.set("type"sv, "resources-available-array"sv); message.set("array"sv, move(array)); send_message(move(message)); m_highest_received_message_index = end_index; m_waiting_for_messages = false; if (m_highest_received_message_index < m_highest_notified_message_index) request_console_messages(); } void FrameActor::request_console_messages() { VERIFY(!m_waiting_for_messages); if (auto tab = m_tab.strong_ref()) { devtools().delegate().request_console_messages(m_tab->description(), m_highest_received_message_index + 1); m_waiting_for_messages = true; } } }