RequestClient.cpp 5.2 KB

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