| 
									
										
										
										
											2022-10-23 22:20:25 +01:00
										 |  |  |  | /*
 | 
					
						
							| 
									
										
										
										
											2023-03-02 23:26:35 +00:00
										 |  |  |  |  * Copyright (c) 2022-2023, Linus Groh <linusg@serenityos.org> | 
					
						
							| 
									
										
										
										
											2022-10-23 22:20:25 +01:00
										 |  |  |  |  * | 
					
						
							|  |  |  |  |  * SPDX-License-Identifier: BSD-2-Clause | 
					
						
							|  |  |  |  |  */ | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | #include <AK/Debug.h>
 | 
					
						
							| 
									
										
										
										
											2022-11-23 13:41:50 +01:00
										 |  |  |  | #include <AK/TypeCasts.h>
 | 
					
						
							| 
									
										
										
										
											2022-10-23 22:20:25 +01:00
										 |  |  |  | #include <LibJS/Runtime/PromiseCapability.h>
 | 
					
						
							|  |  |  |  | #include <LibWeb/Bindings/ExceptionOrUtils.h>
 | 
					
						
							|  |  |  |  | #include <LibWeb/DOM/AbortSignal.h>
 | 
					
						
							|  |  |  |  | #include <LibWeb/Fetch/FetchMethod.h>
 | 
					
						
							|  |  |  |  | #include <LibWeb/Fetch/Fetching/Fetching.h>
 | 
					
						
							|  |  |  |  | #include <LibWeb/Fetch/Fetching/RefCountedFlag.h>
 | 
					
						
							|  |  |  |  | #include <LibWeb/Fetch/Infrastructure/FetchAlgorithms.h>
 | 
					
						
							|  |  |  |  | #include <LibWeb/Fetch/Infrastructure/FetchController.h>
 | 
					
						
							|  |  |  |  | #include <LibWeb/Fetch/Infrastructure/HTTP/Requests.h>
 | 
					
						
							|  |  |  |  | #include <LibWeb/Fetch/Infrastructure/HTTP/Responses.h>
 | 
					
						
							|  |  |  |  | #include <LibWeb/Fetch/Request.h>
 | 
					
						
							|  |  |  |  | #include <LibWeb/Fetch/Response.h>
 | 
					
						
							| 
									
										
										
										
											2023-07-06 07:43:23 -04:00
										 |  |  |  | #include <LibWeb/HTML/Scripting/TemporaryExecutionContext.h>
 | 
					
						
							| 
									
										
										
										
											2025-04-17 15:47:53 -04:00
										 |  |  |  | #include <LibWeb/Streams/ReadableStreamOperations.h>
 | 
					
						
							| 
									
										
										
										
											2023-03-07 18:15:52 +00:00
										 |  |  |  | #include <LibWeb/WebIDL/ExceptionOr.h>
 | 
					
						
							| 
									
										
										
										
											2022-10-23 22:20:25 +01:00
										 |  |  |  | #include <LibWeb/WebIDL/Promise.h>
 | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | namespace Web::Fetch { | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | // https://fetch.spec.whatwg.org/#dom-global-fetch
 | 
					
						
							| 
									
										
										
										
											2024-11-15 04:01:23 +13:00
										 |  |  |  | GC::Ref<WebIDL::Promise> fetch(JS::VM& vm, RequestInfo const& input, RequestInit const& init) | 
					
						
							| 
									
										
										
										
											2022-10-23 22:20:25 +01:00
										 |  |  |  | { | 
					
						
							|  |  |  |  |     auto& realm = *vm.current_realm(); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     // 1. Let p be a new promise.
 | 
					
						
							|  |  |  |  |     auto promise_capability = WebIDL::create_promise(realm); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     // 2. Let requestObject be the result of invoking the initial value of Request as constructor with input and init
 | 
					
						
							|  |  |  |  |     //    as arguments. If this throws an exception, reject p with it and return p.
 | 
					
						
							|  |  |  |  |     auto exception_or_request_object = Request::construct_impl(realm, input, init); | 
					
						
							|  |  |  |  |     if (exception_or_request_object.is_exception()) { | 
					
						
							| 
									
										
										
										
											2024-11-04 14:37:27 +01:00
										 |  |  |  |         auto throw_completion = Bindings::exception_to_throw_completion(vm, exception_or_request_object.exception()); | 
					
						
							| 
									
										
										
										
											2025-04-04 18:11:45 +02:00
										 |  |  |  |         WebIDL::reject_promise(realm, promise_capability, throw_completion.value()); | 
					
						
							| 
									
										
										
										
											2024-10-25 12:38:19 -06:00
										 |  |  |  |         return promise_capability; | 
					
						
							| 
									
										
										
										
											2022-10-23 22:20:25 +01:00
										 |  |  |  |     } | 
					
						
							|  |  |  |  |     auto request_object = exception_or_request_object.release_value(); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     // 3. Let request be requestObject’s request.
 | 
					
						
							|  |  |  |  |     auto request = request_object->request(); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     // 4. If requestObject’s signal is aborted, then:
 | 
					
						
							|  |  |  |  |     if (request_object->signal()->aborted()) { | 
					
						
							|  |  |  |  |         // 1. Abort the fetch() call with p, request, null, and requestObject’s signal’s abort reason.
 | 
					
						
							| 
									
										
										
										
											2023-02-28 17:45:49 +00:00
										 |  |  |  |         abort_fetch(realm, promise_capability, request, nullptr, request_object->signal()->reason()); | 
					
						
							| 
									
										
										
										
											2022-10-23 22:20:25 +01:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |         // 2. Return p.
 | 
					
						
							| 
									
										
										
										
											2024-10-25 12:38:19 -06:00
										 |  |  |  |         return promise_capability; | 
					
						
							| 
									
										
										
										
											2022-10-23 22:20:25 +01:00
										 |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     // 5. Let globalObject be request’s client’s global object.
 | 
					
						
							|  |  |  |  |     auto& global_object = request->client()->global_object(); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     // FIXME: 6. If globalObject is a ServiceWorkerGlobalScope object, then set request’s service-workers mode to "none".
 | 
					
						
							|  |  |  |  |     (void)global_object; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     // 7. Let responseObject be null.
 | 
					
						
							| 
									
										
										
										
											2024-11-15 04:01:23 +13:00
										 |  |  |  |     GC::Ptr<Response> response_object; | 
					
						
							| 
									
										
										
										
											2022-10-23 22:20:25 +01:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |     // 8. Let relevantRealm be this’s relevant Realm.
 | 
					
						
							|  |  |  |  |     // NOTE: This assumes that the running execution context is for the fetch() function call.
 | 
					
						
							|  |  |  |  |     auto& relevant_realm = HTML::relevant_realm(*vm.running_execution_context().function); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     // 9. Let locallyAborted be false.
 | 
					
						
							|  |  |  |  |     // NOTE: This lets us reject promises with predictable timing, when the request to abort comes from the same thread
 | 
					
						
							|  |  |  |  |     //       as the call to fetch.
 | 
					
						
							|  |  |  |  |     auto locally_aborted = Fetching::RefCountedFlag::create(false); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     // 10. Let controller be null.
 | 
					
						
							| 
									
										
										
										
											2024-11-24 20:37:13 +01:00
										 |  |  |  |     auto controller_holder = Infrastructure::FetchControllerHolder::create(vm); | 
					
						
							| 
									
										
										
										
											2022-10-23 22:20:25 +01:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-11-04 21:26:11 +00:00
										 |  |  |  |     // NOTE: Step 11 is done out of order so that the controller is non-null when we capture the GCPtr by copy in the abort algorithm lambda.
 | 
					
						
							|  |  |  |  |     //       This is not observable, AFAICT.
 | 
					
						
							| 
									
										
										
										
											2022-10-23 22:20:25 +01:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |     // 12. Set controller to the result of calling fetch given request and processResponse given response being these
 | 
					
						
							|  |  |  |  |     //     steps:
 | 
					
						
							| 
									
										
										
										
											2024-11-24 20:37:13 +01:00
										 |  |  |  |     auto process_response = [locally_aborted, promise_capability, request, response_object, controller_holder, &relevant_realm](GC::Ref<Infrastructure::Response> response) mutable { | 
					
						
							| 
									
										
										
										
											2022-10-23 22:20:25 +01:00
										 |  |  |  |         // 1. If locallyAborted is true, then abort these steps.
 | 
					
						
							|  |  |  |  |         if (locally_aborted->value()) | 
					
						
							|  |  |  |  |             return; | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-07-06 07:43:23 -04:00
										 |  |  |  |         // AD-HOC: An execution context is required for Promise functions.
 | 
					
						
							| 
									
										
										
										
											2024-10-24 20:39:18 +13:00
										 |  |  |  |         HTML::TemporaryExecutionContext execution_context { relevant_realm }; | 
					
						
							| 
									
										
										
										
											2023-02-28 17:45:49 +00:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-10-23 22:20:25 +01:00
										 |  |  |  |         // 2. If response’s aborted flag is set, then:
 | 
					
						
							|  |  |  |  |         if (response->aborted()) { | 
					
						
							| 
									
										
										
										
											2024-11-24 20:37:13 +01:00
										 |  |  |  |             // 1. Let deserializedError be the result of deserialize a serialized abort reason given controller’s
 | 
					
						
							|  |  |  |  |             //    serialized abort reason and relevantRealm.
 | 
					
						
							|  |  |  |  |             auto deserialized_error = controller_holder->controller()->deserialize_a_serialized_abort_reason(relevant_realm); | 
					
						
							| 
									
										
										
										
											2022-10-23 22:20:25 +01:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |             // 2. Abort the fetch() call with p, request, responseObject, and deserializedError.
 | 
					
						
							| 
									
										
										
										
											2023-09-26 16:50:31 +02:00
										 |  |  |  |             abort_fetch(relevant_realm, promise_capability, request, response_object, deserialized_error); | 
					
						
							| 
									
										
										
										
											2022-10-23 22:20:25 +01:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |             // 3. Abort these steps.
 | 
					
						
							|  |  |  |  |             return; | 
					
						
							|  |  |  |  |         } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |         // 3. If response is a network error, then reject p with a TypeError and abort these steps.
 | 
					
						
							|  |  |  |  |         if (response->is_network_error()) { | 
					
						
							| 
									
										
										
										
											2025-04-02 20:51:45 +13:00
										 |  |  |  |             auto message = response->network_error_message().value_or("Response is a network error"_string); | 
					
						
							| 
									
										
										
										
											2023-09-06 08:18:01 -04:00
										 |  |  |  |             WebIDL::reject_promise(relevant_realm, promise_capability, JS::TypeError::create(relevant_realm, message)); | 
					
						
							| 
									
										
										
										
											2022-10-23 22:20:25 +01:00
										 |  |  |  |             return; | 
					
						
							|  |  |  |  |         } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |         // 4. Set responseObject to the result of creating a Response object, given response, "immutable", and
 | 
					
						
							|  |  |  |  |         //    relevantRealm.
 | 
					
						
							| 
									
										
										
										
											2023-09-26 16:50:31 +02:00
										 |  |  |  |         response_object = Response::create(relevant_realm, response, Headers::Guard::Immutable); | 
					
						
							| 
									
										
										
										
											2022-10-23 22:20:25 +01:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |         // 5. Resolve p with responseObject.
 | 
					
						
							| 
									
										
										
										
											2023-02-28 17:45:49 +00:00
										 |  |  |  |         WebIDL::resolve_promise(relevant_realm, promise_capability, response_object); | 
					
						
							| 
									
										
										
										
											2022-10-23 22:20:25 +01:00
										 |  |  |  |     }; | 
					
						
							| 
									
										
										
										
											2024-11-24 20:37:13 +01:00
										 |  |  |  |     controller_holder->set_controller(MUST(Fetching::fetch( | 
					
						
							| 
									
										
										
										
											2022-10-23 22:20:25 +01:00
										 |  |  |  |         realm, | 
					
						
							|  |  |  |  |         request, | 
					
						
							|  |  |  |  |         Infrastructure::FetchAlgorithms::create(vm, | 
					
						
							|  |  |  |  |             { | 
					
						
							|  |  |  |  |                 .process_request_body_chunk_length = {}, | 
					
						
							|  |  |  |  |                 .process_request_end_of_body = {}, | 
					
						
							|  |  |  |  |                 .process_early_hints_response = {}, | 
					
						
							|  |  |  |  |                 .process_response = move(process_response), | 
					
						
							|  |  |  |  |                 .process_response_end_of_body = {}, | 
					
						
							|  |  |  |  |                 .process_response_consume_body = {}, | 
					
						
							| 
									
										
										
										
											2024-11-24 20:37:13 +01:00
										 |  |  |  |             })))); | 
					
						
							| 
									
										
										
										
											2022-10-23 22:20:25 +01:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-11-04 21:26:11 +00:00
										 |  |  |  |     // 11. Add the following abort steps to requestObject’s signal:
 | 
					
						
							| 
									
										
										
										
											2025-04-10 09:04:01 -04:00
										 |  |  |  |     (void)request_object->signal()->add_abort_algorithm([locally_aborted, request, controller_holder, promise_capability, request_object, response_object, &relevant_realm] { | 
					
						
							| 
									
										
										
										
											2022-11-04 21:26:11 +00:00
										 |  |  |  |         dbgln_if(WEB_FETCH_DEBUG, "Fetch: Request object signal's abort algorithm called"); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |         // 1. Set locallyAborted to true.
 | 
					
						
							|  |  |  |  |         locally_aborted->set_value(true); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |         // 2. Assert: controller is non-null.
 | 
					
						
							| 
									
										
										
										
											2024-11-24 20:37:13 +01:00
										 |  |  |  |         VERIFY(controller_holder->controller()); | 
					
						
							| 
									
										
										
										
											2022-11-04 21:26:11 +00:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |         // 3. Abort controller with requestObject’s signal’s abort reason.
 | 
					
						
							| 
									
										
										
										
											2024-11-24 20:37:13 +01:00
										 |  |  |  |         controller_holder->controller()->abort(relevant_realm, request_object->signal()->reason()); | 
					
						
							| 
									
										
										
										
											2022-11-04 21:26:11 +00:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-04-14 00:00:50 +01:00
										 |  |  |  |         // AD-HOC: An execution context is required for Promise functions.
 | 
					
						
							| 
									
										
										
										
											2024-10-24 20:39:18 +13:00
										 |  |  |  |         HTML::TemporaryExecutionContext execution_context { relevant_realm }; | 
					
						
							| 
									
										
										
										
											2024-04-14 00:00:50 +01:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-11-04 21:26:11 +00:00
										 |  |  |  |         // 4. Abort the fetch() call with p, request, responseObject, and requestObject’s signal’s abort reason.
 | 
					
						
							| 
									
										
										
										
											2023-09-26 16:50:31 +02:00
										 |  |  |  |         abort_fetch(relevant_realm, *promise_capability, request, response_object, request_object->signal()->reason()); | 
					
						
							| 
									
										
										
										
											2022-11-04 21:26:11 +00:00
										 |  |  |  |     }); | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-10-23 22:20:25 +01:00
										 |  |  |  |     // 13. Return p.
 | 
					
						
							| 
									
										
										
										
											2024-10-25 12:38:19 -06:00
										 |  |  |  |     return promise_capability; | 
					
						
							| 
									
										
										
										
											2022-10-23 22:20:25 +01:00
										 |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | // https://fetch.spec.whatwg.org/#abort-fetch
 | 
					
						
							| 
									
										
										
										
											2024-11-15 04:01:23 +13:00
										 |  |  |  | void abort_fetch(JS::Realm& realm, WebIDL::Promise const& promise, GC::Ref<Infrastructure::Request> request, GC::Ptr<Response> response_object, JS::Value error) | 
					
						
							| 
									
										
										
										
											2022-10-23 22:20:25 +01:00
										 |  |  |  | { | 
					
						
							|  |  |  |  |     dbgln_if(WEB_FETCH_DEBUG, "Fetch: Aborting fetch with: request @ {}, error = {}", request.ptr(), error); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     // 1. Reject promise with error.
 | 
					
						
							|  |  |  |  |     // NOTE: This is a no-op if promise has already fulfilled.
 | 
					
						
							| 
									
										
										
										
											2023-02-28 17:45:49 +00:00
										 |  |  |  |     WebIDL::reject_promise(realm, promise, error); | 
					
						
							| 
									
										
										
										
											2022-10-23 22:20:25 +01:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |     // 2. If request’s body is non-null and is readable, then cancel request’s body with error.
 | 
					
						
							| 
									
										
										
										
											2024-11-15 04:01:23 +13:00
										 |  |  |  |     if (auto* body = request->body().get_pointer<GC::Ref<Infrastructure::Body>>(); body != nullptr && (*body)->stream()->is_readable()) { | 
					
						
							| 
									
										
										
										
											2024-11-24 20:39:32 +01:00
										 |  |  |  |         // NOTE: Cancel here is different than the cancel method of stream and refers to https://streams.spec.whatwg.org/#readablestream-cancel
 | 
					
						
							|  |  |  |  |         Streams::readable_stream_cancel((*body)->stream(), error); | 
					
						
							| 
									
										
										
										
											2022-10-23 22:20:25 +01:00
										 |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     // 3. If responseObject is null, then return.
 | 
					
						
							|  |  |  |  |     if (response_object == nullptr) | 
					
						
							|  |  |  |  |         return; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     // 4. Let response be responseObject’s response.
 | 
					
						
							|  |  |  |  |     auto response = response_object->response(); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     // 5. If response’s body is non-null and is readable, then error response’s body with error.
 | 
					
						
							| 
									
										
										
										
											2023-08-18 19:38:13 +02:00
										 |  |  |  |     if (response->body()) { | 
					
						
							| 
									
										
										
										
											2022-10-23 22:20:25 +01:00
										 |  |  |  |         auto stream = response->body()->stream(); | 
					
						
							|  |  |  |  |         if (stream->is_readable()) { | 
					
						
							| 
									
										
										
										
											2024-11-24 20:39:32 +01:00
										 |  |  |  |             stream->error(error); | 
					
						
							| 
									
										
										
										
											2022-10-23 22:20:25 +01:00
										 |  |  |  |         } | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | } |