mirror of
				https://github.com/LadybirdBrowser/ladybird.git
				synced 2025-10-22 16:09:23 +00:00 
			
		
		
		
	We must reply to requests received from the client in the order they are received. The wrench in this requirement is handling requests that must be performed asynchronously, such as fetching the serialized DOM tree from the WebContent process. We currently handle this with a "block token". Async request handlers hold a token that blocks any subsequent responses from being sent. When that token is removed (i.e. the async request now has a response to be sent), the async response is then sent followed by the blocked responses in-order. This strategy had a limitation that we could not handle an actor trying to take 2 block tokens, meaning only one async request could be handled at a time. This has been fine so far, but an upcoming feature (style sheet sources) will break this limitation. The client will request N sources at a time, which would try to take N block tokens. The new strategy is to assign all requests an ID, and store a list of request IDs that are awaiting a response. When the server wants to send a reply, we match the ID of the replied-to message to this list of IDs. If it is not the first in this list, then we are blocked waiting for an earlier reply, and just store the response. When the earlier request(s) receive their response, we can then send out all blocked replies (up to the next request that has not yet received a response).
		
			
				
	
	
		
			168 lines
		
	
	
	
		
			4.8 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			168 lines
		
	
	
	
		
			4.8 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| /*
 | |
|  * Copyright (c) 2025, Tim Flynn <trflynn89@ladybird.org>
 | |
|  *
 | |
|  * SPDX-License-Identifier: BSD-2-Clause
 | |
|  */
 | |
| 
 | |
| #include <AK/JsonObject.h>
 | |
| #include <LibDevTools/Actors/DeviceActor.h>
 | |
| #include <LibDevTools/Actors/PreferenceActor.h>
 | |
| #include <LibDevTools/Actors/ProcessActor.h>
 | |
| #include <LibDevTools/Actors/RootActor.h>
 | |
| #include <LibDevTools/Actors/TabActor.h>
 | |
| #include <LibDevTools/DevToolsDelegate.h>
 | |
| #include <LibDevTools/DevToolsServer.h>
 | |
| 
 | |
| namespace DevTools {
 | |
| 
 | |
| // https://firefox-source-docs.mozilla.org/devtools/backend/protocol.html#the-root-actor
 | |
| NonnullRefPtr<RootActor> RootActor::create(DevToolsServer& devtools, String name)
 | |
| {
 | |
|     auto actor = adopt_ref(*new RootActor(devtools, move(name)));
 | |
| 
 | |
|     JsonObject traits;
 | |
|     traits.set("sources"sv, false);
 | |
|     traits.set("highlightable"sv, true);
 | |
|     traits.set("customHighlighters"sv, true);
 | |
|     traits.set("networkMonitor"sv, false);
 | |
| 
 | |
|     JsonObject message;
 | |
|     message.set("applicationType"sv, "browser"sv);
 | |
|     message.set("traits"sv, move(traits));
 | |
|     actor->send_message(move(message));
 | |
| 
 | |
|     return actor;
 | |
| }
 | |
| 
 | |
| RootActor::RootActor(DevToolsServer& devtools, String name)
 | |
|     : Actor(devtools, move(name))
 | |
| {
 | |
| }
 | |
| 
 | |
| RootActor::~RootActor() = default;
 | |
| 
 | |
| void RootActor::handle_message(Message const& message)
 | |
| {
 | |
|     JsonObject response;
 | |
| 
 | |
|     if (message.type == "connect") {
 | |
|         send_response(message, move(response));
 | |
|         return;
 | |
|     }
 | |
| 
 | |
|     if (message.type == "getRoot"sv) {
 | |
|         response.set("selected"sv, 0);
 | |
| 
 | |
|         for (auto const& actor : devtools().actor_registry()) {
 | |
|             if (is<DeviceActor>(*actor.value))
 | |
|                 response.set("deviceActor"sv, actor.key);
 | |
|             else if (is<PreferenceActor>(*actor.value))
 | |
|                 response.set("preferenceActor"sv, actor.key);
 | |
|         }
 | |
| 
 | |
|         send_response(message, move(response));
 | |
|         return;
 | |
|     }
 | |
| 
 | |
|     if (message.type == "getProcess"sv) {
 | |
|         auto id = get_required_parameter<u64>(message, "id"sv);
 | |
|         if (!id.has_value())
 | |
|             return;
 | |
| 
 | |
|         for (auto const& actor : devtools().actor_registry()) {
 | |
|             auto const* process_actor = as_if<ProcessActor>(*actor.value);
 | |
|             if (!process_actor)
 | |
|                 continue;
 | |
|             if (process_actor->description().id != *id)
 | |
|                 continue;
 | |
| 
 | |
|             response.set("processDescriptor"sv, process_actor->serialize_description());
 | |
|             break;
 | |
|         }
 | |
| 
 | |
|         send_response(message, move(response));
 | |
|         return;
 | |
|     }
 | |
| 
 | |
|     if (message.type == "getTab"sv) {
 | |
|         auto browser_id = get_required_parameter<u64>(message, "browserId"sv);
 | |
|         if (!browser_id.has_value())
 | |
|             return;
 | |
| 
 | |
|         for (auto const& actor : devtools().actor_registry()) {
 | |
|             auto const* tab_actor = as_if<TabActor>(*actor.value);
 | |
|             if (!tab_actor)
 | |
|                 continue;
 | |
|             if (tab_actor->description().id != *browser_id)
 | |
|                 continue;
 | |
| 
 | |
|             response.set("tab"sv, tab_actor->serialize_description());
 | |
|             break;
 | |
|         }
 | |
| 
 | |
|         send_response(message, move(response));
 | |
|         return;
 | |
|     }
 | |
| 
 | |
|     if (message.type == "listAddons"sv) {
 | |
|         response.set("addons"sv, JsonArray {});
 | |
|         send_response(message, move(response));
 | |
|         return;
 | |
|     }
 | |
| 
 | |
|     if (message.type == "listProcesses"sv) {
 | |
|         JsonArray processes;
 | |
| 
 | |
|         for (auto const& actor : devtools().actor_registry()) {
 | |
|             if (auto const* process_actor = as_if<ProcessActor>(*actor.value))
 | |
|                 processes.must_append(process_actor->serialize_description());
 | |
|         }
 | |
| 
 | |
|         response.set("processes"sv, move(processes));
 | |
|         send_response(message, move(response));
 | |
|         return;
 | |
|     }
 | |
| 
 | |
|     if (message.type == "listServiceWorkerRegistrations"sv) {
 | |
|         response.set("registrations"sv, JsonArray {});
 | |
|         send_response(message, move(response));
 | |
|         return;
 | |
|     }
 | |
| 
 | |
|     if (message.type == "listTabs"sv) {
 | |
|         m_has_sent_tab_list_changed_since_last_list_tabs_request = false;
 | |
| 
 | |
|         JsonArray tabs;
 | |
| 
 | |
|         for (auto& tab_description : devtools().delegate().tab_list()) {
 | |
|             auto& actor = devtools().register_actor<TabActor>(move(tab_description));
 | |
|             tabs.must_append(actor.serialize_description());
 | |
|         }
 | |
| 
 | |
|         response.set("tabs"sv, move(tabs));
 | |
|         send_response(message, move(response));
 | |
|         return;
 | |
|     }
 | |
| 
 | |
|     if (message.type == "listWorkers"sv) {
 | |
|         response.set("workers"sv, JsonArray {});
 | |
|         send_response(message, move(response));
 | |
|         return;
 | |
|     }
 | |
| 
 | |
|     send_unrecognized_packet_type_error(message);
 | |
| }
 | |
| 
 | |
| void RootActor::send_tab_list_changed_message()
 | |
| {
 | |
|     if (m_has_sent_tab_list_changed_since_last_list_tabs_request)
 | |
|         return;
 | |
| 
 | |
|     JsonObject message;
 | |
|     message.set("type"sv, "tabListChanged"sv);
 | |
|     send_message(move(message));
 | |
| 
 | |
|     m_has_sent_tab_list_changed_since_last_list_tabs_request = true;
 | |
| }
 | |
| 
 | |
| }
 |