2021-04-24 02:46:49 +03:00
|
|
|
/*
|
|
|
|
* Copyright (c) 2021, Dex♪ <dexes.ttp@gmail.com>
|
|
|
|
*
|
|
|
|
* SPDX-License-Identifier: BSD-2-Clause
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <LibWebSocket/ConnectionInfo.h>
|
|
|
|
#include <LibWebSocket/Message.h>
|
2022-02-25 13:18:30 +03:00
|
|
|
#include <WebSocket/ConnectionFromClient.h>
|
2021-04-24 02:46:49 +03:00
|
|
|
#include <WebSocket/WebSocketClientEndpoint.h>
|
|
|
|
|
|
|
|
namespace WebSocket {
|
|
|
|
|
2022-02-25 13:18:30 +03:00
|
|
|
static HashMap<int, RefPtr<ConnectionFromClient>> s_connections;
|
2021-04-24 02:46:49 +03:00
|
|
|
|
2023-02-09 01:05:44 +03:00
|
|
|
ConnectionFromClient::ConnectionFromClient(NonnullOwnPtr<Core::LocalSocket> socket)
|
2022-02-25 13:18:30 +03:00
|
|
|
: IPC::ConnectionFromClient<WebSocketClientEndpoint, WebSocketServerEndpoint>(*this, move(socket), 1)
|
2021-04-24 02:46:49 +03:00
|
|
|
{
|
2021-12-06 20:11:05 +03:00
|
|
|
s_connections.set(1, *this);
|
2021-04-24 02:46:49 +03:00
|
|
|
}
|
|
|
|
|
2022-02-25 13:18:30 +03:00
|
|
|
void ConnectionFromClient::die()
|
2021-04-24 02:46:49 +03:00
|
|
|
{
|
|
|
|
s_connections.remove(client_id());
|
|
|
|
if (s_connections.is_empty())
|
|
|
|
Core::EventLoop::current().quit(0);
|
|
|
|
}
|
|
|
|
|
2023-12-16 17:19:34 +03:00
|
|
|
Messages::WebSocketServer::ConnectResponse ConnectionFromClient::connect(URL const& url, ByteString const& origin,
|
|
|
|
Vector<ByteString> const& protocols, Vector<ByteString> const& extensions, HashMap<ByteString, ByteString> const& additional_request_headers)
|
2021-04-24 02:46:49 +03:00
|
|
|
{
|
|
|
|
if (!url.is_valid()) {
|
|
|
|
dbgln("WebSocket::Connect: Invalid URL requested: '{}'", url);
|
2021-05-02 05:39:36 +03:00
|
|
|
return -1;
|
2021-04-24 02:46:49 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
ConnectionInfo connection_info(url);
|
2021-05-02 20:54:34 +03:00
|
|
|
connection_info.set_origin(origin);
|
|
|
|
connection_info.set_protocols(protocols);
|
|
|
|
connection_info.set_extensions(extensions);
|
2021-04-24 02:46:49 +03:00
|
|
|
|
|
|
|
Vector<ConnectionInfo::Header> headers;
|
2023-05-15 00:04:47 +03:00
|
|
|
for (auto const& header : additional_request_headers) {
|
2021-04-24 02:46:49 +03:00
|
|
|
headers.append({ header.key, header.value });
|
|
|
|
}
|
|
|
|
connection_info.set_headers(headers);
|
|
|
|
|
|
|
|
VERIFY(m_connection_ids < NumericLimits<i32>::max());
|
|
|
|
auto id = ++m_connection_ids;
|
|
|
|
auto connection = WebSocket::create(move(connection_info));
|
|
|
|
connection->on_open = [this, id]() {
|
|
|
|
did_connect(id);
|
|
|
|
};
|
|
|
|
connection->on_message = [this, id](auto message) {
|
|
|
|
did_receive_message(id, move(message));
|
|
|
|
};
|
|
|
|
connection->on_error = [this, id](auto message) {
|
|
|
|
did_error(id, (i32)message);
|
|
|
|
};
|
2023-12-16 17:19:34 +03:00
|
|
|
connection->on_close = [this, id](u16 code, ByteString reason, bool was_clean) {
|
2021-04-24 02:46:49 +03:00
|
|
|
did_close(id, code, move(reason), was_clean);
|
|
|
|
};
|
|
|
|
|
|
|
|
connection->start();
|
|
|
|
m_connections.set(id, move(connection));
|
2021-05-02 05:39:36 +03:00
|
|
|
return id;
|
2021-04-24 02:46:49 +03:00
|
|
|
}
|
|
|
|
|
2022-02-25 13:18:30 +03:00
|
|
|
Messages::WebSocketServer::ReadyStateResponse ConnectionFromClient::ready_state(i32 connection_id)
|
2021-04-24 02:46:49 +03:00
|
|
|
{
|
2021-05-02 20:54:34 +03:00
|
|
|
RefPtr<WebSocket> connection = m_connections.get(connection_id).value_or({});
|
2021-04-24 02:46:49 +03:00
|
|
|
if (connection) {
|
2021-05-02 05:39:36 +03:00
|
|
|
return (u32)connection->ready_state();
|
2021-04-24 02:46:49 +03:00
|
|
|
}
|
2021-05-02 05:39:36 +03:00
|
|
|
return (u32)ReadyState::Closed;
|
2021-04-24 02:46:49 +03:00
|
|
|
}
|
|
|
|
|
2022-12-31 13:37:10 +03:00
|
|
|
Messages::WebSocketServer::SubprotocolInUseResponse ConnectionFromClient::subprotocol_in_use(i32 connection_id)
|
|
|
|
{
|
|
|
|
RefPtr<WebSocket> connection = m_connections.get(connection_id).value_or({});
|
|
|
|
if (connection) {
|
|
|
|
return connection->subprotocol_in_use();
|
|
|
|
}
|
2023-12-16 17:19:34 +03:00
|
|
|
return ByteString::empty();
|
2022-12-31 13:37:10 +03:00
|
|
|
}
|
|
|
|
|
2022-02-25 13:18:30 +03:00
|
|
|
void ConnectionFromClient::send(i32 connection_id, bool is_text, ByteBuffer const& data)
|
2021-04-24 02:46:49 +03:00
|
|
|
{
|
2021-05-02 20:54:34 +03:00
|
|
|
RefPtr<WebSocket> connection = m_connections.get(connection_id).value_or({});
|
2021-04-24 02:46:49 +03:00
|
|
|
if (connection && connection->ready_state() == ReadyState::Open) {
|
2021-05-02 20:54:34 +03:00
|
|
|
Message websocket_message(data, is_text);
|
2021-04-24 02:46:49 +03:00
|
|
|
connection->send(websocket_message);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-12-16 17:19:34 +03:00
|
|
|
void ConnectionFromClient::close(i32 connection_id, u16 code, ByteString const& reason)
|
2021-04-24 02:46:49 +03:00
|
|
|
{
|
2021-05-02 20:54:34 +03:00
|
|
|
RefPtr<WebSocket> connection = m_connections.get(connection_id).value_or({});
|
2021-04-24 02:46:49 +03:00
|
|
|
if (connection && connection->ready_state() == ReadyState::Open)
|
2021-05-02 20:54:34 +03:00
|
|
|
connection->close(code, reason);
|
2021-04-24 02:46:49 +03:00
|
|
|
}
|
|
|
|
|
2022-02-25 13:18:30 +03:00
|
|
|
Messages::WebSocketServer::SetCertificateResponse ConnectionFromClient::set_certificate(i32 connection_id,
|
2023-12-16 17:19:34 +03:00
|
|
|
[[maybe_unused]] ByteString const& certificate, [[maybe_unused]] ByteString const& key)
|
2021-04-24 02:46:49 +03:00
|
|
|
{
|
2021-05-02 20:54:34 +03:00
|
|
|
RefPtr<WebSocket> connection = m_connections.get(connection_id).value_or({});
|
2021-04-24 02:46:49 +03:00
|
|
|
bool success = false;
|
|
|
|
if (connection) {
|
|
|
|
// NO OP here
|
2021-05-02 20:54:34 +03:00
|
|
|
// connection->set_certificate(certificate, key);
|
2021-04-24 02:46:49 +03:00
|
|
|
success = true;
|
|
|
|
}
|
2021-05-02 05:39:36 +03:00
|
|
|
return success;
|
2021-04-24 02:46:49 +03:00
|
|
|
}
|
|
|
|
|
2022-02-25 13:18:30 +03:00
|
|
|
void ConnectionFromClient::did_connect(i32 connection_id)
|
2021-04-24 02:46:49 +03:00
|
|
|
{
|
2021-05-03 14:33:59 +03:00
|
|
|
async_connected(connection_id);
|
2021-04-24 02:46:49 +03:00
|
|
|
}
|
|
|
|
|
2022-02-25 13:18:30 +03:00
|
|
|
void ConnectionFromClient::did_receive_message(i32 connection_id, Message message)
|
2021-04-24 02:46:49 +03:00
|
|
|
{
|
2021-05-03 14:33:59 +03:00
|
|
|
async_received(connection_id, message.is_text(), message.data());
|
2021-04-24 02:46:49 +03:00
|
|
|
}
|
|
|
|
|
2022-02-25 13:18:30 +03:00
|
|
|
void ConnectionFromClient::did_error(i32 connection_id, i32 message)
|
2021-04-24 02:46:49 +03:00
|
|
|
{
|
2021-05-03 14:33:59 +03:00
|
|
|
async_errored(connection_id, message);
|
2021-04-24 02:46:49 +03:00
|
|
|
}
|
|
|
|
|
2023-12-16 17:19:34 +03:00
|
|
|
void ConnectionFromClient::did_close(i32 connection_id, u16 code, ByteString reason, bool was_clean)
|
2021-04-24 02:46:49 +03:00
|
|
|
{
|
2021-05-03 14:33:59 +03:00
|
|
|
async_closed(connection_id, code, reason, was_clean);
|
2021-08-30 21:12:48 +03:00
|
|
|
deferred_invoke([this, connection_id] {
|
2021-04-24 02:46:49 +03:00
|
|
|
m_connections.remove(connection_id);
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2022-02-25 13:18:30 +03:00
|
|
|
void ConnectionFromClient::did_request_certificates(i32 connection_id)
|
2021-04-24 02:46:49 +03:00
|
|
|
{
|
2021-05-03 14:33:59 +03:00
|
|
|
async_certificate_requested(connection_id);
|
2021-04-24 02:46:49 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
}
|