mirror of
https://github.com/LadybirdBrowser/ladybird
synced 2026-05-11 01:22:43 +02:00
handle_connect_state (used by <link rel=preconnect>) attached easy handles to the multi without setting CURLOPT_RESOLVE, so libcurl spawned its own threaded resolver for them. When the system stub resolver was slow, the thread got stuck and the next ~Request pthread_join'd it on the main thread for many seconds. Route Connect-mode requests through the same DNS + CURLOPT_RESOLVE path that Fetch uses, so libcurl never spawns the thread.
221 lines
7.1 KiB
C++
221 lines
7.1 KiB
C++
/*
|
|
* Copyright (c) 2025-2026, Tim Flynn <trflynn89@ladybird.org>
|
|
*
|
|
* SPDX-License-Identifier: BSD-2-Clause
|
|
*/
|
|
|
|
#pragma once
|
|
|
|
#include <AK/ByteBuffer.h>
|
|
#include <AK/ByteString.h>
|
|
#include <AK/MemoryStream.h>
|
|
#include <AK/Optional.h>
|
|
#include <AK/Time.h>
|
|
#include <LibCore/Proxy.h>
|
|
#include <LibDNS/Resolver.h>
|
|
#include <LibHTTP/Cache/CacheMode.h>
|
|
#include <LibHTTP/Cache/CacheRequest.h>
|
|
#include <LibHTTP/Cookie/IncludeCredentials.h>
|
|
#include <LibHTTP/HeaderList.h>
|
|
#include <LibRequests/NetworkError.h>
|
|
#include <LibRequests/RequestTimingInfo.h>
|
|
#include <LibURL/URL.h>
|
|
#include <RequestServer/CacheLevel.h>
|
|
#include <RequestServer/Forward.h>
|
|
#include <RequestServer/RequestPipe.h>
|
|
#include <RequestServer/RequestType.h>
|
|
|
|
struct curl_slist;
|
|
|
|
namespace RequestServer {
|
|
|
|
class Request final : public HTTP::CacheRequest {
|
|
public:
|
|
static NonnullOwnPtr<Request> fetch(
|
|
u64 request_id,
|
|
Optional<HTTP::DiskCache&> disk_cache,
|
|
HTTP::CacheMode cache_mode,
|
|
ConnectionFromClient& client,
|
|
void* curl_multi,
|
|
Resolver& resolver,
|
|
URL::URL url,
|
|
ByteString method,
|
|
NonnullRefPtr<HTTP::HeaderList> request_headers,
|
|
ByteBuffer request_body,
|
|
HTTP::Cookie::IncludeCredentials include_credentials,
|
|
ByteString alt_svc_cache_path,
|
|
Core::ProxyData proxy_data);
|
|
|
|
static NonnullOwnPtr<Request> connect(
|
|
u64 request_id,
|
|
ConnectionFromClient& client,
|
|
void* curl_multi,
|
|
Resolver& resolver,
|
|
URL::URL url,
|
|
CacheLevel cache_level);
|
|
|
|
static NonnullOwnPtr<Request> revalidate(
|
|
u64 request_id,
|
|
Optional<HTTP::DiskCache&> disk_cache,
|
|
ConnectionFromClient& client,
|
|
void* curl_multi,
|
|
Resolver& resolver,
|
|
URL::URL url,
|
|
ByteString method,
|
|
NonnullRefPtr<HTTP::HeaderList> request_headers,
|
|
ByteBuffer request_body,
|
|
HTTP::Cookie::IncludeCredentials include_credentials,
|
|
ByteString alt_svc_cache_path,
|
|
Core::ProxyData proxy_data);
|
|
|
|
virtual ~Request() override;
|
|
|
|
u64 request_id() const { return m_request_id; }
|
|
RequestType type() const { return m_type; }
|
|
URL::URL const& url() const { return m_url; }
|
|
|
|
virtual void notify_request_unblocked(Badge<HTTP::DiskCache>) override;
|
|
void notify_retrieved_http_cookie(Badge<ConnectionFromClient>, StringView cookie);
|
|
void notify_fetch_complete(Badge<ConnectionFromClient>, int result_code);
|
|
|
|
private:
|
|
enum class State : u8 {
|
|
Init, // Decide whether to service this request from cache or the network.
|
|
ReadCache, // Read the cached response from disk.
|
|
WaitForCache, // Wait for an existing cache entry to complete before proceeding.
|
|
FailedCacheOnly, // An only-if-cached request failed to find a cache entry.
|
|
ServeSubstitution, // Serve content from a local file substitution.
|
|
DNSLookup, // Resolve the URL's host.
|
|
RetrieveCookie, // Retrieve cookies from the UI process.
|
|
Connect, // Issue a network request to connect to the URL.
|
|
Fetch, // Issue a network request to fetch the URL.
|
|
Complete, // Finalize the request with the client.
|
|
Error, // Any error occured during the request's lifetime.
|
|
};
|
|
|
|
static constexpr StringView state_name(State state)
|
|
{
|
|
switch (state) {
|
|
case State::Init:
|
|
return "Init"sv;
|
|
case State::ReadCache:
|
|
return "ReadCache"sv;
|
|
case State::WaitForCache:
|
|
return "WaitForCache"sv;
|
|
case State::FailedCacheOnly:
|
|
return "FailedCacheOnly"sv;
|
|
case State::ServeSubstitution:
|
|
return "ServeSubstitution"sv;
|
|
case State::DNSLookup:
|
|
return "DNSLookup"sv;
|
|
case State::RetrieveCookie:
|
|
return "RetrieveCookie"sv;
|
|
case State::Connect:
|
|
return "Connect"sv;
|
|
case State::Fetch:
|
|
return "Fetch"sv;
|
|
case State::Complete:
|
|
return "Complete"sv;
|
|
case State::Error:
|
|
return "Error"sv;
|
|
}
|
|
VERIFY_NOT_REACHED();
|
|
}
|
|
|
|
Request(
|
|
u64 request_id,
|
|
RequestType type,
|
|
Optional<HTTP::DiskCache&> disk_cache,
|
|
HTTP::CacheMode cache_mode,
|
|
ConnectionFromClient& client,
|
|
void* curl_multi,
|
|
Resolver& resolver,
|
|
URL::URL url,
|
|
ByteString method,
|
|
NonnullRefPtr<HTTP::HeaderList> request_headers,
|
|
ByteBuffer request_body,
|
|
HTTP::Cookie::IncludeCredentials include_credentials,
|
|
ByteString alt_svc_cache_path,
|
|
Core::ProxyData proxy_data);
|
|
|
|
Request(
|
|
u64 request_id,
|
|
ConnectionFromClient& client,
|
|
void* curl_multi,
|
|
Resolver& resolver,
|
|
URL::URL url);
|
|
|
|
void transition_to_state(State);
|
|
void process();
|
|
|
|
void handle_initial_state();
|
|
void handle_read_cache_state();
|
|
void handle_failed_cache_only_state();
|
|
void handle_serve_substitution_state();
|
|
void handle_dns_lookup_state();
|
|
void handle_retrieve_cookie_state();
|
|
void handle_connect_state();
|
|
void handle_fetch_state();
|
|
void handle_complete_state();
|
|
void handle_error_state();
|
|
|
|
static size_t on_header_received(void* buffer, size_t size, size_t nmemb, void* user_data);
|
|
static size_t on_data_received(void* buffer, size_t size, size_t nmemb, void* user_data);
|
|
|
|
ErrorOr<void> inform_client_request_started();
|
|
void transfer_headers_to_client_if_needed();
|
|
ErrorOr<void> write_queued_bytes_without_blocking();
|
|
|
|
virtual bool is_revalidation_request() const override;
|
|
ErrorOr<void> revalidation_failed();
|
|
|
|
bool is_cache_only_request() const;
|
|
|
|
u32 acquire_status_code() const;
|
|
Requests::RequestTimingInfo acquire_timing_info() const;
|
|
|
|
u64 m_request_id { 0 };
|
|
RequestType m_type { RequestType::Fetch };
|
|
State m_state { State::Init };
|
|
|
|
Optional<HTTP::DiskCache&> m_disk_cache;
|
|
HTTP::CacheMode m_cache_mode { HTTP::CacheMode::Default };
|
|
ConnectionFromClient& m_client;
|
|
|
|
void* m_curl_multi_handle { nullptr };
|
|
void* m_curl_easy_handle { nullptr };
|
|
Vector<curl_slist*> m_curl_string_lists;
|
|
Optional<int> m_curl_result_code;
|
|
|
|
NonnullRefPtr<Resolver> m_resolver;
|
|
RefPtr<DNS::LookupResult const> m_dns_result;
|
|
CacheLevel m_connect_cache_level { CacheLevel::ResolveOnly };
|
|
|
|
URL::URL m_url;
|
|
ByteString m_method;
|
|
|
|
UnixDateTime m_request_start_time { UnixDateTime::now() };
|
|
NonnullRefPtr<HTTP::HeaderList> m_request_headers;
|
|
ByteBuffer m_request_body;
|
|
|
|
HTTP::Cookie::IncludeCredentials m_include_credentials { HTTP::Cookie::IncludeCredentials::Yes };
|
|
|
|
ByteString m_alt_svc_cache_path;
|
|
Core::ProxyData m_proxy_data;
|
|
|
|
Optional<u32> m_status_code;
|
|
Optional<String> m_reason_phrase;
|
|
|
|
NonnullRefPtr<HTTP::HeaderList> m_response_headers;
|
|
bool m_sent_response_headers_to_client { false };
|
|
|
|
AllocatingMemoryStream m_response_buffer;
|
|
RefPtr<Core::Notifier> m_client_writer_notifier;
|
|
Optional<RequestPipe> m_client_request_pipe;
|
|
size_t m_bytes_transferred_to_client { 0 };
|
|
|
|
Optional<Requests::NetworkError> m_network_error;
|
|
};
|
|
|
|
}
|