/* * Copyright (c) 2023, Andrew Kaster * * SPDX-License-Identifier: BSD-2-Clause */ #include "WebContentService.h" #include "LadybirdServiceBase.h" #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include static ErrorOr> bind_request_server_service() { return bind_service(&bind_request_server_java); } template ErrorOr, Error> bind_service(void (*)(int, int)); static ErrorOr load_content_filters(); static ErrorOr load_autoplay_allowlist(); ErrorOr service_main(int ipc_socket, int fd_passing_socket) { Core::EventLoop event_loop; Web::Platform::EventLoopPlugin::install(*new Web::Platform::EventLoopPluginSerenity); Web::Platform::ImageCodecPlugin::install(*new Ladybird::ImageCodecPlugin); Web::Platform::AudioCodecPlugin::install_creation_hook([](auto loader) { (void)loader; return Error::from_string_literal("Don't know how to initialize audio in this configuration!"); }); auto request_server_client = TRY(bind_request_server_service()); Web::ResourceLoader::initialize(TRY(WebView::RequestServerAdapter::try_create(move(request_server_client)))); bool is_layout_test_mode = false; Web::HTML::Window::set_internals_object_exposed(is_layout_test_mode); Web::Platform::FontPlugin::install(*new Ladybird::FontPlugin(is_layout_test_mode)); TRY(Web::Bindings::initialize_main_thread_vm()); auto maybe_content_filter_error = load_content_filters(); if (maybe_content_filter_error.is_error()) dbgln("Failed to load content filters: {}", maybe_content_filter_error.error()); auto maybe_autoplay_allowlist_error = load_autoplay_allowlist(); if (maybe_autoplay_allowlist_error.is_error()) dbgln("Failed to load autoplay allowlist: {}", maybe_autoplay_allowlist_error.error()); auto webcontent_socket = TRY(Core::LocalSocket::adopt_fd(ipc_socket)); auto webcontent_client = TRY(WebContent::ConnectionFromClient::try_create(move(webcontent_socket))); webcontent_client->set_fd_passing_socket(TRY(Core::LocalSocket::adopt_fd(fd_passing_socket))); return event_loop.exec(); } template ErrorOr> bind_service(void (*bind_method)(int, int)) { int socket_fds[2] {}; TRY(Core::System::socketpair(AF_LOCAL, SOCK_STREAM, 0, socket_fds)); int ui_fd = socket_fds[0]; int server_fd = socket_fds[1]; int fd_passing_socket_fds[2] {}; TRY(Core::System::socketpair(AF_LOCAL, SOCK_STREAM, 0, fd_passing_socket_fds)); int ui_fd_passing_fd = fd_passing_socket_fds[0]; int server_fd_passing_fd = fd_passing_socket_fds[1]; // NOTE: The java object takes ownership of the socket fds (*bind_method)(server_fd, server_fd_passing_fd); auto socket = TRY(Core::LocalSocket::adopt_fd(ui_fd)); TRY(socket->set_blocking(true)); auto new_client = TRY(try_make_ref_counted(move(socket))); new_client->set_fd_passing_socket(TRY(Core::LocalSocket::adopt_fd(ui_fd_passing_fd))); return new_client; } static ErrorOr load_content_filters() { auto file_or_error = Core::File::open(ByteString::formatted("{}/home/anon/.config/BrowserContentFilters.txt", s_serenity_resource_root), Core::File::OpenMode::Read); if (file_or_error.is_error()) file_or_error = Core::File::open(ByteString::formatted("{}/res/ladybird/BrowserContentFilters.txt", s_serenity_resource_root), Core::File::OpenMode::Read); if (file_or_error.is_error()) return file_or_error.release_error(); auto file = file_or_error.release_value(); auto ad_filter_list = TRY(Core::InputBufferedFile::create(move(file))); auto buffer = TRY(ByteBuffer::create_uninitialized(4096)); Vector patterns; while (TRY(ad_filter_list->can_read_line())) { auto line = TRY(ad_filter_list->read_line(buffer)); if (line.is_empty()) continue; auto pattern = TRY(String::from_utf8(line)); TRY(patterns.try_append(move(pattern))); } auto& content_filter = Web::ContentFilter::the(); TRY(content_filter.set_patterns(patterns)); return {}; } static ErrorOr load_autoplay_allowlist() { auto file_or_error = Core::File::open(TRY(String::formatted("{}/home/anon/.config/BrowserAutoplayAllowlist.txt", s_serenity_resource_root)), Core::File::OpenMode::Read); if (file_or_error.is_error()) file_or_error = Core::File::open(TRY(String::formatted("{}/res/ladybird/BrowserAutoplayAllowlist.txt", s_serenity_resource_root)), Core::File::OpenMode::Read); if (file_or_error.is_error()) return file_or_error.release_error(); auto file = file_or_error.release_value(); auto allowlist = TRY(Core::InputBufferedFile::create(move(file))); auto buffer = TRY(ByteBuffer::create_uninitialized(4096)); Vector origins; while (TRY(allowlist->can_read_line())) { auto line = TRY(allowlist->read_line(buffer)); if (line.is_empty()) continue; auto domain = TRY(String::from_utf8(line)); TRY(origins.try_append(move(domain))); } auto& autoplay_allowlist = Web::PermissionsPolicy::AutoplayAllowlist::the(); TRY(autoplay_allowlist.enable_for_origins(origins)); return {}; }