// Copyright (c) 2012 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 "libcef/browser/devtools_delegate.h" #include "libcef/browser/devtools_scheme_handler.h" #include "libcef/common/content_client.h" #include #include #include "base/command_line.h" #include "base/md5.h" #include "base/rand_util.h" #include "base/strings/stringprintf.h" #include "base/strings/string_number_conversions.h" #include "base/time.h" #include "content/public/browser/devtools_http_handler.h" #include "content/public/browser/render_process_host.h" #include "content/public/browser/render_view_host.h" #include "content/public/common/content_switches.h" #include "content/public/common/url_constants.h" #include "grit/cef_resources.h" #include "net/socket/tcp_listen_socket.h" #include "ui/base/layout.h" #include "ui/base/resource/resource_bundle.h" // CefDevToolsBindingHandler CefDevToolsBindingHandler::CefDevToolsBindingHandler() { } std::string CefDevToolsBindingHandler::GetIdentifier( content::DevToolsAgentHost* agent_host) { GarbageCollect(); const std::string& identifier = GetIdentifier(agent_host->GetRenderViewHost()); agents_map_[identifier] = agent_host; return identifier; } content::DevToolsAgentHost* CefDevToolsBindingHandler::ForIdentifier( const std::string& identifier) { GarbageCollect(); // Return the existing agent host, if any. AgentsMap::const_iterator it = agents_map_.find(identifier); if (it != agents_map_.end()) return it->second; // Iterate through the existing RVH instances to find a match. for (content::RenderProcessHost::iterator it( content::RenderProcessHost::AllHostsIterator()); !it.IsAtEnd(); it.Advance()) { content::RenderProcessHost* render_process_host = it.GetCurrentValue(); DCHECK(render_process_host); // Ignore processes that don't have a connection, such as crashed contents. if (!render_process_host->HasConnection()) continue; content::RenderWidgetHost::List widgets = content::RenderWidgetHost::GetRenderWidgetHosts(); for (size_t i = 0; i < widgets.size(); ++i) { if (!widgets[i]->IsRenderView()) continue; content::RenderViewHost* host = content::RenderViewHost::From(widgets[i]); if (GetIdentifier(host) == identifier) { // May create a new agent host. scoped_refptr agent_host( content::DevToolsAgentHost::GetOrCreateFor(host)); agents_map_[identifier] = agent_host; return agent_host; } } } return NULL; } std::string CefDevToolsBindingHandler::GetIdentifier( content::RenderViewHost* rvh) { int process_id = rvh->GetProcess()->GetID(); int routing_id = rvh->GetRoutingID(); if (random_seed_.empty()) { // Generate a random seed that is used to make identifier guessing more // difficult. random_seed_ = base::StringPrintf("%lf|%u", base::Time::Now().ToDoubleT(), base::RandInt(0, INT_MAX)); } // Create a key that combines RVH IDs and the random seed. std::string key = base::StringPrintf("%d|%d|%s", process_id, routing_id, random_seed_.c_str()); // Return an MD5 hash of the key. return base::MD5String(key); } void CefDevToolsBindingHandler::GarbageCollect() { AgentsMap::iterator it = agents_map_.begin(); while (it != agents_map_.end()) { if (!it->second->GetRenderViewHost()) agents_map_.erase(it++); else ++it; } } // CefDevToolsDelegate CefDevToolsDelegate::CefDevToolsDelegate(int port) { devtools_http_handler_ = content::DevToolsHttpHandler::Start( new net::TCPListenSocketFactory("127.0.0.1", port), "", this); binding_.reset(new CefDevToolsBindingHandler()); devtools_http_handler_->SetDevToolsAgentHostBinding(binding_.get()); } CefDevToolsDelegate::~CefDevToolsDelegate() { } void CefDevToolsDelegate::Stop() { // The call below destroys this. devtools_http_handler_->Stop(); } std::string CefDevToolsDelegate::GetDiscoveryPageHTML() { return CefContentClient::Get()->GetDataResource( IDR_CEF_DEVTOOLS_DISCOVERY_PAGE, ui::SCALE_FACTOR_NONE).as_string(); } bool CefDevToolsDelegate::BundlesFrontendResources() { return true; } base::FilePath CefDevToolsDelegate::GetDebugFrontendDir() { return base::FilePath(); } std::string CefDevToolsDelegate::GetPageThumbnailData(const GURL& url) { return std::string(); } content::RenderViewHost* CefDevToolsDelegate::CreateNewTarget() { return NULL; } content::DevToolsHttpHandlerDelegate::TargetType CefDevToolsDelegate::GetTargetType(content::RenderViewHost*) { return kTargetTypeTab; } std::string CefDevToolsDelegate::GetViewDescription(content::RenderViewHost*) { return std::string(); } scoped_refptr CefDevToolsDelegate::CreateSocketForTethering( net::StreamListenSocket::Delegate* delegate, std::string* name) { return NULL; } std::string CefDevToolsDelegate::GetDevToolsURL(content::RenderViewHost* rvh, bool http_scheme) { const CommandLine& command_line = *CommandLine::ForCurrentProcess(); std::string port_str = command_line.GetSwitchValueASCII(switches::kRemoteDebuggingPort); DCHECK(!port_str.empty()); int port; if (!base::StringToInt(port_str, &port)) return std::string(); std::string page_id = binding_->GetIdentifier(rvh); std::string host = http_scheme ? base::StringPrintf("http://localhost:%d/devtools/", port) : base::StringPrintf("%s://%s/", chrome::kChromeDevToolsScheme, scheme::kChromeDevToolsHost); return base::StringPrintf( "%sdevtools.html?ws=localhost:%d/devtools/page/%s", host.c_str(), port, page_id.c_str()); }