mirror of
https://github.com/RGBCube/serenity
synced 2025-07-25 16:17:45 +00:00
LibWeb: Abstract the LibProtocol WebSockets connection
Much like the ImageDecoder change, this moves the underlying connection of the Web::WebSockets class from LibWeb to LibWebView, removing the need for LibProtocol in LibWeb for this specific use-case.
This commit is contained in:
parent
2198091bbc
commit
2a359695c6
6 changed files with 255 additions and 54 deletions
|
@ -1,5 +1,5 @@
|
||||||
/*
|
/*
|
||||||
* Copyright (c) 2021, Dex♪ <dexes.ttp@gmail.com>
|
* Copyright (c) 2021-2022, Dex♪ <dexes.ttp@gmail.com>
|
||||||
*
|
*
|
||||||
* SPDX-License-Identifier: BSD-2-Clause
|
* SPDX-License-Identifier: BSD-2-Clause
|
||||||
*/
|
*/
|
||||||
|
@ -8,8 +8,6 @@
|
||||||
#include <LibJS/Parser.h>
|
#include <LibJS/Parser.h>
|
||||||
#include <LibJS/Runtime/ArrayBuffer.h>
|
#include <LibJS/Runtime/ArrayBuffer.h>
|
||||||
#include <LibJS/Runtime/FunctionObject.h>
|
#include <LibJS/Runtime/FunctionObject.h>
|
||||||
#include <LibProtocol/WebSocket.h>
|
|
||||||
#include <LibProtocol/WebSocketClient.h>
|
|
||||||
#include <LibWeb/Bindings/EventWrapper.h>
|
#include <LibWeb/Bindings/EventWrapper.h>
|
||||||
#include <LibWeb/Bindings/WebSocketWrapper.h>
|
#include <LibWeb/Bindings/WebSocketWrapper.h>
|
||||||
#include <LibWeb/DOM/DOMException.h>
|
#include <LibWeb/DOM/DOMException.h>
|
||||||
|
@ -28,29 +26,27 @@
|
||||||
|
|
||||||
namespace Web::WebSockets {
|
namespace Web::WebSockets {
|
||||||
|
|
||||||
|
static RefPtr<WebSocketClientManager> s_websocket_client_manager;
|
||||||
|
|
||||||
|
void WebSocketClientManager::initialize(RefPtr<WebSocketClientManager> websocket_client_manager)
|
||||||
|
{
|
||||||
|
s_websocket_client_manager = websocket_client_manager;
|
||||||
|
}
|
||||||
|
|
||||||
WebSocketClientManager& WebSocketClientManager::the()
|
WebSocketClientManager& WebSocketClientManager::the()
|
||||||
{
|
{
|
||||||
static RefPtr<WebSocketClientManager> s_the;
|
if (!s_websocket_client_manager) [[unlikely]] {
|
||||||
if (!s_the)
|
dbgln("Web::WebSockets::WebSocketClientManager was not initialized!");
|
||||||
s_the = WebSocketClientManager::try_create().release_value_but_fixme_should_propagate_errors();
|
VERIFY_NOT_REACHED();
|
||||||
return *s_the;
|
}
|
||||||
|
return *s_websocket_client_manager;
|
||||||
}
|
}
|
||||||
|
|
||||||
ErrorOr<NonnullRefPtr<WebSocketClientManager>> WebSocketClientManager::try_create()
|
WebSocketClientSocket::WebSocketClientSocket() = default;
|
||||||
{
|
|
||||||
auto websocket_client = TRY(Protocol::WebSocketClient::try_create());
|
|
||||||
return adopt_nonnull_ref_or_enomem(new (nothrow) WebSocketClientManager(move(websocket_client)));
|
|
||||||
}
|
|
||||||
|
|
||||||
WebSocketClientManager::WebSocketClientManager(NonnullRefPtr<Protocol::WebSocketClient> websocket_client)
|
WebSocketClientSocket::~WebSocketClientSocket() = default;
|
||||||
: m_websocket_client(move(websocket_client))
|
|
||||||
{
|
|
||||||
}
|
|
||||||
|
|
||||||
RefPtr<Protocol::WebSocket> WebSocketClientManager::connect(const AK::URL& url, String const& origin)
|
WebSocketClientManager::WebSocketClientManager() = default;
|
||||||
{
|
|
||||||
return m_websocket_client->connect(url, origin);
|
|
||||||
}
|
|
||||||
|
|
||||||
// https://websockets.spec.whatwg.org/#dom-websocket-websocket
|
// https://websockets.spec.whatwg.org/#dom-websocket-websocket
|
||||||
DOM::ExceptionOr<NonnullRefPtr<WebSocket>> WebSocket::create_with_global_object(Bindings::WindowObject& window, String const& url)
|
DOM::ExceptionOr<NonnullRefPtr<WebSocket>> WebSocket::create_with_global_object(Bindings::WindowObject& window, String const& url)
|
||||||
|
@ -107,18 +103,7 @@ WebSocket::ReadyState WebSocket::ready_state() const
|
||||||
{
|
{
|
||||||
if (!m_websocket)
|
if (!m_websocket)
|
||||||
return WebSocket::ReadyState::Closed;
|
return WebSocket::ReadyState::Closed;
|
||||||
auto ready_state = const_cast<Protocol::WebSocket&>(*m_websocket).ready_state();
|
return const_cast<WebSocketClientSocket&>(*m_websocket).ready_state();
|
||||||
switch (ready_state) {
|
|
||||||
case Protocol::WebSocket::ReadyState::Connecting:
|
|
||||||
return WebSocket::ReadyState::Connecting;
|
|
||||||
case Protocol::WebSocket::ReadyState::Open:
|
|
||||||
return WebSocket::ReadyState::Open;
|
|
||||||
case Protocol::WebSocket::ReadyState::Closing:
|
|
||||||
return WebSocket::ReadyState::Closing;
|
|
||||||
case Protocol::WebSocket::ReadyState::Closed:
|
|
||||||
return WebSocket::ReadyState::Closed;
|
|
||||||
}
|
|
||||||
return WebSocket::ReadyState::Closed;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// https://websockets.spec.whatwg.org/#dom-websocket-extensions
|
// https://websockets.spec.whatwg.org/#dom-websocket-extensions
|
||||||
|
@ -212,7 +197,7 @@ void WebSocket::on_close(u16 code, String reason, bool was_clean)
|
||||||
// https://websockets.spec.whatwg.org/#feedback-from-the-protocol
|
// https://websockets.spec.whatwg.org/#feedback-from-the-protocol
|
||||||
void WebSocket::on_message(ByteBuffer message, bool is_text)
|
void WebSocket::on_message(ByteBuffer message, bool is_text)
|
||||||
{
|
{
|
||||||
if (m_websocket->ready_state() != Protocol::WebSocket::ReadyState::Open)
|
if (m_websocket->ready_state() != WebSocket::ReadyState::Open)
|
||||||
return;
|
return;
|
||||||
if (is_text) {
|
if (is_text) {
|
||||||
auto text_message = String(ReadonlyBytes(message));
|
auto text_message = String(ReadonlyBytes(message));
|
||||||
|
|
|
@ -1,5 +1,5 @@
|
||||||
/*
|
/*
|
||||||
* Copyright (c) 2021, Dex♪ <dexes.ttp@gmail.com>
|
* Copyright (c) 2021-2022, Dex♪ <dexes.ttp@gmail.com>
|
||||||
*
|
*
|
||||||
* SPDX-License-Identifier: BSD-2-Clause
|
* SPDX-License-Identifier: BSD-2-Clause
|
||||||
*/
|
*/
|
||||||
|
@ -23,26 +23,10 @@
|
||||||
E(onopen, HTML::EventNames::open) \
|
E(onopen, HTML::EventNames::open) \
|
||||||
E(onmessage, HTML::EventNames::message)
|
E(onmessage, HTML::EventNames::message)
|
||||||
|
|
||||||
namespace Protocol {
|
|
||||||
class WebSocketClient;
|
|
||||||
class WebSocket;
|
|
||||||
}
|
|
||||||
|
|
||||||
namespace Web::WebSockets {
|
namespace Web::WebSockets {
|
||||||
|
|
||||||
class WebSocketClientManager : public Core::Object {
|
class WebSocketClientSocket;
|
||||||
C_OBJECT_ABSTRACT(WebSocketClientManager)
|
class WebSocketClientManager;
|
||||||
public:
|
|
||||||
static WebSocketClientManager& the();
|
|
||||||
|
|
||||||
RefPtr<Protocol::WebSocket> connect(const AK::URL&, String const& origin);
|
|
||||||
|
|
||||||
private:
|
|
||||||
static ErrorOr<NonnullRefPtr<WebSocketClientManager>> try_create();
|
|
||||||
WebSocketClientManager(NonnullRefPtr<Protocol::WebSocketClient>);
|
|
||||||
|
|
||||||
RefPtr<Protocol::WebSocketClient> m_websocket_client;
|
|
||||||
};
|
|
||||||
|
|
||||||
class WebSocket final
|
class WebSocket final
|
||||||
: public RefCounted<WebSocket>
|
: public RefCounted<WebSocket>
|
||||||
|
@ -106,7 +90,55 @@ private:
|
||||||
|
|
||||||
AK::URL m_url;
|
AK::URL m_url;
|
||||||
String m_binary_type { "blob" };
|
String m_binary_type { "blob" };
|
||||||
RefPtr<Protocol::WebSocket> m_websocket;
|
RefPtr<WebSocketClientSocket> m_websocket;
|
||||||
|
};
|
||||||
|
|
||||||
|
class WebSocketClientSocket : public RefCounted<WebSocketClientSocket> {
|
||||||
|
public:
|
||||||
|
virtual ~WebSocketClientSocket();
|
||||||
|
|
||||||
|
struct CertificateAndKey {
|
||||||
|
String certificate;
|
||||||
|
String key;
|
||||||
|
};
|
||||||
|
|
||||||
|
struct Message {
|
||||||
|
ByteBuffer data;
|
||||||
|
bool is_text { false };
|
||||||
|
};
|
||||||
|
|
||||||
|
enum class Error {
|
||||||
|
CouldNotEstablishConnection,
|
||||||
|
ConnectionUpgradeFailed,
|
||||||
|
ServerClosedSocket,
|
||||||
|
};
|
||||||
|
|
||||||
|
virtual Web::WebSockets::WebSocket::ReadyState ready_state() = 0;
|
||||||
|
|
||||||
|
virtual void send(ByteBuffer binary_or_text_message, bool is_text) = 0;
|
||||||
|
virtual void send(StringView text_message) = 0;
|
||||||
|
virtual void close(u16 code = 1005, String reason = {}) = 0;
|
||||||
|
|
||||||
|
Function<void()> on_open;
|
||||||
|
Function<void(Message)> on_message;
|
||||||
|
Function<void(Error)> on_error;
|
||||||
|
Function<void(u16 code, String reason, bool was_clean)> on_close;
|
||||||
|
Function<CertificateAndKey()> on_certificate_requested;
|
||||||
|
|
||||||
|
protected:
|
||||||
|
explicit WebSocketClientSocket();
|
||||||
|
};
|
||||||
|
|
||||||
|
class WebSocketClientManager : public Core::Object {
|
||||||
|
C_OBJECT_ABSTRACT(WebSocketClientManager)
|
||||||
|
public:
|
||||||
|
static void initialize(RefPtr<WebSocketClientManager>);
|
||||||
|
static WebSocketClientManager& the();
|
||||||
|
|
||||||
|
virtual RefPtr<WebSocketClientSocket> connect(AK::URL const&, String const& origin) = 0;
|
||||||
|
|
||||||
|
protected:
|
||||||
|
explicit WebSocketClientManager();
|
||||||
};
|
};
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -4,6 +4,7 @@ set(SOURCES
|
||||||
OutOfProcessWebView.cpp
|
OutOfProcessWebView.cpp
|
||||||
StylePropertiesModel.cpp
|
StylePropertiesModel.cpp
|
||||||
WebContentClient.cpp
|
WebContentClient.cpp
|
||||||
|
WebSocketClientAdapter.cpp
|
||||||
)
|
)
|
||||||
|
|
||||||
set(GENERATED_SOURCES
|
set(GENERATED_SOURCES
|
||||||
|
@ -14,6 +15,6 @@ set(GENERATED_SOURCES
|
||||||
)
|
)
|
||||||
|
|
||||||
serenity_lib(LibWebView webview)
|
serenity_lib(LibWebView webview)
|
||||||
target_link_libraries(LibWebView LibGfx LibGUI LibImageDecoderClient LibIPC LibWeb)
|
target_link_libraries(LibWebView LibGfx LibGUI LibImageDecoderClient LibIPC LibProtocol LibWeb)
|
||||||
|
|
||||||
add_subdirectory(DumpLayoutTree)
|
add_subdirectory(DumpLayoutTree)
|
||||||
|
|
126
Userland/Libraries/LibWebView/WebSocketClientAdapter.cpp
Normal file
126
Userland/Libraries/LibWebView/WebSocketClientAdapter.cpp
Normal file
|
@ -0,0 +1,126 @@
|
||||||
|
/*
|
||||||
|
* Copyright (c) 2022, Dex♪ <dexes.ttp@gmail.com>
|
||||||
|
*
|
||||||
|
* SPDX-License-Identifier: BSD-2-Clause
|
||||||
|
*/
|
||||||
|
|
||||||
|
#include <LibProtocol/WebSocket.h>
|
||||||
|
#include <LibProtocol/WebSocketClient.h>
|
||||||
|
#include <LibWebView/WebSocketClientAdapter.h>
|
||||||
|
|
||||||
|
namespace WebView {
|
||||||
|
|
||||||
|
RefPtr<WebSocketClientSocketAdapter> WebSocketClientSocketAdapter::create(NonnullRefPtr<Protocol::WebSocket> websocket)
|
||||||
|
{
|
||||||
|
return adopt_ref(*new WebSocketClientSocketAdapter(move(websocket)));
|
||||||
|
}
|
||||||
|
|
||||||
|
WebSocketClientSocketAdapter::WebSocketClientSocketAdapter(NonnullRefPtr<Protocol::WebSocket> websocket)
|
||||||
|
: m_websocket(move(websocket))
|
||||||
|
{
|
||||||
|
m_websocket->on_open = [weak_this = make_weak_ptr()] {
|
||||||
|
if (auto strong_this = weak_this.strong_ref())
|
||||||
|
if (strong_this->on_open)
|
||||||
|
strong_this->on_open();
|
||||||
|
};
|
||||||
|
m_websocket->on_message = [weak_this = make_weak_ptr()](auto message) {
|
||||||
|
if (auto strong_this = weak_this.strong_ref()) {
|
||||||
|
if (strong_this->on_message) {
|
||||||
|
strong_this->on_message(Web::WebSockets::WebSocketClientSocket::Message {
|
||||||
|
.data = move(message.data),
|
||||||
|
.is_text = message.is_text,
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
};
|
||||||
|
m_websocket->on_error = [weak_this = make_weak_ptr()](auto error) {
|
||||||
|
if (auto strong_this = weak_this.strong_ref()) {
|
||||||
|
if (strong_this->on_error) {
|
||||||
|
switch (error) {
|
||||||
|
case Protocol::WebSocket::Error::CouldNotEstablishConnection:
|
||||||
|
strong_this->on_error(Web::WebSockets::WebSocketClientSocket::Error::CouldNotEstablishConnection);
|
||||||
|
return;
|
||||||
|
case Protocol::WebSocket::Error::ConnectionUpgradeFailed:
|
||||||
|
strong_this->on_error(Web::WebSockets::WebSocketClientSocket::Error::ConnectionUpgradeFailed);
|
||||||
|
return;
|
||||||
|
case Protocol::WebSocket::Error::ServerClosedSocket:
|
||||||
|
strong_this->on_error(Web::WebSockets::WebSocketClientSocket::Error::ServerClosedSocket);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
VERIFY_NOT_REACHED();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
};
|
||||||
|
m_websocket->on_close = [weak_this = make_weak_ptr()](u16 code, String reason, bool was_clean) {
|
||||||
|
if (auto strong_this = weak_this.strong_ref())
|
||||||
|
if (strong_this->on_close)
|
||||||
|
strong_this->on_close(code, move(reason), was_clean);
|
||||||
|
};
|
||||||
|
m_websocket->on_certificate_requested = [weak_this = make_weak_ptr()] {
|
||||||
|
if (auto strong_this = weak_this.strong_ref()) {
|
||||||
|
if (strong_this->on_certificate_requested) {
|
||||||
|
auto certificate_and_key = weak_this->on_certificate_requested();
|
||||||
|
return Protocol::WebSocket::CertificateAndKey {
|
||||||
|
.certificate = move(certificate_and_key.certificate),
|
||||||
|
.key = move(certificate_and_key.key),
|
||||||
|
};
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return Protocol::WebSocket::CertificateAndKey {};
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
WebSocketClientSocketAdapter::~WebSocketClientSocketAdapter() = default;
|
||||||
|
|
||||||
|
Web::WebSockets::WebSocket::ReadyState WebSocketClientSocketAdapter::ready_state()
|
||||||
|
{
|
||||||
|
switch (m_websocket->ready_state()) {
|
||||||
|
case Protocol::WebSocket::ReadyState::Connecting:
|
||||||
|
return Web::WebSockets::WebSocket::ReadyState::Connecting;
|
||||||
|
case Protocol::WebSocket::ReadyState::Open:
|
||||||
|
return Web::WebSockets::WebSocket::ReadyState::Open;
|
||||||
|
case Protocol::WebSocket::ReadyState::Closing:
|
||||||
|
return Web::WebSockets::WebSocket::ReadyState::Closing;
|
||||||
|
case Protocol::WebSocket::ReadyState::Closed:
|
||||||
|
return Web::WebSockets::WebSocket::ReadyState::Closed;
|
||||||
|
}
|
||||||
|
VERIFY_NOT_REACHED();
|
||||||
|
}
|
||||||
|
|
||||||
|
void WebSocketClientSocketAdapter::send(ByteBuffer binary_or_text_message, bool is_text)
|
||||||
|
{
|
||||||
|
m_websocket->send(binary_or_text_message, is_text);
|
||||||
|
}
|
||||||
|
|
||||||
|
void WebSocketClientSocketAdapter::send(StringView text_message)
|
||||||
|
{
|
||||||
|
m_websocket->send(text_message);
|
||||||
|
}
|
||||||
|
|
||||||
|
void WebSocketClientSocketAdapter::close(u16 code, String reason)
|
||||||
|
{
|
||||||
|
m_websocket->close(code, reason);
|
||||||
|
}
|
||||||
|
|
||||||
|
ErrorOr<NonnullRefPtr<WebSocketClientManagerAdapter>> WebSocketClientManagerAdapter::try_create()
|
||||||
|
{
|
||||||
|
auto websocket_client = TRY(Protocol::WebSocketClient::try_create());
|
||||||
|
return adopt_nonnull_ref_or_enomem(new (nothrow) WebSocketClientManagerAdapter(move(websocket_client)));
|
||||||
|
}
|
||||||
|
|
||||||
|
WebSocketClientManagerAdapter::WebSocketClientManagerAdapter(NonnullRefPtr<Protocol::WebSocketClient> websocket_client)
|
||||||
|
: m_websocket_client(move(websocket_client))
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
WebSocketClientManagerAdapter::~WebSocketClientManagerAdapter() = default;
|
||||||
|
|
||||||
|
RefPtr<Web::WebSockets::WebSocketClientSocket> WebSocketClientManagerAdapter::connect(const AK::URL& url, String const& origin)
|
||||||
|
{
|
||||||
|
auto underlying_websocket = m_websocket_client->connect(url, origin);
|
||||||
|
if (!underlying_websocket)
|
||||||
|
return {};
|
||||||
|
return WebSocketClientSocketAdapter::create(underlying_websocket.release_nonnull());
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
54
Userland/Libraries/LibWebView/WebSocketClientAdapter.h
Normal file
54
Userland/Libraries/LibWebView/WebSocketClientAdapter.h
Normal file
|
@ -0,0 +1,54 @@
|
||||||
|
/*
|
||||||
|
* Copyright (c) 2022, Dex♪ <dexes.ttp@gmail.com>
|
||||||
|
*
|
||||||
|
* SPDX-License-Identifier: BSD-2-Clause
|
||||||
|
*/
|
||||||
|
|
||||||
|
#pragma once
|
||||||
|
|
||||||
|
#include <AK/Error.h>
|
||||||
|
#include <AK/NonnullRefPtr.h>
|
||||||
|
#include <AK/Weakable.h>
|
||||||
|
#include <LibWeb/WebSockets/WebSocket.h>
|
||||||
|
|
||||||
|
namespace Protocol {
|
||||||
|
class WebSocket;
|
||||||
|
class WebSocketClient;
|
||||||
|
};
|
||||||
|
|
||||||
|
namespace WebView {
|
||||||
|
|
||||||
|
class WebSocketClientSocketAdapter
|
||||||
|
: public Web::WebSockets::WebSocketClientSocket
|
||||||
|
, public Weakable<WebSocketClientSocketAdapter> {
|
||||||
|
public:
|
||||||
|
static RefPtr<WebSocketClientSocketAdapter> create(NonnullRefPtr<Protocol::WebSocket>);
|
||||||
|
virtual ~WebSocketClientSocketAdapter() override;
|
||||||
|
|
||||||
|
virtual Web::WebSockets::WebSocket::ReadyState ready_state() override;
|
||||||
|
|
||||||
|
virtual void send(ByteBuffer binary_or_text_message, bool is_text) override;
|
||||||
|
virtual void send(StringView text_message) override;
|
||||||
|
virtual void close(u16 code = 1005, String reason = {}) override;
|
||||||
|
|
||||||
|
private:
|
||||||
|
WebSocketClientSocketAdapter(NonnullRefPtr<Protocol::WebSocket>);
|
||||||
|
|
||||||
|
NonnullRefPtr<Protocol::WebSocket> m_websocket;
|
||||||
|
};
|
||||||
|
|
||||||
|
class WebSocketClientManagerAdapter : public Web::WebSockets::WebSocketClientManager {
|
||||||
|
public:
|
||||||
|
static ErrorOr<NonnullRefPtr<WebSocketClientManagerAdapter>> try_create();
|
||||||
|
|
||||||
|
virtual ~WebSocketClientManagerAdapter() override;
|
||||||
|
|
||||||
|
virtual RefPtr<Web::WebSockets::WebSocketClientSocket> connect(const AK::URL&, String const& origin) override;
|
||||||
|
|
||||||
|
private:
|
||||||
|
WebSocketClientManagerAdapter(NonnullRefPtr<Protocol::WebSocketClient>);
|
||||||
|
|
||||||
|
NonnullRefPtr<Protocol::WebSocketClient> m_websocket_client;
|
||||||
|
};
|
||||||
|
|
||||||
|
}
|
|
@ -10,7 +10,9 @@
|
||||||
#include <LibIPC/SingleServer.h>
|
#include <LibIPC/SingleServer.h>
|
||||||
#include <LibMain/Main.h>
|
#include <LibMain/Main.h>
|
||||||
#include <LibWeb/ImageDecoding.h>
|
#include <LibWeb/ImageDecoding.h>
|
||||||
|
#include <LibWeb/WebSockets/WebSocket.h>
|
||||||
#include <LibWebView/ImageDecoderClientAdapter.h>
|
#include <LibWebView/ImageDecoderClientAdapter.h>
|
||||||
|
#include <LibWebView/WebSocketClientAdapter.h>
|
||||||
#include <WebContent/ConnectionFromClient.h>
|
#include <WebContent/ConnectionFromClient.h>
|
||||||
|
|
||||||
ErrorOr<int> serenity_main(Main::Arguments)
|
ErrorOr<int> serenity_main(Main::Arguments)
|
||||||
|
@ -25,6 +27,7 @@ ErrorOr<int> serenity_main(Main::Arguments)
|
||||||
TRY(Core::System::unveil(nullptr, nullptr));
|
TRY(Core::System::unveil(nullptr, nullptr));
|
||||||
|
|
||||||
Web::ImageDecoding::Decoder::initialize(WebView::ImageDecoderClientAdapter::create());
|
Web::ImageDecoding::Decoder::initialize(WebView::ImageDecoderClientAdapter::create());
|
||||||
|
Web::WebSockets::WebSocketClientManager::initialize(TRY(WebView::WebSocketClientManagerAdapter::try_create()));
|
||||||
|
|
||||||
auto client = TRY(IPC::take_over_accepted_client_from_system_server<WebContent::ConnectionFromClient>());
|
auto client = TRY(IPC::take_over_accepted_client_from_system_server<WebContent::ConnectionFromClient>());
|
||||||
return event_loop.exec();
|
return event_loop.exec();
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue