mirror of
https://github.com/LadybirdBrowser/ladybird.git
synced 2024-11-22 07:30:19 +00:00
158 lines
5.5 KiB
C++
158 lines
5.5 KiB
C++
/*
|
|
* Copyright (c) 2018-2020, Andreas Kling <andreas@ladybird.org>
|
|
*
|
|
* SPDX-License-Identifier: BSD-2-Clause
|
|
*/
|
|
|
|
#include <LibRequests/Request.h>
|
|
#include <LibRequests/RequestClient.h>
|
|
|
|
namespace Requests {
|
|
|
|
RequestClient::RequestClient(IPC::Transport transport)
|
|
: IPC::ConnectionToServer<RequestClientEndpoint, RequestServerEndpoint>(*this, move(transport))
|
|
{
|
|
}
|
|
|
|
RequestClient::~RequestClient() = default;
|
|
|
|
void RequestClient::die()
|
|
{
|
|
// FIXME: Gracefully handle this, or relaunch and reconnect to RequestServer.
|
|
warnln("\033[31;1m {} Lost connection to RequestServer\033[0m", getpid());
|
|
}
|
|
|
|
void RequestClient::ensure_connection(URL::URL const& url, ::RequestServer::CacheLevel cache_level)
|
|
{
|
|
async_ensure_connection(url, cache_level);
|
|
}
|
|
|
|
RefPtr<Request> RequestClient::start_request(ByteString const& method, URL::URL const& url, HTTP::HeaderMap const& request_headers, ReadonlyBytes request_body, Core::ProxyData const& proxy_data)
|
|
{
|
|
auto body_result = ByteBuffer::copy(request_body);
|
|
if (body_result.is_error())
|
|
return nullptr;
|
|
|
|
static i32 s_next_request_id = 0;
|
|
auto request_id = s_next_request_id++;
|
|
|
|
IPCProxy::async_start_request(request_id, method, url, request_headers, body_result.release_value(), proxy_data);
|
|
auto request = Request::create_from_id({}, *this, request_id);
|
|
m_requests.set(request_id, request);
|
|
return request;
|
|
}
|
|
|
|
void RequestClient::request_started(i32 request_id, IPC::File const& response_file)
|
|
{
|
|
auto request = m_requests.get(request_id);
|
|
if (!request.has_value()) {
|
|
warnln("Received response for non-existent request {}", request_id);
|
|
return;
|
|
}
|
|
|
|
auto response_fd = response_file.take_fd();
|
|
request.value()->set_request_fd({}, response_fd);
|
|
}
|
|
|
|
bool RequestClient::stop_request(Badge<Request>, Request& request)
|
|
{
|
|
if (!m_requests.contains(request.id()))
|
|
return false;
|
|
return IPCProxy::stop_request(request.id());
|
|
}
|
|
|
|
bool RequestClient::set_certificate(Badge<Request>, Request& request, ByteString certificate, ByteString key)
|
|
{
|
|
if (!m_requests.contains(request.id()))
|
|
return false;
|
|
return IPCProxy::set_certificate(request.id(), move(certificate), move(key));
|
|
}
|
|
|
|
void RequestClient::request_finished(i32 request_id, u64 total_size, Optional<NetworkError> const& network_error)
|
|
{
|
|
RefPtr<Request> request;
|
|
if ((request = m_requests.get(request_id).value_or(nullptr))) {
|
|
request->did_finish({}, total_size, network_error);
|
|
}
|
|
m_requests.remove(request_id);
|
|
}
|
|
|
|
void RequestClient::headers_became_available(i32 request_id, HTTP::HeaderMap const& response_headers, Optional<u32> const& status_code, Optional<String> const& reason_phrase)
|
|
{
|
|
auto request = const_cast<Request*>(m_requests.get(request_id).value_or(nullptr));
|
|
if (!request) {
|
|
warnln("Received headers for non-existent request {}", request_id);
|
|
return;
|
|
}
|
|
request->did_receive_headers({}, response_headers, status_code, reason_phrase);
|
|
}
|
|
|
|
void RequestClient::certificate_requested(i32 request_id)
|
|
{
|
|
if (auto request = const_cast<Request*>(m_requests.get(request_id).value_or(nullptr))) {
|
|
request->did_request_certificates({});
|
|
}
|
|
}
|
|
|
|
RefPtr<WebSocket> RequestClient::websocket_connect(const URL::URL& url, ByteString const& origin, Vector<ByteString> const& protocols, Vector<ByteString> const& extensions, HTTP::HeaderMap const& request_headers)
|
|
{
|
|
auto websocket_id = m_next_websocket_id++;
|
|
IPCProxy::async_websocket_connect(websocket_id, url, origin, protocols, extensions, request_headers);
|
|
auto connection = WebSocket::create_from_id({}, *this, websocket_id);
|
|
m_websockets.set(websocket_id, connection);
|
|
return connection;
|
|
}
|
|
|
|
void RequestClient::websocket_connected(i64 websocket_id)
|
|
{
|
|
auto maybe_connection = m_websockets.get(websocket_id);
|
|
if (maybe_connection.has_value())
|
|
maybe_connection.value()->did_open({});
|
|
}
|
|
|
|
void RequestClient::websocket_received(i64 websocket_id, bool is_text, ByteBuffer const& data)
|
|
{
|
|
auto maybe_connection = m_websockets.get(websocket_id);
|
|
if (maybe_connection.has_value())
|
|
maybe_connection.value()->did_receive({}, data, is_text);
|
|
}
|
|
|
|
void RequestClient::websocket_errored(i64 websocket_id, i32 message)
|
|
{
|
|
auto maybe_connection = m_websockets.get(websocket_id);
|
|
if (maybe_connection.has_value())
|
|
maybe_connection.value()->did_error({}, message);
|
|
}
|
|
|
|
void RequestClient::websocket_closed(i64 websocket_id, u16 code, ByteString const& reason, bool clean)
|
|
{
|
|
auto maybe_connection = m_websockets.get(websocket_id);
|
|
if (maybe_connection.has_value())
|
|
maybe_connection.value()->did_close({}, code, reason, clean);
|
|
}
|
|
|
|
void RequestClient::websocket_ready_state_changed(i64 websocket_id, u32 ready_state)
|
|
{
|
|
auto maybe_connection = m_websockets.get(websocket_id);
|
|
if (maybe_connection.has_value()) {
|
|
VERIFY(ready_state <= static_cast<u32>(WebSocket::ReadyState::Closed));
|
|
maybe_connection.value()->set_ready_state(static_cast<WebSocket::ReadyState>(ready_state));
|
|
}
|
|
}
|
|
|
|
void RequestClient::websocket_subprotocol(i64 websocket_id, ByteString const& subprotocol)
|
|
{
|
|
auto maybe_connection = m_websockets.get(websocket_id);
|
|
if (maybe_connection.has_value()) {
|
|
maybe_connection.value()->set_subprotocol_in_use(subprotocol);
|
|
}
|
|
}
|
|
|
|
void RequestClient::websocket_certificate_requested(i64 websocket_id)
|
|
{
|
|
auto maybe_connection = m_websockets.get(websocket_id);
|
|
if (maybe_connection.has_value())
|
|
maybe_connection.value()->did_request_certificates({});
|
|
}
|
|
|
|
}
|