| OLD | NEW |
| 1 // Copyright 2016 The Chromium Authors. All rights reserved. | 1 // Copyright 2016 The Chromium Authors. All rights reserved. |
| 2 // Use of this source code is governed by a BSD-style license that can be | 2 // Use of this source code is governed by a BSD-style license that can be |
| 3 // found in the LICENSE file. | 3 // found in the LICENSE file. |
| 4 | 4 |
| 5 #include "components/ui_devtools/devtools_server.h" | 5 #include "components/ui_devtools/devtools_server.h" |
| 6 | 6 |
| 7 #include "base/bind.h" | 7 #include "base/bind.h" |
| 8 #include "base/bind_helpers.h" | 8 #include "base/bind_helpers.h" |
| 9 #include "base/command_line.h" | 9 #include "base/command_line.h" |
| 10 #include "base/format_macros.h" | 10 #include "base/format_macros.h" |
| 11 #include "base/memory/ptr_util.h" | 11 #include "base/memory/ptr_util.h" |
| 12 #include "base/strings/string_number_conversions.h" | 12 #include "base/strings/string_number_conversions.h" |
| 13 #include "base/strings/stringprintf.h" | 13 #include "base/strings/stringprintf.h" |
| 14 #include "base/values.h" | 14 #include "base/values.h" |
| 15 #include "components/ui_devtools/switches.h" | 15 #include "components/ui_devtools/switches.h" |
| 16 #include "net/base/net_errors.h" | 16 #include "net/base/net_errors.h" |
| 17 #include "net/log/net_log.h" | 17 #include "net/log/net_log.h" |
| 18 #include "net/server/http_server_request_info.h" | 18 #include "net/server/http_server_request_info.h" |
| 19 #include "net/socket/server_socket.h" | 19 #include "net/socket/server_socket.h" |
| 20 #include "net/socket/tcp_server_socket.h" | 20 #include "net/socket/tcp_server_socket.h" |
| 21 | 21 |
| 22 namespace ui { | 22 namespace ui { |
| 23 namespace devtools { | 23 namespace devtools { |
| 24 | 24 |
| 25 namespace { | 25 namespace { |
| 26 const char kChromeDeveloperToolsPrefix[] = | 26 const char kChromeDeveloperToolsPrefix[] = |
| 27 "chrome-devtools://devtools/bundled/inspector.html?ws="; | 27 "chrome-devtools://devtools/bundled/inspector.html?ws="; |
| 28 } // namespace | 28 } // namespace |
| 29 | 29 |
| 30 UiDevToolsServer::UiDevToolsServer() | 30 UiDevToolsServer::UiDevToolsServer( |
| 31 : thread_(new base::Thread("UiDevToolsServerThread")) {} | 31 scoped_refptr<base::SingleThreadTaskRunner> task_runner) |
| 32 : task_runner_(task_runner) { |
| 33 if (task_runner_) |
| 34 return; |
| 35 // If task_runner not passed in, create an I/O thread the server can run on |
| 36 thread_.reset(new base::Thread("UiDevToolsServerThread")); |
| 37 base::Thread::Options options; |
| 38 options.message_loop_type = base::MessageLoop::TYPE_IO; |
| 39 CHECK(thread_->StartWithOptions(options)); |
| 40 task_runner_ = thread_->task_runner(); |
| 41 } |
| 32 | 42 |
| 33 UiDevToolsServer::~UiDevToolsServer() {} | 43 UiDevToolsServer::~UiDevToolsServer() {} |
| 34 | 44 |
| 35 // static | 45 // static |
| 36 std::unique_ptr<UiDevToolsServer> UiDevToolsServer::Create() { | 46 std::unique_ptr<UiDevToolsServer> UiDevToolsServer::Create( |
| 47 scoped_refptr<base::SingleThreadTaskRunner> task_runner) { |
| 37 std::unique_ptr<UiDevToolsServer> server; | 48 std::unique_ptr<UiDevToolsServer> server; |
| 38 if (base::CommandLine::ForCurrentProcess()->HasSwitch(kEnableUiDevTools)) { | 49 if (base::CommandLine::ForCurrentProcess()->HasSwitch(kEnableUiDevTools)) { |
| 39 // TODO(mhashmi): Change port if more than one inspectable clients | 50 // TODO(mhashmi): Change port if more than one inspectable clients |
| 40 int port = 9223; // Default port is 9223 | 51 int port = 9223; // Default port is 9223 |
| 41 base::StringToInt( | 52 base::StringToInt( |
| 42 base::CommandLine::ForCurrentProcess()->GetSwitchValueASCII( | 53 base::CommandLine::ForCurrentProcess()->GetSwitchValueASCII( |
| 43 kEnableUiDevTools), | 54 kEnableUiDevTools), |
| 44 &port); | 55 &port); |
| 45 server.reset(new UiDevToolsServer()); | 56 server.reset(new UiDevToolsServer(task_runner)); |
| 46 server->Start("127.0.0.1", port); | 57 server->Start("127.0.0.1", port); |
| 47 } | 58 } |
| 48 return server; | 59 return server; |
| 49 } | 60 } |
| 50 | 61 |
| 51 void UiDevToolsServer::AttachClient(std::unique_ptr<UiDevToolsClient> client) { | 62 void UiDevToolsServer::AttachClient(std::unique_ptr<UiDevToolsClient> client) { |
| 52 clients_.push_back(std::move(client)); | 63 clients_.push_back(std::move(client)); |
| 53 } | 64 } |
| 54 | 65 |
| 55 void UiDevToolsServer::SendOverWebSocket(int connection_id, | 66 void UiDevToolsServer::SendOverWebSocket(int connection_id, |
| 56 const String& message) { | 67 const String& message) { |
| 57 thread_->task_runner()->PostTask( | 68 task_runner_->PostTask( |
| 58 FROM_HERE, | 69 FROM_HERE, |
| 59 base::Bind(&net::HttpServer::SendOverWebSocket, | 70 base::Bind(&net::HttpServer::SendOverWebSocket, |
| 60 base::Unretained(server_.get()), connection_id, message)); | 71 base::Unretained(server_.get()), connection_id, message)); |
| 61 } | 72 } |
| 62 | 73 |
| 63 void UiDevToolsServer::Start(const std::string& address_string, uint16_t port) { | 74 void UiDevToolsServer::Start(const std::string& address_string, uint16_t port) { |
| 64 if (thread_ && thread_->IsRunning()) | 75 task_runner_->PostTask( |
| 65 return; | 76 FROM_HERE, base::Bind(&UiDevToolsServer::StartServer, |
| 66 | 77 base::Unretained(this), address_string, port)); |
| 67 // Start IO thread upon which all the methods will run | |
| 68 base::Thread::Options options; | |
| 69 options.message_loop_type = base::MessageLoop::TYPE_IO; | |
| 70 if (thread_->StartWithOptions(options)) { | |
| 71 thread_->task_runner()->PostTask( | |
| 72 FROM_HERE, base::Bind(&UiDevToolsServer::StartServer, | |
| 73 base::Unretained(this), address_string, port)); | |
| 74 } | |
| 75 } | 78 } |
| 76 | 79 |
| 77 void UiDevToolsServer::StartServer(const std::string& address_string, | 80 void UiDevToolsServer::StartServer(const std::string& address_string, |
| 78 uint16_t port) { | 81 uint16_t port) { |
| 82 DCHECK(!server_); |
| 79 std::unique_ptr<net::ServerSocket> socket( | 83 std::unique_ptr<net::ServerSocket> socket( |
| 80 new net::TCPServerSocket(nullptr, net::NetLogSource())); | 84 new net::TCPServerSocket(nullptr, net::NetLogSource())); |
| 81 constexpr int kBacklog = 1; | 85 constexpr int kBacklog = 1; |
| 82 if (socket->ListenWithAddressAndPort(address_string, port, kBacklog) != | 86 if (socket->ListenWithAddressAndPort(address_string, port, kBacklog) != |
| 83 net::OK) | 87 net::OK) |
| 84 return; | 88 return; |
| 85 server_ = base::MakeUnique<net::HttpServer>(std::move(socket), this); | 89 server_ = base::MakeUnique<net::HttpServer>(std::move(socket), this); |
| 86 } | 90 } |
| 87 | 91 |
| 88 // HttpServer::Delegate Implementation | 92 // HttpServer::Delegate Implementation |
| (...skipping 13 matching lines...) Expand all Loading... |
| 102 "them:</h3>"; | 106 "them:</h3>"; |
| 103 net::IPEndPoint ip; | 107 net::IPEndPoint ip; |
| 104 server_->GetLocalAddress(&ip); | 108 server_->GetLocalAddress(&ip); |
| 105 for (ClientsList::size_type i = 0; i != clients_.size(); i++) { | 109 for (ClientsList::size_type i = 0; i != clients_.size(); i++) { |
| 106 clientHTML += base::StringPrintf( | 110 clientHTML += base::StringPrintf( |
| 107 "<p><strong>%s</strong> (%s%s/%" PRIuS ")</p>", | 111 "<p><strong>%s</strong> (%s%s/%" PRIuS ")</p>", |
| 108 clients_[i]->name().c_str(), kChromeDeveloperToolsPrefix, | 112 clients_[i]->name().c_str(), kChromeDeveloperToolsPrefix, |
| 109 ip.ToString().c_str(), i); | 113 ip.ToString().c_str(), i); |
| 110 } | 114 } |
| 111 clientHTML += "</html>"; | 115 clientHTML += "</html>"; |
| 112 thread_->task_runner()->PostTask( | 116 task_runner_->PostTask( |
| 113 FROM_HERE, | 117 FROM_HERE, |
| 114 base::Bind(&net::HttpServer::Send200, base::Unretained(server_.get()), | 118 base::Bind(&net::HttpServer::Send200, base::Unretained(server_.get()), |
| 115 connection_id, clientHTML, "text/html")); | 119 connection_id, clientHTML, "text/html")); |
| 120 return; |
| 116 } | 121 } |
| 122 task_runner_->PostTask( |
| 123 FROM_HERE, base::Bind(&net::HttpServer::Send404, |
| 124 base::Unretained(server_.get()), connection_id)); |
| 117 } | 125 } |
| 118 | 126 |
| 119 void UiDevToolsServer::OnWebSocketRequest( | 127 void UiDevToolsServer::OnWebSocketRequest( |
| 120 int connection_id, | 128 int connection_id, |
| 121 const net::HttpServerRequestInfo& info) { | 129 const net::HttpServerRequestInfo& info) { |
| 122 size_t target_id = 0; | 130 size_t target_id = 0; |
| 123 if (info.path.empty() || | 131 if (info.path.empty() || |
| 124 !base::StringToSizeT(info.path.substr(1), &target_id) || | 132 !base::StringToSizeT(info.path.substr(1), &target_id) || |
| 125 target_id > clients_.size()) | 133 target_id > clients_.size()) |
| 126 return; | 134 return; |
| 127 | 135 |
| 128 UiDevToolsClient* client = clients_[target_id].get(); | 136 UiDevToolsClient* client = clients_[target_id].get(); |
| 129 // Only one user can inspect the client at a time | 137 // Only one user can inspect the client at a time |
| 130 if (client->connected()) | 138 if (client->connected()) |
| 131 return; | 139 return; |
| 132 client->set_connection_id(connection_id); | 140 client->set_connection_id(connection_id); |
| 133 connections_[connection_id] = client; | 141 connections_[connection_id] = client; |
| 134 thread_->task_runner()->PostTask( | 142 task_runner_->PostTask( |
| 135 FROM_HERE, | 143 FROM_HERE, |
| 136 base::Bind(&net::HttpServer::AcceptWebSocket, | 144 base::Bind(&net::HttpServer::AcceptWebSocket, |
| 137 base::Unretained(server_.get()), connection_id, info)); | 145 base::Unretained(server_.get()), connection_id, info)); |
| 138 } | 146 } |
| 139 | 147 |
| 140 void UiDevToolsServer::OnWebSocketMessage(int connection_id, | 148 void UiDevToolsServer::OnWebSocketMessage(int connection_id, |
| 141 const std::string& data) { | 149 const std::string& data) { |
| 142 ConnectionsMap::iterator it = connections_.find(connection_id); | 150 ConnectionsMap::iterator it = connections_.find(connection_id); |
| 143 DCHECK(it != connections_.end()); | 151 DCHECK(it != connections_.end()); |
| 144 UiDevToolsClient* client = it->second; | 152 UiDevToolsClient* client = it->second; |
| 145 DCHECK(client); | 153 DCHECK(client); |
| 146 thread_->task_runner()->PostTask( | 154 task_runner_->PostTask(FROM_HERE, base::Bind(&UiDevToolsClient::Dispatch, |
| 147 FROM_HERE, | 155 base::Unretained(client), data)); |
| 148 base::Bind(&UiDevToolsClient::Dispatch, base::Unretained(client), data)); | |
| 149 } | 156 } |
| 150 | 157 |
| 151 void UiDevToolsServer::OnClose(int connection_id) { | 158 void UiDevToolsServer::OnClose(int connection_id) { |
| 152 ConnectionsMap::iterator it = connections_.find(connection_id); | 159 ConnectionsMap::iterator it = connections_.find(connection_id); |
| 153 DCHECK(it != connections_.end()); | 160 if (it == connections_.end()) |
| 161 return; |
| 154 UiDevToolsClient* client = it->second; | 162 UiDevToolsClient* client = it->second; |
| 155 DCHECK(client); | 163 DCHECK(client); |
| 156 client->set_connection_id(UiDevToolsClient::kNotConnected); | 164 client->set_connection_id(UiDevToolsClient::kNotConnected); |
| 157 connections_.erase(it); | 165 connections_.erase(it); |
| 158 } | 166 } |
| 159 | 167 |
| 160 } // namespace devtools | 168 } // namespace devtools |
| 161 } // namespace ui | 169 } // namespace ui |
| OLD | NEW |