mirror of
				https://github.com/LadybirdBrowser/ladybird.git
				synced 2025-10-25 17:39:27 +00:00 
			
		
		
		
	
		
			Some checks are pending
		
		
	
	CI / macOS, arm64, Sanitizer_CI, Clang (push) Waiting to run
				
			CI / Linux, x86_64, Fuzzers_CI, Clang (push) Waiting to run
				
			CI / Linux, x86_64, Sanitizer_CI, GNU (push) Waiting to run
				
			CI / Linux, x86_64, Sanitizer_CI, Clang (push) Waiting to run
				
			Package the js repl as a binary artifact / macOS, arm64 (push) Waiting to run
				
			Package the js repl as a binary artifact / Linux, x86_64 (push) Waiting to run
				
			Run test262 and test-wasm / run_and_update_results (push) Waiting to run
				
			Lint Code / lint (push) Waiting to run
				
			Label PRs with merge conflicts / auto-labeler (push) Waiting to run
				
			Push notes / build (push) Waiting to run
				
			This change implements following behavior defined in the spec: https://html.spec.whatwg.org/multipage/web-messaging.html#examples-5 > The start() method, whether called explicitly or implicitly (by setting onmessage), starts the flow of messages: messages posted on message ports are initially paused, so that they don't get dropped on the floor before the script has had a chance to set up its handlers. Now we don't read bytes from the underlying transport socket until the message port transitions to the enabled state. This required the following places to explicitly enable the message port, because now, when it actually matters, we must do so, or otherwise sent messages will get stuck: - `onmessage` attribute setter in DedicatedWorkerGlobalScope, because it implicitly sets the onmessage handler for the worker's underlying port. - Stream API operations where the message port enabling steps were previously marked as FIXMEs.
		
			
				
	
	
		
			96 lines
		
	
	
	
		
			3.7 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			96 lines
		
	
	
	
		
			3.7 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| /*
 | |
|  * Copyright (c) 2024, Andrew Kaster <akaster@serenityos.org>
 | |
|  *
 | |
|  * SPDX-License-Identifier: BSD-2-Clause
 | |
|  */
 | |
| 
 | |
| #include <LibWeb/Bindings/DedicatedWorkerExposedInterfaces.h>
 | |
| #include <LibWeb/Bindings/DedicatedWorkerGlobalScopePrototype.h>
 | |
| #include <LibWeb/Bindings/Intrinsics.h>
 | |
| #include <LibWeb/HTML/DedicatedWorkerGlobalScope.h>
 | |
| #include <LibWeb/HTML/EventHandler.h>
 | |
| #include <LibWeb/HTML/EventNames.h>
 | |
| #include <LibWeb/HTML/MessageEvent.h>
 | |
| #include <LibWeb/HTML/MessagePort.h>
 | |
| 
 | |
| namespace Web::HTML {
 | |
| 
 | |
| GC_DEFINE_ALLOCATOR(DedicatedWorkerGlobalScope);
 | |
| 
 | |
| DedicatedWorkerGlobalScope::DedicatedWorkerGlobalScope(JS::Realm& realm, GC::Ref<Web::Page> page)
 | |
|     : WorkerGlobalScope(realm, page)
 | |
| {
 | |
|     m_legacy_platform_object_flags = LegacyPlatformObjectFlags { .has_global_interface_extended_attribute = true };
 | |
| }
 | |
| 
 | |
| DedicatedWorkerGlobalScope::~DedicatedWorkerGlobalScope() = default;
 | |
| 
 | |
| void DedicatedWorkerGlobalScope::initialize_web_interfaces_impl()
 | |
| {
 | |
|     auto& realm = this->realm();
 | |
|     add_dedicated_worker_exposed_interfaces(*this);
 | |
| 
 | |
|     DedicatedWorkerGlobalScopeGlobalMixin::initialize(realm, *this);
 | |
| 
 | |
|     Base::initialize_web_interfaces_impl();
 | |
| }
 | |
| 
 | |
| // https://html.spec.whatwg.org/multipage/workers.html#dom-dedicatedworkerglobalscope-close
 | |
| void DedicatedWorkerGlobalScope::close()
 | |
| {
 | |
|     // The close() method steps are to close a worker given this.
 | |
|     close_a_worker();
 | |
| }
 | |
| 
 | |
| void DedicatedWorkerGlobalScope::finalize()
 | |
| {
 | |
|     Base::finalize();
 | |
|     WindowOrWorkerGlobalScopeMixin::finalize();
 | |
| }
 | |
| 
 | |
| // https://html.spec.whatwg.org/multipage/workers.html#dom-dedicatedworkerglobalscope-postmessage-options
 | |
| WebIDL::ExceptionOr<void> DedicatedWorkerGlobalScope::post_message(JS::Value message, StructuredSerializeOptions const& options)
 | |
| {
 | |
|     // The postMessage(message, transfer) and postMessage(message, options) methods on DedicatedWorkerGlobalScope objects act as if,
 | |
|     // when invoked, it immediately invoked the respective postMessage(message, transfer) and postMessage(message, options)
 | |
|     // on the port, with the same arguments, and returned the same return value.
 | |
|     return m_internal_port->post_message(message, options);
 | |
| }
 | |
| 
 | |
| // https://html.spec.whatwg.org/multipage/workers.html#dom-dedicatedworkerglobalscope-postmessage
 | |
| WebIDL::ExceptionOr<void> DedicatedWorkerGlobalScope::post_message(JS::Value message, Vector<GC::Root<JS::Object>> const& transfer)
 | |
| {
 | |
|     // The postMessage(message, transfer) and postMessage(message, options) methods on DedicatedWorkerGlobalScope objects act as if,
 | |
|     // when invoked, it immediately invoked the respective postMessage(message, transfer) and postMessage(message, options)
 | |
|     // on the port, with the same arguments, and returned the same return value.
 | |
|     return m_internal_port->post_message(message, transfer);
 | |
| }
 | |
| 
 | |
| WebIDL::CallbackType* DedicatedWorkerGlobalScope::onmessage()
 | |
| {
 | |
|     return event_handler_attribute(EventNames::message);
 | |
| }
 | |
| 
 | |
| void DedicatedWorkerGlobalScope::set_onmessage(WebIDL::CallbackType* callback)
 | |
| {
 | |
|     set_event_handler_attribute(EventNames::message, callback);
 | |
| 
 | |
|     // NOTE: This onmessage attribute setter implicitly sets worker's underlying MessagePort's onmessage attribute, so this
 | |
|     //       spec behavior also applies here:
 | |
|     // https://html.spec.whatwg.org/multipage/web-messaging.html#message-ports:handler-messageeventtarget-onmessage
 | |
|     // The first time a MessagePort object's onmessage IDL attribute is set, the port's port message queue must be enabled,
 | |
|     // as if the start() method had been called.
 | |
|     m_internal_port->start();
 | |
| }
 | |
| 
 | |
| void DedicatedWorkerGlobalScope::set_onmessageerror(WebIDL::CallbackType* callback)
 | |
| {
 | |
|     set_event_handler_attribute(EventNames::messageerror, callback);
 | |
| }
 | |
| 
 | |
| WebIDL::CallbackType* DedicatedWorkerGlobalScope::onmessageerror()
 | |
| {
 | |
|     return event_handler_attribute(EventNames::messageerror);
 | |
| }
 | |
| 
 | |
| }
 |