RequestClient.cpp 5.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158
  1. /*
  2. * Copyright (c) 2018-2020, Andreas Kling <kling@serenityos.org>
  3. *
  4. * SPDX-License-Identifier: BSD-2-Clause
  5. */
  6. #include <LibRequests/Request.h>
  7. #include <LibRequests/RequestClient.h>
  8. namespace Requests {
  9. RequestClient::RequestClient(NonnullOwnPtr<Core::LocalSocket> socket)
  10. : IPC::ConnectionToServer<RequestClientEndpoint, RequestServerEndpoint>(*this, move(socket))
  11. {
  12. }
  13. RequestClient::~RequestClient() = default;
  14. void RequestClient::die()
  15. {
  16. // FIXME: Gracefully handle this, or relaunch and reconnect to RequestServer.
  17. warnln("\033[31;1m {} Lost connection to RequestServer\033[0m", getpid());
  18. }
  19. void RequestClient::ensure_connection(URL::URL const& url, ::RequestServer::CacheLevel cache_level)
  20. {
  21. async_ensure_connection(url, cache_level);
  22. }
  23. 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)
  24. {
  25. auto body_result = ByteBuffer::copy(request_body);
  26. if (body_result.is_error())
  27. return nullptr;
  28. static i32 s_next_request_id = 0;
  29. auto request_id = s_next_request_id++;
  30. IPCProxy::async_start_request(request_id, method, url, request_headers, body_result.release_value(), proxy_data);
  31. auto request = Request::create_from_id({}, *this, request_id);
  32. m_requests.set(request_id, request);
  33. return request;
  34. }
  35. void RequestClient::request_started(i32 request_id, IPC::File const& response_file)
  36. {
  37. auto request = m_requests.get(request_id);
  38. if (!request.has_value()) {
  39. warnln("Received response for non-existent request {}", request_id);
  40. return;
  41. }
  42. auto response_fd = response_file.take_fd();
  43. request.value()->set_request_fd({}, response_fd);
  44. }
  45. bool RequestClient::stop_request(Badge<Request>, Request& request)
  46. {
  47. if (!m_requests.contains(request.id()))
  48. return false;
  49. return IPCProxy::stop_request(request.id());
  50. }
  51. bool RequestClient::set_certificate(Badge<Request>, Request& request, ByteString certificate, ByteString key)
  52. {
  53. if (!m_requests.contains(request.id()))
  54. return false;
  55. return IPCProxy::set_certificate(request.id(), move(certificate), move(key));
  56. }
  57. void RequestClient::request_finished(i32 request_id, bool success, u64 total_size)
  58. {
  59. RefPtr<Request> request;
  60. if ((request = m_requests.get(request_id).value_or(nullptr))) {
  61. request->did_finish({}, success, total_size);
  62. }
  63. m_requests.remove(request_id);
  64. }
  65. void RequestClient::headers_became_available(i32 request_id, HTTP::HeaderMap const& response_headers, Optional<u32> const& status_code)
  66. {
  67. auto request = const_cast<Request*>(m_requests.get(request_id).value_or(nullptr));
  68. if (!request) {
  69. warnln("Received headers for non-existent request {}", request_id);
  70. return;
  71. }
  72. request->did_receive_headers({}, response_headers, status_code);
  73. }
  74. void RequestClient::certificate_requested(i32 request_id)
  75. {
  76. if (auto request = const_cast<Request*>(m_requests.get(request_id).value_or(nullptr))) {
  77. request->did_request_certificates({});
  78. }
  79. }
  80. 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)
  81. {
  82. auto websocket_id = m_next_websocket_id++;
  83. IPCProxy::async_websocket_connect(websocket_id, url, origin, protocols, extensions, request_headers);
  84. auto connection = WebSocket::create_from_id({}, *this, websocket_id);
  85. m_websockets.set(websocket_id, connection);
  86. return connection;
  87. }
  88. void RequestClient::websocket_connected(i64 websocket_id)
  89. {
  90. auto maybe_connection = m_websockets.get(websocket_id);
  91. if (maybe_connection.has_value())
  92. maybe_connection.value()->did_open({});
  93. }
  94. void RequestClient::websocket_received(i64 websocket_id, bool is_text, ByteBuffer const& data)
  95. {
  96. auto maybe_connection = m_websockets.get(websocket_id);
  97. if (maybe_connection.has_value())
  98. maybe_connection.value()->did_receive({}, data, is_text);
  99. }
  100. void RequestClient::websocket_errored(i64 websocket_id, i32 message)
  101. {
  102. auto maybe_connection = m_websockets.get(websocket_id);
  103. if (maybe_connection.has_value())
  104. maybe_connection.value()->did_error({}, message);
  105. }
  106. void RequestClient::websocket_closed(i64 websocket_id, u16 code, ByteString const& reason, bool clean)
  107. {
  108. auto maybe_connection = m_websockets.get(websocket_id);
  109. if (maybe_connection.has_value())
  110. maybe_connection.value()->did_close({}, code, reason, clean);
  111. }
  112. void RequestClient::websocket_ready_state_changed(i64 websocket_id, u32 ready_state)
  113. {
  114. auto maybe_connection = m_websockets.get(websocket_id);
  115. if (maybe_connection.has_value()) {
  116. VERIFY(ready_state <= static_cast<u32>(WebSocket::ReadyState::Closed));
  117. maybe_connection.value()->set_ready_state(static_cast<WebSocket::ReadyState>(ready_state));
  118. }
  119. }
  120. void RequestClient::websocket_subprotocol(i64 websocket_id, ByteString const& subprotocol)
  121. {
  122. auto maybe_connection = m_websockets.get(websocket_id);
  123. if (maybe_connection.has_value()) {
  124. maybe_connection.value()->set_subprotocol_in_use(subprotocol);
  125. }
  126. }
  127. void RequestClient::websocket_certificate_requested(i64 websocket_id)
  128. {
  129. auto maybe_connection = m_websockets.get(websocket_id);
  130. if (maybe_connection.has_value())
  131. maybe_connection.value()->did_request_certificates({});
  132. }
  133. }