2024-05-26 08:03:29 -04:00
|
|
|
|
/*
|
|
|
|
|
|
* Copyright (c) 2024, Tim Flynn <trflynn89@serenityos.org>
|
2025-04-17 07:58:24 -04:00
|
|
|
|
* Copyright (c) 2025, Aliaksandr Kalenik <kalenik.aliaksandr@gmail.com>
|
2024-05-26 08:03:29 -04:00
|
|
|
|
*
|
|
|
|
|
|
* SPDX-License-Identifier: BSD-2-Clause
|
|
|
|
|
|
*/
|
|
|
|
|
|
|
2024-11-15 04:01:23 +13:00
|
|
|
|
#include <LibGC/Function.h>
|
2024-05-26 08:03:29 -04:00
|
|
|
|
#include <LibWeb/Bindings/ExceptionOrUtils.h>
|
|
|
|
|
|
#include <LibWeb/Fetch/Fetching/FetchedDataReceiver.h>
|
|
|
|
|
|
#include <LibWeb/Fetch/Infrastructure/FetchParams.h>
|
|
|
|
|
|
#include <LibWeb/Fetch/Infrastructure/Task.h>
|
|
|
|
|
|
#include <LibWeb/HTML/Scripting/ExceptionReporter.h>
|
|
|
|
|
|
#include <LibWeb/HTML/Scripting/TemporaryExecutionContext.h>
|
2025-04-17 15:47:53 -04:00
|
|
|
|
#include <LibWeb/Streams/ReadableStream.h>
|
2024-05-26 08:03:29 -04:00
|
|
|
|
#include <LibWeb/WebIDL/Promise.h>
|
|
|
|
|
|
|
|
|
|
|
|
namespace Web::Fetch::Fetching {
|
|
|
|
|
|
|
2024-11-15 04:01:23 +13:00
|
|
|
|
GC_DEFINE_ALLOCATOR(FetchedDataReceiver);
|
2024-05-26 08:03:29 -04:00
|
|
|
|
|
2024-11-15 04:01:23 +13:00
|
|
|
|
FetchedDataReceiver::FetchedDataReceiver(GC::Ref<Infrastructure::FetchParams const> fetch_params, GC::Ref<Streams::ReadableStream> stream)
|
2024-05-26 08:03:29 -04:00
|
|
|
|
: 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);
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2024-11-15 04:01:23 +13:00
|
|
|
|
void FetchedDataReceiver::set_pending_promise(GC::Ref<WebIDL::Promise> promise)
|
2024-05-26 08:03:29 -04:00
|
|
|
|
{
|
2025-04-17 07:58:24 -04:00
|
|
|
|
VERIFY(!m_pending_promise);
|
|
|
|
|
|
VERIFY(!m_has_unfulfilled_promise);
|
2024-05-26 08:03:29 -04:00
|
|
|
|
m_pending_promise = promise;
|
|
|
|
|
|
|
2025-04-17 07:58:24 -04:00
|
|
|
|
if (!m_buffer.is_empty()) {
|
|
|
|
|
|
pull_bytes_into_stream(move(m_buffer));
|
|
|
|
|
|
} else if (m_lifecycle_state == LifecycleState::ReadyToClose) {
|
|
|
|
|
|
close_stream();
|
2024-05-26 08:03:29 -04:00
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// This implements the parallel steps of the pullAlgorithm in HTTP-network-fetch.
|
2025-04-17 07:58:24 -04:00
|
|
|
|
// https://fetch.spec.whatwg.org/#ref-for-in-parallel⑤
|
|
|
|
|
|
void FetchedDataReceiver::handle_network_bytes(ReadonlyBytes bytes, NetworkState state)
|
2024-05-26 08:03:29 -04:00
|
|
|
|
{
|
2025-04-17 07:58:24 -04:00
|
|
|
|
VERIFY(m_lifecycle_state == LifecycleState::Receiving);
|
|
|
|
|
|
|
|
|
|
|
|
if (state == NetworkState::Complete) {
|
|
|
|
|
|
VERIFY(bytes.is_empty());
|
|
|
|
|
|
m_lifecycle_state = LifecycleState::CompletePending;
|
|
|
|
|
|
}
|
2024-05-26 08:03:29 -04:00
|
|
|
|
|
|
|
|
|
|
if (!m_pending_promise) {
|
2025-04-17 07:58:24 -04:00
|
|
|
|
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;
|
2024-05-26 08:03:29 -04:00
|
|
|
|
return;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2025-04-17 07:58:24 -04:00
|
|
|
|
// 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);
|
|
|
|
|
|
|
2024-05-26 08:03:29 -04:00
|
|
|
|
// 3. Queue a fetch task to run the following steps, with fetchParams’s task destination.
|
2025-04-17 07:58:24 -04:00
|
|
|
|
VERIFY(!m_has_unfulfilled_promise);
|
|
|
|
|
|
m_has_unfulfilled_promise = true;
|
2024-05-26 08:03:29 -04:00
|
|
|
|
Infrastructure::queue_fetch_task(
|
|
|
|
|
|
m_fetch_params->controller(),
|
2025-07-16 12:29:26 +02:00
|
|
|
|
m_fetch_params->task_destination(),
|
2025-04-17 07:58:24 -04:00
|
|
|
|
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);
|
|
|
|
|
|
|
2024-10-24 20:39:18 +13:00
|
|
|
|
HTML::TemporaryExecutionContext execution_context { m_stream->realm(), HTML::TemporaryExecutionContext::CallbacksEnabled::Yes };
|
2024-05-26 08:03:29 -04:00
|
|
|
|
|
|
|
|
|
|
// 1. Pull from bytes buffer into stream.
|
2024-12-08 17:28:44 +13:00
|
|
|
|
if (auto result = m_stream->pull_from_bytes(move(bytes)); result.is_error()) {
|
2024-11-04 14:37:27 +01:00
|
|
|
|
auto throw_completion = Bindings::exception_to_throw_completion(m_stream->vm(), result.release_error());
|
2024-05-26 08:03:29 -04:00
|
|
|
|
|
|
|
|
|
|
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.
|
2025-04-17 07:58:24 -04:00
|
|
|
|
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;
|
2024-05-26 08:03:29 -04:00
|
|
|
|
}));
|
2025-04-17 07:58:24 -04:00
|
|
|
|
|
|
|
|
|
|
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();
|
2024-05-26 08:03:29 -04:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
}
|