// Copyright 2015 The Chromium Authors. All rights reserved. // Use of this source code is governed by a BSD-style license that can be // found in the LICENSE file. #include "headless/lib/browser/headless_devtools.h" #include #include #include "base/files/file_path.h" #include "base/memory/ptr_util.h" #include "content/public/browser/devtools_agent_host.h" #include "content/public/browser/devtools_socket_factory.h" #include "content/public/browser/navigation_entry.h" #include "headless/grit/headless_lib_resources.h" #include "headless/public/headless_browser.h" #include "net/base/net_errors.h" #include "net/log/net_log_source.h" #include "net/socket/tcp_server_socket.h" #include "ui/base/resource/resource_bundle.h" namespace headless { namespace { const int kBackLog = 10; class TCPEndpointServerSocketFactory : public content::DevToolsSocketFactory { public: explicit TCPEndpointServerSocketFactory(const net::IPEndPoint& endpoint) : endpoint_(endpoint) { DCHECK(endpoint_.address().IsValid()); } private: std::unique_ptr CreateForHttpServer() override { std::unique_ptr socket( new net::TCPServerSocket(nullptr, net::NetLogSource())); if (socket->Listen(endpoint_, kBackLog) != net::OK) return std::unique_ptr(); return socket; } std::unique_ptr CreateForTethering( std::string* out_name) override { return nullptr; } net::IPEndPoint endpoint_; DISALLOW_COPY_AND_ASSIGN(TCPEndpointServerSocketFactory); }; #if defined(OS_POSIX) class TCPAdoptServerSocketFactory : public content::DevToolsSocketFactory { public: // Construct a factory to use an already-open, already-listening socket. explicit TCPAdoptServerSocketFactory(const size_t socket_fd) : socket_fd_(socket_fd) {} private: std::unique_ptr CreateForHttpServer() override { std::unique_ptr tsock( new net::TCPServerSocket(nullptr, net::NetLogSource())); if (tsock->AdoptSocket(socket_fd_) != net::OK) { LOG(ERROR) << "Failed to adopt open socket"; return std::unique_ptr(); } // Note that we assume that the socket is already listening, so unlike // TCPEndpointServerSocketFactory, we don't call Listen. return std::unique_ptr(std::move(tsock)); } std::unique_ptr CreateForTethering( std::string* out_name) override { return nullptr; } size_t socket_fd_; DISALLOW_COPY_AND_ASSIGN(TCPAdoptServerSocketFactory); }; #else // defined(OS_POSIX) // Placeholder class to use when a socket_fd is passed in on non-Posix. class DummyTCPServerSocketFactory : public content::DevToolsSocketFactory { public: explicit DummyTCPServerSocketFactory() {} private: std::unique_ptr CreateForHttpServer() override { return nullptr; } std::unique_ptr CreateForTethering( std::string* out_name) override { return nullptr; } DISALLOW_COPY_AND_ASSIGN(DummyTCPServerSocketFactory); }; #endif // defined(OS_POSIX) } // namespace void StartLocalDevToolsHttpHandler(HeadlessBrowser::Options* options) { std::unique_ptr socket_factory; if (options->devtools_socket_fd == 0) { const net::IPEndPoint& endpoint = options->devtools_endpoint; socket_factory.reset(new TCPEndpointServerSocketFactory(endpoint)); } else { #if defined(OS_POSIX) const uint16_t socket_fd = options->devtools_socket_fd; socket_factory.reset(new TCPAdoptServerSocketFactory(socket_fd)); #else LOG(ERROR) << "Can't inherit an open socket on non-Posix systems"; socket_factory.reset(new DummyTCPServerSocketFactory()); #endif } content::DevToolsAgentHost::StartRemoteDebuggingServer( std::move(socket_factory), std::string(), options->user_data_dir, // TODO(altimin): Figure a proper value for this. base::FilePath(), options->product_name_and_version, options->user_agent); } void StopLocalDevToolsHttpHandler() { content::DevToolsAgentHost::StopRemoteDebuggingServer(); } } // namespace headless