/* * Copyright (c) 2022-2024, Tim Flynn * * SPDX-License-Identifier: BSD-2-Clause */ #include #include #include #include #include #include #include #include #include #include #include #include static Vector certificates; static ErrorOr launch_process(StringView application, ReadonlySpan arguments) { auto paths = TRY(WebView::get_paths_for_helper_process(application)); ErrorOr result = Error::from_string_literal("All paths failed to launch"); for (auto const& path : paths) { auto path_view = path.view(); result = Core::Process::spawn(path_view, arguments, {}, Core::Process::KeepAsChild::Yes); if (!result.is_error()) break; } return result; } static Vector create_arguments(ByteString const& socket_path, bool force_cpu_painting) { Vector arguments { "--webdriver-content-path"sv, socket_path, }; Vector certificate_args; for (auto const& certificate : certificates) { certificate_args.append(ByteString::formatted("--certificate={}", certificate)); arguments.append(certificate_args.last().view().characters_without_null_termination()); } arguments.append("--allow-popups"sv); arguments.append("--force-new-process"sv); arguments.append("--enable-autoplay"sv); if (force_cpu_painting) arguments.append("--force-cpu-painting"sv); arguments.append("about:blank"sv); return arguments; } static ErrorOr launch_browser(ByteString const& socket_path, bool force_cpu_painting) { auto arguments = create_arguments(socket_path, force_cpu_painting); return launch_process("Ladybird"sv, arguments.span()); } static ErrorOr launch_headless_browser(ByteString const& socket_path, bool force_cpu_painting) { auto arguments = create_arguments(socket_path, force_cpu_painting); return launch_process("headless-browser"sv, arguments.span()); } ErrorOr serenity_main(Main::Arguments arguments) { AK::set_rich_debug_enabled(true); auto listen_address = "0.0.0.0"sv; int port = 8000; bool force_cpu_painting = false; bool headless = false; Core::ArgsParser args_parser; args_parser.add_option(listen_address, "IP address to listen on", "listen-address", 'l', "listen_address"); args_parser.add_option(port, "Port to listen on", "port", 'p', "port"); args_parser.add_option(certificates, "Path to a certificate file", "certificate", 'C', "certificate"); args_parser.add_option(force_cpu_painting, "Launch browser with GPU painting disabled", "force-cpu-painting"); args_parser.add_option(headless, "Launch browser without a graphical interface", "headless"); args_parser.parse(arguments); auto ipv4_address = IPv4Address::from_string(listen_address); if (!ipv4_address.has_value()) { warnln("Invalid listen address: {}", listen_address); return 1; } if ((u16)port != port) { warnln("Invalid port number: {}", port); return 1; } WebView::platform_init(); Web::WebDriver::set_default_interface_mode(headless ? Web::WebDriver::InterfaceMode::Headless : Web::WebDriver::InterfaceMode::Graphical); auto webdriver_socket_path = ByteString::formatted("{}/webdriver", TRY(Core::StandardPaths::runtime_directory())); TRY(Core::Directory::create(webdriver_socket_path, Core::Directory::CreateDirectories::Yes)); Core::EventLoop loop; auto server = TRY(Core::TCPServer::try_create()); // FIXME: Propagate errors server->on_ready_to_accept = [&] { auto maybe_client_socket = server->accept(); if (maybe_client_socket.is_error()) { warnln("Failed to accept the client: {}", maybe_client_socket.error()); return; } auto maybe_buffered_socket = Core::BufferedTCPSocket::create(maybe_client_socket.release_value()); if (maybe_buffered_socket.is_error()) { warnln("Could not obtain a buffered socket for the client: {}", maybe_buffered_socket.error()); return; } auto launch_browser_callback = [&](ByteString const& socket_path) { return launch_browser(socket_path, force_cpu_painting); }; auto launch_headless_browser_callback = [&](ByteString const& socket_path) { return launch_headless_browser(socket_path, force_cpu_painting); }; auto maybe_client = WebDriver::Client::try_create(maybe_buffered_socket.release_value(), { move(launch_browser_callback), move(launch_headless_browser_callback) }, server); if (maybe_client.is_error()) { warnln("Could not create a WebDriver client: {}", maybe_client.error()); return; } }; TRY(server->listen(ipv4_address.value(), port, Core::TCPServer::AllowAddressReuse::Yes)); outln("Listening on {}:{}", ipv4_address.value(), port); return loop.exec(); }