mirror of
				https://github.com/RGBCube/serenity
				synced 2025-10-25 07:52:32 +00:00 
			
		
		
		
	 bdd3a16b16
			
		
	
	
		bdd3a16b16
		
	
	
	
	
		
			
			Making the body GC-allocated allows us to avoid using `JS::Handle` for `m_stream` in its members.
		
			
				
	
	
		
			654 lines
		
	
	
	
		
			28 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			654 lines
		
	
	
	
		
			28 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| /*
 | ||
|  * Copyright (c) 2022-2023, Linus Groh <linusg@serenityos.org>
 | ||
|  *
 | ||
|  * SPDX-License-Identifier: BSD-2-Clause
 | ||
|  */
 | ||
| 
 | ||
| #include <LibJS/Runtime/Completion.h>
 | ||
| #include <LibWeb/Bindings/Intrinsics.h>
 | ||
| #include <LibWeb/Bindings/RequestPrototype.h>
 | ||
| #include <LibWeb/DOM/AbortSignal.h>
 | ||
| #include <LibWeb/Fetch/Enums.h>
 | ||
| #include <LibWeb/Fetch/Headers.h>
 | ||
| #include <LibWeb/Fetch/Infrastructure/HTTP/Bodies.h>
 | ||
| #include <LibWeb/Fetch/Infrastructure/HTTP/Headers.h>
 | ||
| #include <LibWeb/Fetch/Infrastructure/HTTP/Methods.h>
 | ||
| #include <LibWeb/Fetch/Infrastructure/HTTP/Requests.h>
 | ||
| #include <LibWeb/Fetch/Request.h>
 | ||
| #include <LibWeb/HTML/Scripting/Environments.h>
 | ||
| #include <LibWeb/ReferrerPolicy/ReferrerPolicy.h>
 | ||
| #include <LibWeb/URL/URL.h>
 | ||
| 
 | ||
| namespace Web::Fetch {
 | ||
| 
 | ||
| Request::Request(JS::Realm& realm, JS::NonnullGCPtr<Infrastructure::Request> request)
 | ||
|     : PlatformObject(realm)
 | ||
|     , m_request(request)
 | ||
| {
 | ||
| }
 | ||
| 
 | ||
| Request::~Request() = default;
 | ||
| 
 | ||
| void Request::initialize(JS::Realm& realm)
 | ||
| {
 | ||
|     Base::initialize(realm);
 | ||
|     set_prototype(&Bindings::ensure_web_prototype<Bindings::RequestPrototype>(realm, "Request"));
 | ||
| }
 | ||
| 
 | ||
| void Request::visit_edges(Cell::Visitor& visitor)
 | ||
| {
 | ||
|     Base::visit_edges(visitor);
 | ||
|     visitor.visit(m_request);
 | ||
|     visitor.visit(m_headers);
 | ||
|     visitor.visit(m_signal);
 | ||
| }
 | ||
| 
 | ||
| // https://fetch.spec.whatwg.org/#concept-body-mime-type
 | ||
| // https://fetch.spec.whatwg.org/#ref-for-concept-body-mime-type%E2%91%A0
 | ||
| ErrorOr<Optional<MimeSniff::MimeType>> Request::mime_type_impl() const
 | ||
| {
 | ||
|     // Objects including the Body interface mixin need to define an associated MIME type algorithm which takes no arguments and returns failure or a MIME type.
 | ||
|     // A Request object’s MIME type is to return the result of extracting a MIME type from its request’s header list.
 | ||
|     return m_request->header_list()->extract_mime_type();
 | ||
| }
 | ||
| 
 | ||
| // https://fetch.spec.whatwg.org/#concept-body-body
 | ||
| // https://fetch.spec.whatwg.org/#ref-for-concept-body-body%E2%91%A7
 | ||
| JS::GCPtr<Infrastructure::Body const> Request::body_impl() const
 | ||
| {
 | ||
|     // Objects including the Body interface mixin have an associated body (null or a body).
 | ||
|     // A Request object’s body is its request’s body.
 | ||
|     return m_request->body().visit(
 | ||
|         [](JS::NonnullGCPtr<Infrastructure::Body> const& b) -> JS::GCPtr<Infrastructure::Body const> { return b; },
 | ||
|         [](Empty) -> JS::GCPtr<Infrastructure::Body const> { return nullptr; },
 | ||
|         // A byte sequence will be safely extracted into a body early on in fetch.
 | ||
|         [](ByteBuffer const&) -> JS::GCPtr<Infrastructure::Body const> { VERIFY_NOT_REACHED(); });
 | ||
| }
 | ||
| 
 | ||
| // https://fetch.spec.whatwg.org/#concept-body-body
 | ||
| // https://fetch.spec.whatwg.org/#ref-for-concept-body-body%E2%91%A7
 | ||
| JS::GCPtr<Infrastructure::Body> Request::body_impl()
 | ||
| {
 | ||
|     // Objects including the Body interface mixin have an associated body (null or a body).
 | ||
|     // A Request object’s body is its request’s body.
 | ||
|     return m_request->body().visit(
 | ||
|         [](JS::NonnullGCPtr<Infrastructure::Body>& b) -> JS::GCPtr<Infrastructure::Body> { return b; },
 | ||
|         [](Empty) -> JS::GCPtr<Infrastructure::Body> { return {}; },
 | ||
|         // A byte sequence will be safely extracted into a body early on in fetch.
 | ||
|         [](ByteBuffer&) -> JS::GCPtr<Infrastructure::Body> { VERIFY_NOT_REACHED(); });
 | ||
| }
 | ||
| 
 | ||
| // https://fetch.spec.whatwg.org/#request-create
 | ||
| JS::NonnullGCPtr<Request> Request::create(JS::Realm& realm, JS::NonnullGCPtr<Infrastructure::Request> request, Headers::Guard guard)
 | ||
| {
 | ||
|     // 1. Let requestObject be a new Request object with realm.
 | ||
|     // 2. Set requestObject’s request to request.
 | ||
|     auto request_object = realm.heap().allocate<Request>(realm, realm, request);
 | ||
| 
 | ||
|     // 3. Set requestObject’s headers to a new Headers object with realm, whose headers list is request’s headers list and guard is guard.
 | ||
|     request_object->m_headers = realm.heap().allocate<Headers>(realm, realm, request->header_list());
 | ||
|     request_object->m_headers->set_guard(guard);
 | ||
| 
 | ||
|     // 4. Set requestObject’s signal to a new AbortSignal object with realm.
 | ||
|     request_object->m_signal = realm.heap().allocate<DOM::AbortSignal>(realm, realm);
 | ||
| 
 | ||
|     // 5. Return requestObject.
 | ||
|     return request_object;
 | ||
| }
 | ||
| 
 | ||
| // https://fetch.spec.whatwg.org/#dom-request
 | ||
| WebIDL::ExceptionOr<JS::NonnullGCPtr<Request>> Request::construct_impl(JS::Realm& realm, RequestInfo const& input, RequestInit const& init)
 | ||
| {
 | ||
|     auto& vm = realm.vm();
 | ||
| 
 | ||
|     // Referred to as 'this' in the spec.
 | ||
|     auto request_object = realm.heap().allocate<Request>(realm, realm, Infrastructure::Request::create(vm));
 | ||
| 
 | ||
|     // 1. Let request be null.
 | ||
|     JS::GCPtr<Infrastructure::Request> input_request;
 | ||
| 
 | ||
|     // 2. Let fallbackMode be null.
 | ||
|     Optional<Infrastructure::Request::Mode> fallback_mode;
 | ||
| 
 | ||
|     // 3. Let baseURL be this’s relevant settings object’s API base URL.
 | ||
|     auto base_url = HTML::relevant_settings_object(*request_object).api_base_url();
 | ||
| 
 | ||
|     // 4. Let signal be null.
 | ||
|     DOM::AbortSignal* input_signal = nullptr;
 | ||
| 
 | ||
|     // 5. If input is a string, then:
 | ||
|     if (input.has<String>()) {
 | ||
|         // 1. Let parsedURL be the result of parsing input with baseURL.
 | ||
|         auto parsed_url = URL::parse(input.get<String>(), base_url);
 | ||
| 
 | ||
|         // 2. If parsedURL is failure, then throw a TypeError.
 | ||
|         if (!parsed_url.is_valid())
 | ||
|             return WebIDL::SimpleException { WebIDL::SimpleExceptionType::TypeError, "Input URL is not valid"sv };
 | ||
| 
 | ||
|         // 3. If parsedURL includes credentials, then throw a TypeError.
 | ||
|         if (parsed_url.includes_credentials())
 | ||
|             return WebIDL::SimpleException { WebIDL::SimpleExceptionType::TypeError, "Input URL must not include credentials"sv };
 | ||
| 
 | ||
|         // 4. Set request to a new request whose URL is parsedURL.
 | ||
|         input_request = Infrastructure::Request::create(vm);
 | ||
|         input_request->set_url(move(parsed_url));
 | ||
| 
 | ||
|         // 5. Set fallbackMode to "cors".
 | ||
|         fallback_mode = Infrastructure::Request::Mode::CORS;
 | ||
|     }
 | ||
|     // 6. Otherwise:
 | ||
|     else {
 | ||
|         // 1. Assert: input is a Request object.
 | ||
|         VERIFY(input.has<JS::Handle<Request>>());
 | ||
| 
 | ||
|         // 2. Set request to input’s request.
 | ||
|         input_request = input.get<JS::Handle<Request>>()->request();
 | ||
| 
 | ||
|         // 3. Set signal to input’s signal.
 | ||
|         input_signal = input.get<JS::Handle<Request>>()->signal();
 | ||
|     }
 | ||
| 
 | ||
|     // 7. Let origin be this’s relevant settings object’s origin.
 | ||
|     auto const& origin = HTML::relevant_settings_object(*request_object).origin();
 | ||
| 
 | ||
|     // 8. Let window be "client".
 | ||
|     auto window = Infrastructure::Request::WindowType { Infrastructure::Request::Window::Client };
 | ||
| 
 | ||
|     // 9. If request’s window is an environment settings object and its origin is same origin with origin, then set window to request’s window.
 | ||
|     if (input_request->window().has<JS::GCPtr<HTML::EnvironmentSettingsObject>>()) {
 | ||
|         auto eso = input_request->window().get<JS::GCPtr<HTML::EnvironmentSettingsObject>>();
 | ||
|         if (eso->origin().is_same_origin(origin))
 | ||
|             window = input_request->window();
 | ||
|     }
 | ||
| 
 | ||
|     // 10. If init["window"] exists and is non-null, then throw a TypeError.
 | ||
|     if (init.window.has_value() && !init.window->is_null())
 | ||
|         return WebIDL::SimpleException { WebIDL::SimpleExceptionType::TypeError, "The 'window' property must be omitted or null"sv };
 | ||
| 
 | ||
|     // 11. If init["window"] exists, then set window to "no-window".
 | ||
|     if (init.window.has_value())
 | ||
|         window = Infrastructure::Request::Window::NoWindow;
 | ||
| 
 | ||
|     // 12. Set request to a new request with the following properties:
 | ||
|     // NOTE: This is done at the beginning as the 'this' value Request object
 | ||
|     //       cannot exist with a null Infrastructure::Request.
 | ||
|     auto request = request_object->request();
 | ||
| 
 | ||
|     // URL
 | ||
|     //     request’s URL.
 | ||
|     request->set_url(input_request->url());
 | ||
| 
 | ||
|     // method
 | ||
|     //     request’s method.
 | ||
|     request->set_method(TRY_OR_THROW_OOM(vm, ByteBuffer::copy(input_request->method())));
 | ||
| 
 | ||
|     // header list
 | ||
|     //     A copy of request’s header list.
 | ||
|     auto header_list_copy = Infrastructure::HeaderList::create(vm);
 | ||
|     for (auto& header : *input_request->header_list())
 | ||
|         TRY_OR_THROW_OOM(vm, header_list_copy->append(header));
 | ||
|     request->set_header_list(header_list_copy);
 | ||
| 
 | ||
|     // unsafe-request flag
 | ||
|     //     Set.
 | ||
|     request->set_unsafe_request(true);
 | ||
| 
 | ||
|     // client
 | ||
|     //     This’s relevant settings object.
 | ||
|     request->set_client(&HTML::relevant_settings_object(*request_object));
 | ||
| 
 | ||
|     // window
 | ||
|     //     window.
 | ||
|     request->set_window(window);
 | ||
| 
 | ||
|     // priority
 | ||
|     //     request’s priority.
 | ||
|     request->set_priority(input_request->priority());
 | ||
| 
 | ||
|     // origin
 | ||
|     //     request’s origin. The propagation of the origin is only significant for navigation requests being handled by a service worker. In this scenario a request can have an origin that is different from the current client.
 | ||
|     request->set_origin(input_request->origin());
 | ||
| 
 | ||
|     // referrer
 | ||
|     //     request’s referrer.
 | ||
|     request->set_referrer(input_request->referrer());
 | ||
| 
 | ||
|     // referrer policy
 | ||
|     //     request’s referrer policy.
 | ||
|     request->set_referrer_policy(input_request->referrer_policy());
 | ||
| 
 | ||
|     // mode
 | ||
|     //     request’s mode.
 | ||
|     request->set_mode(input_request->mode());
 | ||
| 
 | ||
|     // credentials mode
 | ||
|     //     request’s credentials mode.
 | ||
|     request->set_credentials_mode(input_request->credentials_mode());
 | ||
| 
 | ||
|     // cache mode
 | ||
|     //     request’s cache mode.
 | ||
|     request->set_cache_mode(input_request->cache_mode());
 | ||
| 
 | ||
|     // redirect mode
 | ||
|     //     request’s redirect mode.
 | ||
|     request->set_redirect_mode(input_request->redirect_mode());
 | ||
| 
 | ||
|     // integrity metadata
 | ||
|     //     request’s integrity metadata.
 | ||
|     request->set_integrity_metadata(input_request->integrity_metadata());
 | ||
| 
 | ||
|     // keepalive
 | ||
|     //     request’s keepalive.
 | ||
|     request->set_keepalive(input_request->keepalive());
 | ||
| 
 | ||
|     // reload-navigation flag
 | ||
|     //     request’s reload-navigation flag.
 | ||
|     request->set_reload_navigation(input_request->reload_navigation());
 | ||
| 
 | ||
|     // history-navigation flag
 | ||
|     //     request’s history-navigation flag.
 | ||
|     request->set_history_navigation(input_request->history_navigation());
 | ||
| 
 | ||
|     // URL list
 | ||
|     //     A clone of request’s URL list.
 | ||
|     request->set_url_list(input_request->url_list());
 | ||
| 
 | ||
|     // initiator type
 | ||
|     //     "fetch".
 | ||
|     request->set_initiator_type(Infrastructure::Request::InitiatorType::Fetch);
 | ||
| 
 | ||
|     // 13. If init is not empty, then:
 | ||
|     if (!init.is_empty()) {
 | ||
|         // 1. If request’s mode is "navigate", then set it to "same-origin".
 | ||
|         if (request->mode() == Infrastructure::Request::Mode::Navigate)
 | ||
|             request->set_mode(Infrastructure::Request::Mode::SameOrigin);
 | ||
| 
 | ||
|         // 2. Unset request’s reload-navigation flag.
 | ||
|         request->set_reload_navigation(false);
 | ||
| 
 | ||
|         // 3. Unset request’s history-navigation flag.
 | ||
|         request->set_history_navigation(false);
 | ||
| 
 | ||
|         // 4. Set request’s origin to "client".
 | ||
|         request->set_origin(Infrastructure::Request::Origin::Client);
 | ||
| 
 | ||
|         // 5. Set request’s referrer to "client".
 | ||
|         request->set_referrer(Infrastructure::Request::Referrer::Client);
 | ||
| 
 | ||
|         // 6. Set request’s referrer policy to the empty string.
 | ||
|         request->set_referrer_policy({});
 | ||
| 
 | ||
|         // 7. Set request’s URL to request’s current URL.
 | ||
|         request->set_url(request->current_url());
 | ||
| 
 | ||
|         // 8. Set request’s URL list to « request’s URL ».
 | ||
|         // NOTE: This is done implicitly by assigning the initial URL above.
 | ||
|     }
 | ||
| 
 | ||
|     // 14. If init["referrer"] exists, then:
 | ||
|     if (init.referrer.has_value()) {
 | ||
|         // 1. Let referrer be init["referrer"].
 | ||
|         auto const& referrer = *init.referrer;
 | ||
| 
 | ||
|         // 2. If referrer is the empty string, then set request’s referrer to "no-referrer".
 | ||
|         if (referrer.is_empty()) {
 | ||
|             request->set_referrer(Infrastructure::Request::Referrer::NoReferrer);
 | ||
|         }
 | ||
|         // 3. Otherwise:
 | ||
|         else {
 | ||
|             // 1. Let parsedReferrer be the result of parsing referrer with baseURL.
 | ||
|             auto parsed_referrer = URL::parse(referrer, base_url);
 | ||
| 
 | ||
|             // 2. If parsedReferrer is failure, then throw a TypeError.
 | ||
|             if (!parsed_referrer.is_valid())
 | ||
|                 return WebIDL::SimpleException { WebIDL::SimpleExceptionType::TypeError, "Referrer must be a valid URL"sv };
 | ||
| 
 | ||
|             // 3. If one of the following is true
 | ||
|             // - parsedReferrer’s scheme is "about" and path is the string "client"
 | ||
|             // - parsedReferrer’s origin is not same origin with origin
 | ||
|             // then set request’s referrer to "client".
 | ||
|             // FIXME: Actually use the given origin once we have https://url.spec.whatwg.org/#concept-url-origin.
 | ||
|             if ((parsed_referrer.scheme() == "about"sv && parsed_referrer.serialize_path() == "client"sv) || !HTML::Origin().is_same_origin(origin)) {
 | ||
|                 request->set_referrer(Infrastructure::Request::Referrer::Client);
 | ||
|             }
 | ||
|             // 4. Otherwise, set request’s referrer to parsedReferrer.
 | ||
|             else {
 | ||
|                 request->set_referrer(move(parsed_referrer));
 | ||
|             }
 | ||
|         }
 | ||
|     }
 | ||
| 
 | ||
|     // 15. If init["referrerPolicy"] exists, then set request’s referrer policy to it.
 | ||
|     if (init.referrer_policy.has_value())
 | ||
|         request->set_referrer_policy(from_bindings_enum(*init.referrer_policy));
 | ||
| 
 | ||
|     // 16. Let mode be init["mode"] if it exists, and fallbackMode otherwise.
 | ||
|     auto mode = init.mode.has_value()
 | ||
|         ? from_bindings_enum(*init.mode)
 | ||
|         : fallback_mode;
 | ||
| 
 | ||
|     // 17. If mode is "navigate", then throw a TypeError.
 | ||
|     if (mode == Infrastructure::Request::Mode::Navigate)
 | ||
|         return WebIDL::SimpleException { WebIDL::SimpleExceptionType::TypeError, "Mode must not be 'navigate"sv };
 | ||
| 
 | ||
|     // 18. If mode is non-null, set request’s mode to mode.
 | ||
|     if (mode.has_value())
 | ||
|         request->set_mode(*mode);
 | ||
| 
 | ||
|     // 19. If init["credentials"] exists, then set request’s credentials mode to it.
 | ||
|     if (init.credentials.has_value())
 | ||
|         request->set_credentials_mode(from_bindings_enum(*init.credentials));
 | ||
| 
 | ||
|     // 20. If init["cache"] exists, then set request’s cache mode to it.
 | ||
|     if (init.cache.has_value())
 | ||
|         request->set_cache_mode(from_bindings_enum(*init.cache));
 | ||
| 
 | ||
|     // 21. If request’s cache mode is "only-if-cached" and request’s mode is not "same-origin", then throw a TypeError.
 | ||
|     if (request->cache_mode() == Infrastructure::Request::CacheMode::OnlyIfCached && request->mode() != Infrastructure::Request::Mode::SameOrigin)
 | ||
|         return WebIDL::SimpleException { WebIDL::SimpleExceptionType::TypeError, "Mode must be 'same-origin' when cache mode is 'only-if-cached'"sv };
 | ||
| 
 | ||
|     // 22. If init["redirect"] exists, then set request’s redirect mode to it.
 | ||
|     if (init.redirect.has_value())
 | ||
|         request->set_redirect_mode(from_bindings_enum(*init.redirect));
 | ||
| 
 | ||
|     // 23. If init["integrity"] exists, then set request’s integrity metadata to it.
 | ||
|     if (init.integrity.has_value())
 | ||
|         request->set_integrity_metadata(*init.integrity);
 | ||
| 
 | ||
|     // 24. If init["keepalive"] exists, then set request’s keepalive to it.
 | ||
|     if (init.keepalive.has_value())
 | ||
|         request->set_keepalive(*init.keepalive);
 | ||
| 
 | ||
|     // 25. If init["method"] exists, then:
 | ||
|     if (init.method.has_value()) {
 | ||
|         // 1. Let method be init["method"].
 | ||
|         auto method = *init.method;
 | ||
| 
 | ||
|         // 2. If method is not a method or method is a forbidden method, then throw a TypeError.
 | ||
|         if (!Infrastructure::is_method(method.bytes()))
 | ||
|             return WebIDL::SimpleException { WebIDL::SimpleExceptionType::TypeError, "Method has invalid value"sv };
 | ||
|         if (Infrastructure::is_forbidden_method(method.bytes()))
 | ||
|             return WebIDL::SimpleException { WebIDL::SimpleExceptionType::TypeError, "Method must not be one of CONNECT, TRACE, or TRACK"sv };
 | ||
| 
 | ||
|         // 3. Normalize method.
 | ||
|         method = TRY_OR_THROW_OOM(vm, String::from_utf8(TRY_OR_THROW_OOM(vm, Infrastructure::normalize_method(method.bytes()))));
 | ||
| 
 | ||
|         // 4. Set request’s method to method.
 | ||
|         request->set_method(MUST(ByteBuffer::copy(method.bytes())));
 | ||
|     }
 | ||
| 
 | ||
|     // 26. If init["signal"] exists, then set signal to it.
 | ||
|     if (init.signal.has_value())
 | ||
|         input_signal = *init.signal;
 | ||
| 
 | ||
|     // 27. Set this’s request to request.
 | ||
|     // NOTE: This is done at the beginning as the 'this' value Request object
 | ||
|     //       cannot exist with a null Infrastructure::Request.
 | ||
| 
 | ||
|     // 28. Set this’s signal to a new AbortSignal object with this’s relevant Realm.
 | ||
|     auto& this_relevant_realm = HTML::relevant_realm(*request_object);
 | ||
|     request_object->m_signal = realm.heap().allocate<DOM::AbortSignal>(this_relevant_realm, this_relevant_realm);
 | ||
| 
 | ||
|     // 29. If signal is not null, then make this’s signal follow signal.
 | ||
|     if (input_signal != nullptr)
 | ||
|         request_object->m_signal->follow(*input_signal);
 | ||
| 
 | ||
|     // 30. Set this’s headers to a new Headers object with this’s relevant Realm, whose header list is request’s header list and guard is "request".
 | ||
|     request_object->m_headers = realm.heap().allocate<Headers>(realm, realm, request->header_list());
 | ||
|     request_object->m_headers->set_guard(Headers::Guard::Request);
 | ||
| 
 | ||
|     // 31. If this’s request’s mode is "no-cors", then:
 | ||
|     if (request_object->request()->mode() == Infrastructure::Request::Mode::NoCORS) {
 | ||
|         // 1. If this’s request’s method is not a CORS-safelisted method, then throw a TypeError.
 | ||
|         if (!Infrastructure::is_cors_safelisted_method(request_object->request()->method()))
 | ||
|             return WebIDL::SimpleException { WebIDL::SimpleExceptionType::TypeError, "Method must be one of GET, HEAD, or POST"sv };
 | ||
| 
 | ||
|         // 2. Set this’s headers’s guard to "request-no-cors".
 | ||
|         request_object->headers()->set_guard(Headers::Guard::RequestNoCORS);
 | ||
|     }
 | ||
| 
 | ||
|     // 32. If init is not empty, then:
 | ||
|     if (!init.is_empty()) {
 | ||
|         // 1. Let headers be a copy of this’s headers and its associated header list.
 | ||
|         auto headers = Variant<HeadersInit, JS::NonnullGCPtr<Infrastructure::HeaderList>> { request_object->headers()->header_list() };
 | ||
| 
 | ||
|         // 2. If init["headers"] exists, then set headers to init["headers"].
 | ||
|         if (init.headers.has_value())
 | ||
|             headers = *init.headers;
 | ||
| 
 | ||
|         // 3. Empty this’s headers’s header list.
 | ||
|         request_object->headers()->header_list()->clear();
 | ||
| 
 | ||
|         // 4. If headers is a Headers object, then for each header of its header list, append header to this’s headers.
 | ||
|         if (auto* header_list = headers.get_pointer<JS::NonnullGCPtr<Infrastructure::HeaderList>>()) {
 | ||
|             for (auto& header : *header_list->ptr())
 | ||
|                 TRY(request_object->headers()->append(TRY_OR_THROW_OOM(vm, Infrastructure::Header::from_string_pair(header.name, header.value))));
 | ||
|         }
 | ||
|         // 5. Otherwise, fill this’s headers with headers.
 | ||
|         else {
 | ||
|             TRY(request_object->headers()->fill(headers.get<HeadersInit>()));
 | ||
|         }
 | ||
|     }
 | ||
| 
 | ||
|     // 33. Let inputBody be input’s request’s body if input is a Request object; otherwise null.
 | ||
|     Optional<Infrastructure::Request::BodyType const&> input_body;
 | ||
|     if (input.has<JS::Handle<Request>>())
 | ||
|         input_body = input.get<JS::Handle<Request>>()->request()->body();
 | ||
| 
 | ||
|     // 34. If either init["body"] exists and is non-null or inputBody is non-null, and request’s method is `GET` or `HEAD`, then throw a TypeError.
 | ||
|     if (((init.body.has_value() && (*init.body).has_value()) || (input_body.has_value() && !input_body.value().has<Empty>())) && StringView { request->method() }.is_one_of("GET"sv, "HEAD"sv))
 | ||
|         return WebIDL::SimpleException { WebIDL::SimpleExceptionType::TypeError, "Method must not be GET or HEAD when body is provided"sv };
 | ||
| 
 | ||
|     // 35. Let initBody be null.
 | ||
|     JS::GCPtr<Infrastructure::Body> init_body;
 | ||
| 
 | ||
|     // 36. If init["body"] exists and is non-null, then:
 | ||
|     if (init.body.has_value() && (*init.body).has_value()) {
 | ||
|         // 1. Let bodyWithType be the result of extracting init["body"], with keepalive set to request’s keepalive.
 | ||
|         auto body_with_type = TRY(extract_body(realm, (*init.body).value(), request->keepalive()));
 | ||
| 
 | ||
|         // 2. Set initBody to bodyWithType’s body.
 | ||
|         init_body = body_with_type.body;
 | ||
| 
 | ||
|         // 3. Let type be bodyWithType’s type.
 | ||
|         auto const& type = body_with_type.type;
 | ||
| 
 | ||
|         // 4. If type is non-null and this’s headers’s header list does not contain `Content-Type`, then append (`Content-Type`, type) to this’s headers.
 | ||
|         if (type.has_value() && !request_object->headers()->header_list()->contains("Content-Type"sv.bytes()))
 | ||
|             TRY(request_object->headers()->append(TRY_OR_THROW_OOM(vm, Infrastructure::Header::from_string_pair("Content-Type"sv, type->span()))));
 | ||
|     }
 | ||
| 
 | ||
|     // 37. Let inputOrInitBody be initBody if it is non-null; otherwise inputBody.
 | ||
|     Optional<Infrastructure::Request::BodyType> input_or_init_body = init_body
 | ||
|         ? Infrastructure::Request::BodyType { *init_body }
 | ||
|         : input_body;
 | ||
| 
 | ||
|     // 38. If inputOrInitBody is non-null and inputOrInitBody’s source is null, then:
 | ||
|     // FIXME: The spec doesn't check if inputOrInitBody is a body before accessing source.
 | ||
|     if (input_or_init_body.has_value() && input_or_init_body->has<JS::NonnullGCPtr<Infrastructure::Body>>() && input_or_init_body->get<JS::NonnullGCPtr<Infrastructure::Body>>()->source().has<Empty>()) {
 | ||
|         // 1. If initBody is non-null and init["duplex"] does not exist, then throw a TypeError.
 | ||
|         if (init_body && !init.duplex.has_value())
 | ||
|             return WebIDL::SimpleException { WebIDL::SimpleExceptionType::TypeError, "Body without source requires 'duplex' value to be set"sv };
 | ||
| 
 | ||
|         // 2. If this’s request’s mode is neither "same-origin" nor "cors", then throw a TypeError.
 | ||
|         if (request_object->request()->mode() != Infrastructure::Request::Mode::SameOrigin && request_object->request()->mode() != Infrastructure::Request::Mode::CORS)
 | ||
|             return WebIDL::SimpleException { WebIDL::SimpleExceptionType::TypeError, "Request mode must be 'same-origin' or 'cors'"sv };
 | ||
| 
 | ||
|         // 3. Set this’s request’s use-CORS-preflight flag.
 | ||
|         request_object->request()->set_use_cors_preflight(true);
 | ||
|     }
 | ||
| 
 | ||
|     // 39. Let finalBody be inputOrInitBody.
 | ||
|     auto const& final_body = input_or_init_body;
 | ||
| 
 | ||
|     // 40. If initBody is null and inputBody is non-null, then:
 | ||
|     if (!init_body && input_body.has_value()) {
 | ||
|         // 2. If input is unusable, then throw a TypeError.
 | ||
|         if (input.has<JS::Handle<Request>>() && input.get<JS::Handle<Request>>()->is_unusable())
 | ||
|             return WebIDL::SimpleException { WebIDL::SimpleExceptionType::TypeError, "Request is unusable"sv };
 | ||
| 
 | ||
|         // FIXME: 2. Set finalBody to the result of creating a proxy for inputBody.
 | ||
|     }
 | ||
| 
 | ||
|     // 41. Set this’s request’s body to finalBody.
 | ||
|     if (final_body.has_value())
 | ||
|         request_object->request()->set_body(*final_body);
 | ||
| 
 | ||
|     return JS::NonnullGCPtr { *request_object };
 | ||
| }
 | ||
| 
 | ||
| // https://fetch.spec.whatwg.org/#dom-request-method
 | ||
| WebIDL::ExceptionOr<String> Request::method() const
 | ||
| {
 | ||
|     auto& vm = this->vm();
 | ||
| 
 | ||
|     // The method getter steps are to return this’s request’s method.
 | ||
|     return TRY_OR_THROW_OOM(vm, String::from_utf8(m_request->method()));
 | ||
| }
 | ||
| 
 | ||
| // https://fetch.spec.whatwg.org/#dom-request-url
 | ||
| WebIDL::ExceptionOr<String> Request::url() const
 | ||
| {
 | ||
|     auto& vm = this->vm();
 | ||
| 
 | ||
|     // The url getter steps are to return this’s request’s URL, serialized.
 | ||
|     return TRY_OR_THROW_OOM(vm, String::from_deprecated_string(m_request->url().serialize()));
 | ||
| }
 | ||
| 
 | ||
| // https://fetch.spec.whatwg.org/#dom-request-headers
 | ||
| JS::NonnullGCPtr<Headers> Request::headers() const
 | ||
| {
 | ||
|     // The headers getter steps are to return this’s headers.
 | ||
|     return *m_headers;
 | ||
| }
 | ||
| 
 | ||
| // https://fetch.spec.whatwg.org/#dom-request-destination
 | ||
| Bindings::RequestDestination Request::destination() const
 | ||
| {
 | ||
|     // The destination getter are to return this’s request’s destination.
 | ||
|     return to_bindings_enum(m_request->destination());
 | ||
| }
 | ||
| 
 | ||
| // https://fetch.spec.whatwg.org/#dom-request-referrer
 | ||
| WebIDL::ExceptionOr<String> Request::referrer() const
 | ||
| {
 | ||
|     auto& vm = this->vm();
 | ||
|     return m_request->referrer().visit(
 | ||
|         [&](Infrastructure::Request::Referrer const& referrer) -> WebIDL::ExceptionOr<String> {
 | ||
|             switch (referrer) {
 | ||
|             // 1. If this’s request’s referrer is "no-referrer", then return the empty string.
 | ||
|             case Infrastructure::Request::Referrer::NoReferrer:
 | ||
|                 return String {};
 | ||
|             // 2. If this’s request’s referrer is "client", then return "about:client".
 | ||
|             case Infrastructure::Request::Referrer::Client:
 | ||
|                 return "about:client"_string;
 | ||
|             default:
 | ||
|                 VERIFY_NOT_REACHED();
 | ||
|             }
 | ||
|         },
 | ||
|         [&](AK::URL const& url) -> WebIDL::ExceptionOr<String> {
 | ||
|             // 3. Return this’s request’s referrer, serialized.
 | ||
|             return TRY_OR_THROW_OOM(vm, String::from_deprecated_string(url.serialize()));
 | ||
|         });
 | ||
| }
 | ||
| 
 | ||
| // https://fetch.spec.whatwg.org/#dom-request-referrerpolicy
 | ||
| Bindings::ReferrerPolicy Request::referrer_policy() const
 | ||
| {
 | ||
|     // The referrerPolicy getter steps are to return this’s request’s referrer policy.
 | ||
|     return to_bindings_enum(m_request->referrer_policy());
 | ||
| }
 | ||
| 
 | ||
| // https://fetch.spec.whatwg.org/#dom-request-mode
 | ||
| Bindings::RequestMode Request::mode() const
 | ||
| {
 | ||
|     // The mode getter steps are to return this’s request’s mode.
 | ||
|     return to_bindings_enum(m_request->mode());
 | ||
| }
 | ||
| 
 | ||
| // https://fetch.spec.whatwg.org/#dom-request-credentials
 | ||
| Bindings::RequestCredentials Request::credentials() const
 | ||
| {
 | ||
|     // The credentials getter steps are to return this’s request’s credentials mode.
 | ||
|     return to_bindings_enum(m_request->credentials_mode());
 | ||
| }
 | ||
| 
 | ||
| // https://fetch.spec.whatwg.org/#dom-request-cache
 | ||
| Bindings::RequestCache Request::cache() const
 | ||
| {
 | ||
|     // The cache getter steps are to return this’s request’s cache mode.
 | ||
|     return to_bindings_enum(m_request->cache_mode());
 | ||
| }
 | ||
| 
 | ||
| // https://fetch.spec.whatwg.org/#dom-request-redirect
 | ||
| Bindings::RequestRedirect Request::redirect() const
 | ||
| {
 | ||
|     // The redirect getter steps are to return this’s request’s redirect mode.
 | ||
|     return to_bindings_enum(m_request->redirect_mode());
 | ||
| }
 | ||
| 
 | ||
| // https://fetch.spec.whatwg.org/#dom-request-integrity
 | ||
| String Request::integrity() const
 | ||
| {
 | ||
|     // The integrity getter steps are to return this’s request’s integrity metadata.
 | ||
|     return m_request->integrity_metadata();
 | ||
| }
 | ||
| 
 | ||
| // https://fetch.spec.whatwg.org/#dom-request-keepalive
 | ||
| bool Request::keepalive() const
 | ||
| {
 | ||
|     // The keepalive getter steps are to return this’s request’s keepalive.
 | ||
|     return m_request->keepalive();
 | ||
| }
 | ||
| 
 | ||
| // https://fetch.spec.whatwg.org/#dom-request-isreloadnavigation
 | ||
| bool Request::is_reload_navigation() const
 | ||
| {
 | ||
|     // The isReloadNavigation getter steps are to return true if this’s request’s reload-navigation flag is set; otherwise false.
 | ||
|     return m_request->reload_navigation();
 | ||
| }
 | ||
| 
 | ||
| // https://fetch.spec.whatwg.org/#dom-request-ishistorynavigation
 | ||
| bool Request::is_history_navigation() const
 | ||
| {
 | ||
|     // The isHistoryNavigation getter steps are to return true if this’s request’s history-navigation flag is set; otherwise false.
 | ||
|     return m_request->history_navigation();
 | ||
| }
 | ||
| 
 | ||
| // https://fetch.spec.whatwg.org/#dom-request-signal
 | ||
| JS::NonnullGCPtr<DOM::AbortSignal> Request::signal() const
 | ||
| {
 | ||
|     // The signal getter steps are to return this’s signal.
 | ||
|     return *m_signal;
 | ||
| }
 | ||
| 
 | ||
| // https://fetch.spec.whatwg.org/#dom-request-duplex
 | ||
| Bindings::RequestDuplex Request::duplex() const
 | ||
| {
 | ||
|     // The duplex getter steps are to return "half".
 | ||
|     return Bindings::RequestDuplex::Half;
 | ||
| }
 | ||
| 
 | ||
| // https://fetch.spec.whatwg.org/#dom-request-clone
 | ||
| WebIDL::ExceptionOr<JS::NonnullGCPtr<Request>> Request::clone() const
 | ||
| {
 | ||
|     auto& realm = this->realm();
 | ||
| 
 | ||
|     // 1. If this is unusable, then throw a TypeError.
 | ||
|     if (is_unusable())
 | ||
|         return WebIDL::SimpleException { WebIDL::SimpleExceptionType::TypeError, "Request is unusable"sv };
 | ||
| 
 | ||
|     // 2. Let clonedRequest be the result of cloning this’s request.
 | ||
|     auto cloned_request = m_request->clone(realm);
 | ||
| 
 | ||
|     // 3. Let clonedRequestObject be the result of creating a Request object, given clonedRequest, this’s headers’s guard, and this’s relevant Realm.
 | ||
|     auto cloned_request_object = Request::create(HTML::relevant_realm(*this), cloned_request, m_headers->guard());
 | ||
| 
 | ||
|     // 4. Make clonedRequestObject’s signal follow this’s signal.
 | ||
|     cloned_request_object->m_signal->follow(*m_signal);
 | ||
| 
 | ||
|     // 5. Return clonedRequestObject.
 | ||
|     return cloned_request_object;
 | ||
| }
 | ||
| 
 | ||
| }
 |