/* * Copyright (c) 2018-2020, Andreas Kling * * SPDX-License-Identifier: BSD-2-Clause */ #include #include namespace Protocol { RequestClient::RequestClient(NonnullOwnPtr socket) : IPC::ConnectionToServer(*this, move(socket)) { } void RequestClient::ensure_connection(URL const& url, ::RequestServer::CacheLevel cache_level) { async_ensure_connection(url, cache_level); } template RefPtr RequestClient::start_request(ByteString const& method, URL const& url, HashMap const& request_headers, ReadonlyBytes request_body, Core::ProxyData const& proxy_data) { auto headers_or_error = request_headers.template clone>(); if (headers_or_error.is_error()) return nullptr; 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, headers_or_error.release_value(), 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) { if (!m_requests.contains(request.id())) return false; return IPCProxy::stop_request(request.id()); } bool RequestClient::set_certificate(Badge, 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, bool success, u64 total_size) { RefPtr request; if ((request = m_requests.get(request_id).value_or(nullptr))) { request->did_finish({}, success, total_size); } m_requests.remove(request_id); } void RequestClient::request_progress(i32 request_id, Optional const& total_size, u64 downloaded_size) { if (auto request = const_cast(m_requests.get(request_id).value_or(nullptr))) { request->did_progress({}, total_size, downloaded_size); } } void RequestClient::headers_became_available(i32 request_id, HashMap const& response_headers, Optional const& status_code) { auto request = const_cast(m_requests.get(request_id).value_or(nullptr)); if (!request) { warnln("Received headers for non-existent request {}", request_id); return; } auto response_headers_clone_or_error = response_headers.clone(); if (response_headers_clone_or_error.is_error()) { warnln("Error while receiving headers for request {}: {}", request_id, response_headers_clone_or_error.error()); return; } request->did_receive_headers({}, response_headers_clone_or_error.release_value(), status_code); } void RequestClient::certificate_requested(i32 request_id) { if (auto request = const_cast(m_requests.get(request_id).value_or(nullptr))) { request->did_request_certificates({}); } } RefPtr RequestClient::websocket_connect(const URL& url, ByteString const& origin, Vector const& protocols, Vector const& extensions, HashMap const& request_headers) { auto headers_or_error = request_headers.clone(); if (headers_or_error.is_error()) return nullptr; auto connection_id = IPCProxy::websocket_connect(url, origin, protocols, extensions, headers_or_error.release_value()); if (connection_id < 0) return nullptr; auto connection = WebSocket::create_from_id({}, *this, connection_id); m_websockets.set(connection_id, connection); return connection; } void RequestClient::websocket_connected(i32 connection_id) { auto maybe_connection = m_websockets.get(connection_id); if (maybe_connection.has_value()) maybe_connection.value()->did_open({}); } void RequestClient::websocket_received(i32 connection_id, bool is_text, ByteBuffer const& data) { auto maybe_connection = m_websockets.get(connection_id); if (maybe_connection.has_value()) maybe_connection.value()->did_receive({}, data, is_text); } void RequestClient::websocket_errored(i32 connection_id, i32 message) { auto maybe_connection = m_websockets.get(connection_id); if (maybe_connection.has_value()) maybe_connection.value()->did_error({}, message); } void RequestClient::websocket_closed(i32 connection_id, u16 code, ByteString const& reason, bool clean) { auto maybe_connection = m_websockets.get(connection_id); if (maybe_connection.has_value()) maybe_connection.value()->did_close({}, code, reason, clean); } void RequestClient::websocket_certificate_requested(i32 connection_id) { auto maybe_connection = m_websockets.get(connection_id); if (maybe_connection.has_value()) maybe_connection.value()->did_request_certificates({}); } } template RefPtr Protocol::RequestClient::start_request(ByteString const& method, URL const&, HashMap const& request_headers, ReadonlyBytes request_body, Core::ProxyData const&); template RefPtr Protocol::RequestClient::start_request(ByteString const& method, URL const&, HashMap const& request_headers, ReadonlyBytes request_body, Core::ProxyData const&);