/* * Copyright (c) 2024, Tim Flynn * Copyright (c) 2025, Aliaksandr Kalenik * * SPDX-License-Identifier: BSD-2-Clause */ #include #include #include #include #include #include #include #include #include namespace Web::Fetch::Fetching { GC_DEFINE_ALLOCATOR(FetchedDataReceiver); FetchedDataReceiver::FetchedDataReceiver(GC::Ref fetch_params, GC::Ref stream) : m_fetch_params(fetch_params) , m_stream(stream) { } FetchedDataReceiver::~FetchedDataReceiver() = default; void FetchedDataReceiver::visit_edges(Visitor& visitor) { Base::visit_edges(visitor); visitor.visit(m_fetch_params); visitor.visit(m_stream); visitor.visit(m_pending_promise); } void FetchedDataReceiver::set_pending_promise(GC::Ref promise) { VERIFY(!m_pending_promise); VERIFY(!m_has_unfulfilled_promise); m_pending_promise = promise; if (!m_buffer.is_empty()) { pull_bytes_into_stream(move(m_buffer)); } else if (m_lifecycle_state == LifecycleState::ReadyToClose) { close_stream(); } } // This implements the parallel steps of the pullAlgorithm in HTTP-network-fetch. // https://fetch.spec.whatwg.org/#ref-for-in-parallel⑤ void FetchedDataReceiver::handle_network_bytes(ReadonlyBytes bytes, NetworkState state) { VERIFY(m_lifecycle_state == LifecycleState::Receiving); if (state == NetworkState::Complete) { VERIFY(bytes.is_empty()); m_lifecycle_state = LifecycleState::CompletePending; } if (!m_pending_promise) { if (state == NetworkState::Ongoing) m_buffer.append(bytes); if (m_lifecycle_state == LifecycleState::CompletePending && m_buffer.is_empty() && !m_has_unfulfilled_promise) m_lifecycle_state = LifecycleState::ReadyToClose; return; } // 1. If one or more bytes have been transmitted from response’s message body, then: if (!bytes.is_empty()) { // 1. Let bytes be the transmitted bytes. // FIXME: 2. Let codings be the result of extracting header list values given `Content-Encoding` and response’s header list. // FIXME: 3. Increase response’s body info’s encoded size by bytes’s length. // FIXME: 4. Set bytes to the result of handling content codings given codings and bytes. // FIXME: 5. Increase response’s body info’s decoded size by bytes’s length. // FIXME: 6. If bytes is failure, then terminate fetchParams’s controller. // 7. Append bytes to buffer. pull_bytes_into_stream(MUST(ByteBuffer::copy(bytes))); // FIXME: 8. If the size of buffer is larger than an upper limit chosen by the user agent, ask the user agent // to suspend the ongoing fetch. return; } // 2. Otherwise, if the bytes transmission for response’s message body is done normally and stream is readable, // then close stream, and abort these in-parallel steps. if (m_stream->is_readable()) { VERIFY(m_lifecycle_state == LifecycleState::CompletePending); close_stream(); } } // This implements the parallel steps of the pullAlgorithm in HTTP-network-fetch. // https://fetch.spec.whatwg.org/#ref-for-in-parallel④ void FetchedDataReceiver::pull_bytes_into_stream(ByteBuffer&& bytes) { // FIXME: 1. If the size of buffer is smaller than a lower limit chosen by the user agent and the ongoing fetch // is suspended, resume the fetch. // 2. Wait until buffer is not empty. VERIFY(!bytes.is_empty()); VERIFY(m_lifecycle_state == LifecycleState::Receiving || m_lifecycle_state == LifecycleState::CompletePending); // 3. Queue a fetch task to run the following steps, with fetchParams’s task destination. VERIFY(!m_has_unfulfilled_promise); m_has_unfulfilled_promise = true; Infrastructure::queue_fetch_task( m_fetch_params->controller(), m_fetch_params->task_destination(), GC::create_function(heap(), [this, bytes = move(bytes), pending_promise = m_pending_promise]() mutable { m_has_unfulfilled_promise = false; VERIFY(m_lifecycle_state == LifecycleState::Receiving || m_lifecycle_state == LifecycleState::CompletePending); HTML::TemporaryExecutionContext execution_context { m_stream->realm(), HTML::TemporaryExecutionContext::CallbacksEnabled::Yes }; // 1. Pull from bytes buffer into stream. if (auto result = m_stream->pull_from_bytes(move(bytes)); result.is_error()) { auto throw_completion = Bindings::exception_to_throw_completion(m_stream->vm(), result.release_error()); dbgln("FetchedDataReceiver: Stream error pulling bytes"); HTML::report_exception(throw_completion, m_stream->realm()); return; } // 2. If stream is errored, then terminate fetchParams’s controller. if (m_stream->is_errored()) m_fetch_params->controller()->terminate(); // 3. Resolve promise with undefined. WebIDL::resolve_promise(m_stream->realm(), *pending_promise, JS::js_undefined()); if (m_lifecycle_state == LifecycleState::CompletePending && m_buffer.is_empty()) m_lifecycle_state = LifecycleState::ReadyToClose; })); m_pending_promise = {}; } void FetchedDataReceiver::close_stream() { VERIFY(m_has_unfulfilled_promise == 0); WebIDL::resolve_promise(m_stream->realm(), *m_pending_promise, JS::js_undefined()); m_pending_promise = {}; m_lifecycle_state = LifecycleState::Closed; m_stream->close(); } }