1 // Copyright 2013 The Chromium Authors. All rights reserved.
2 // Use of this source code is governed by a BSD-style license that can be
3 // found in the LICENSE file.
5 #include "base/strings/string_number_conversions.h"
6 #include "chrome/browser/devtools/devtools_adb_bridge.h"
7 #include "chrome/browser/ui/browser.h"
8 #include "chrome/test/base/in_process_browser_test.h"
9 #include "content/public/browser/browser_thread.h"
10 #include "content/public/test/browser_test.h"
11 #include "content/public/test/test_utils.h"
12 #include "net/base/host_port_pair.h"
13 #include "net/base/io_buffer.h"
14 #include "net/base/ip_endpoint.h"
15 #include "net/base/net_errors.h"
16 #include "net/base/net_log.h"
17 #include "net/socket/stream_socket.h"
18 #include "net/socket/tcp_server_socket.h"
20 const char kOpenedUnixSocketsCommand
[] = "shell:cat /proc/net/unix";
21 const char kDeviceModelCommand
[] = "shell:getprop ro.product.model";
22 const char kDumpsysCommand
[] = "shell:dumpsys window policy";
23 const char kListProcessesCommand
[] = "shell:ps";
24 const char kInstalledChromePackagesCommand
[] = "shell:pm list packages";
25 const char kDeviceModel
[] = "Nexus 8";
27 const char kSampleOpenedUnixSocketsWithoutBrowsers
[] =
28 "Num RefCount Protocol Flags Type St Inode Path\n"
29 "00000000: 00000004 00000000"
30 " 00000000 0002 01 3328 /dev/socket/wpa_wlan0\n"
31 "00000000: 00000002 00000000"
32 " 00010000 0001 01 5394 /dev/socket/vold\n";
34 const char kSampleDumpsys
[] =
35 "WINDOW MANAGER POLICY STATE (dumpsys window policy)\r\n"
36 " mStable=(0,50)-(720,1184)\r\n";
38 const char kSampleListProcesses
[] =
39 "USER PID PPID VSIZE RSS WCHAN PC NAME\n"
40 "root 1 0 688 508 ffffffff 00000000 S /init\n";
42 const char kSampleListPackages
[] = "package:com.example.app";
44 static const int kBufferSize
= 16*1024;
45 static const int kAdbPort
= 5037;
47 static const int kAdbMessageHeaderSize
= 4;
49 // This is single connection server which listens on specified port and
50 // simplifies asynchronous IO.
51 // To write custom server, extend this class and implement TryProcessData
52 // method which is invoked everytime data arrives. In case of successful data
53 // processing(e.g. enough data collected already to parse client reply/request)
54 // return amount of bytes processed to throw them away from buffer
55 // To send data, SendData method should be used. This method is non-blocking
56 // and appends data to be sent to internal buffer.
57 // Since all calls are non-blocking and no callbacks are given, internal
58 // overflows may occur in case too heavy traffic.
59 // In case of heavy traffic performance may suffer because of memcpy calls.
60 class SingleConnectionServer
{
62 SingleConnectionServer(net::IPEndPoint endpoint
, int buffer_size
);
63 virtual ~SingleConnectionServer();
66 virtual int TryProcessData(const char* data
, int size
) = 0;
67 void SendData(const char* data
, int size
);
70 void AcceptConnection();
71 void OnAccepted(int result
);
74 void OnDataRead(int count
);
77 void OnDataWritten(int count
);
81 scoped_ptr
<net::TCPServerSocket
> server_socket_
;
82 scoped_ptr
<net::StreamSocket
> client_socket_
;
83 scoped_refptr
<net::GrowableIOBuffer
> input_buffer_
;
84 scoped_refptr
<net::GrowableIOBuffer
> output_buffer_
;
86 DISALLOW_COPY_AND_ASSIGN(SingleConnectionServer
);
89 SingleConnectionServer::SingleConnectionServer(net::IPEndPoint endpoint
,
91 : bytes_to_write_(0) {
92 CHECK(content::BrowserThread::CurrentlyOn(content::BrowserThread::IO
));
94 input_buffer_
= new net::GrowableIOBuffer();
95 input_buffer_
->SetCapacity(buffer_size
);
97 output_buffer_
= new net::GrowableIOBuffer();
99 server_socket_
.reset(new net::TCPServerSocket(NULL
, net::NetLog::Source()));
100 server_socket_
->Listen(endpoint
, 1);
105 SingleConnectionServer::~SingleConnectionServer() {
106 CHECK(content::BrowserThread::CurrentlyOn(content::BrowserThread::IO
));
108 server_socket_
.reset();
110 if (client_socket_
) {
111 client_socket_
->Disconnect();
112 client_socket_
.reset();
116 void SingleConnectionServer::SendData(const char* data
, int size
) {
117 CHECK(content::BrowserThread::CurrentlyOn(content::BrowserThread::IO
));
119 if ((output_buffer_
->offset() + bytes_to_write_
+ size
) >
120 output_buffer_
->capacity()) {
121 // If not enough space without relocation
122 if (output_buffer_
->capacity() < (bytes_to_write_
+ size
)) {
123 // If even buffer is not enough
124 int new_size
= std::max(output_buffer_
->capacity() * 2, size
* 2);
125 output_buffer_
->SetCapacity(new_size
);
127 memmove(output_buffer_
->StartOfBuffer(),
128 output_buffer_
->data(),
130 output_buffer_
->set_offset(0);
133 memcpy(output_buffer_
->data() + bytes_to_write_
, data
, size
);
134 bytes_to_write_
+= size
;
136 if (bytes_to_write_
== size
)
137 // If write loop wasn't yet started, then start it
141 void SingleConnectionServer::AcceptConnection() {
142 if (client_socket_
) {
143 client_socket_
->Disconnect();
144 client_socket_
.reset();
147 int accept_result
= server_socket_
->Accept(&client_socket_
,
148 base::Bind(&SingleConnectionServer::OnAccepted
, base::Unretained(this)));
150 if (accept_result
!= net::ERR_IO_PENDING
)
151 content::BrowserThread::PostTask(
152 content::BrowserThread::IO
,
154 base::Bind(&SingleConnectionServer::OnAccepted
,
155 base::Unretained(this),
159 void SingleConnectionServer::OnAccepted(int result
) {
160 ASSERT_EQ(result
, 0); // Fails if the socket is already in use.
164 void SingleConnectionServer::ReadData() {
165 CHECK(content::BrowserThread::CurrentlyOn(content::BrowserThread::IO
));
167 if (input_buffer_
->RemainingCapacity() == 0)
168 input_buffer_
->SetCapacity(input_buffer_
->capacity() * 2);
170 int read_result
= client_socket_
->Read(
172 input_buffer_
->RemainingCapacity(),
173 base::Bind(&SingleConnectionServer::OnDataRead
, base::Unretained(this)));
175 if (read_result
!= net::ERR_IO_PENDING
)
176 OnDataRead(read_result
);
179 void SingleConnectionServer::OnDataRead(int count
) {
185 input_buffer_
->set_offset(input_buffer_
->offset() + count
);
190 char* data
= input_buffer_
->StartOfBuffer();
191 int data_size
= input_buffer_
->offset();
193 bytes_processed
= TryProcessData(data
, data_size
);
195 if (bytes_processed
) {
196 memmove(data
, data
+ bytes_processed
, data_size
- bytes_processed
);
197 input_buffer_
->set_offset( data_size
- bytes_processed
);
199 } while (bytes_processed
);
201 // Posting is needed not to enter deep recursion in case too synchronous IO
202 content::BrowserThread::PostTask(content::BrowserThread::IO
, FROM_HERE
,
203 base::Bind(&SingleConnectionServer::ReadData
, base::Unretained(this)));
206 void SingleConnectionServer::WriteData() {
207 CHECK(content::BrowserThread::CurrentlyOn(content::BrowserThread::IO
));
208 CHECK_GE(output_buffer_
->capacity(),
209 output_buffer_
->offset() + bytes_to_write_
) << "Overflow";
211 int write_result
= client_socket_
->Write(
214 base::Bind(&SingleConnectionServer::OnDataWritten
,
215 base::Unretained(this)));
216 if (write_result
!= net::ERR_IO_PENDING
)
217 OnDataWritten(write_result
);
220 void SingleConnectionServer::OnDataWritten(int count
) {
221 CHECK(content::BrowserThread::CurrentlyOn(content::BrowserThread::IO
));
228 CHECK_GE(output_buffer_
->capacity(),
229 output_buffer_
->offset() + bytes_to_write_
) << "Overflow";
231 bytes_to_write_
-= count
;
232 output_buffer_
->set_offset(output_buffer_
->offset() + count
);
234 if (bytes_to_write_
!= 0)
235 // Posting is needed not to enter deep recursion in case too synchronous IO
236 content::BrowserThread::PostTask(content::BrowserThread::IO
, FROM_HERE
,
237 base::Bind(&SingleConnectionServer::WriteData
, base::Unretained(this)));
241 class MockAdbServer
: public SingleConnectionServer
{
243 MockAdbServer(net::IPEndPoint endpoint
, int buffer_size
)
244 : SingleConnectionServer(endpoint
, buffer_size
)
247 virtual ~MockAdbServer() {}
250 virtual int TryProcessData(const char* data
, int size
) OVERRIDE
{
251 CHECK(content::BrowserThread::CurrentlyOn(content::BrowserThread::IO
));
253 if (size
>= kAdbMessageHeaderSize
) {
254 std::string
message_header(data
, kAdbMessageHeaderSize
);
257 EXPECT_TRUE(base::HexStringToInt(message_header
, &message_size
));
259 if (size
>= message_size
+ kAdbMessageHeaderSize
) {
260 std::string
message_body(data
+ kAdbMessageHeaderSize
, message_size
);
262 ProcessCommand(message_body
);
264 return kAdbMessageHeaderSize
+ message_size
;
271 void ProcessCommand(const std::string
& command
) {
272 CHECK(content::BrowserThread::CurrentlyOn(content::BrowserThread::IO
));
274 if (command
== "host:devices") {
275 SendResponse("01498B321301A00A\tdevice\n01498B2B0D01300E\toffline");
276 } else if (command
== "host:transport:01498B321301A00A") {
278 } else if (command
== kDeviceModelCommand
) {
279 SendResponse(kDeviceModel
);
280 } else if (command
== kOpenedUnixSocketsCommand
) {
281 SendResponse(kSampleOpenedUnixSocketsWithoutBrowsers
);
282 } else if (command
== kDumpsysCommand
) {
283 SendResponse(kSampleDumpsys
);
284 } else if (command
== kListProcessesCommand
) {
285 SendResponse(kSampleListProcesses
);
286 } else if (command
== kInstalledChromePackagesCommand
) {
287 SendResponse(kSampleListPackages
);
289 NOTREACHED() << "Unknown command - " << command
;
293 void SendResponse(const std::string
& response
) {
294 DCHECK(content::BrowserThread::CurrentlyOn(content::BrowserThread::IO
));
296 std::stringstream response_stream
;
297 response_stream
<< "OKAY";
299 int size
= response
.size();
301 static const char kHexChars
[] = "0123456789ABCDEF";
302 for (int i
= 3; i
>= 0; i
--)
303 response_stream
<< kHexChars
[ (size
>> 4*i
) & 0x0f ];
304 response_stream
<< response
;
307 std::string response_data
= response_stream
.str();
308 SendData(response_data
.c_str(), response_data
.size());
312 class AdbClientSocketTest
: public InProcessBrowserTest
,
313 public DevToolsAdbBridge::Listener
{
317 DCHECK(content::BrowserThread::CurrentlyOn(content::BrowserThread::UI
));
319 content::BrowserThread::PostTaskAndReply(
320 content::BrowserThread::IO
,
322 base::Bind(&AdbClientSocketTest::StartMockAdbServer
,
323 base::Unretained(this)),
324 base::Bind(&AdbClientSocketTest::AddListener
,
325 base::Unretained(this)));
328 virtual void RemoteDevicesChanged(DevToolsAdbBridge::RemoteDevices
* devices
)
330 DCHECK(content::BrowserThread::CurrentlyOn(content::BrowserThread::UI
));
331 adb_bridge_
->RemoveListener(this);
336 void CheckDevices() {
337 #if defined(DEBUG_DEVTOOLS)
338 // Mock device is added
339 ASSERT_EQ(3U, devices_
.size());
341 ASSERT_EQ(2U, devices_
.size());
344 scoped_refptr
<DevToolsAdbBridge::RemoteDevice
> online_device_
;
345 scoped_refptr
<DevToolsAdbBridge::RemoteDevice
> offline_device_
;
347 for (DevToolsAdbBridge::RemoteDevices::const_iterator it
=
348 devices_
.begin(); it
!= devices_
.end(); ++it
) {
349 if ((*it
)->GetSerial() == "01498B321301A00A")
350 online_device_
= *it
;
351 else if ((*it
)->GetSerial() == "01498B2B0D01300E")
352 offline_device_
= *it
;
355 ASSERT_EQ(online_device_
->GetSerial(), "01498B321301A00A");
356 ASSERT_TRUE(online_device_
->device()->is_connected());
357 ASSERT_FALSE(offline_device_
->device()->is_connected());
359 ASSERT_EQ(online_device_
->GetModel(), kDeviceModel
);
360 ASSERT_EQ(online_device_
->browsers().size(), 0U);
361 ASSERT_EQ(online_device_
->screen_size(), gfx::Size(720, 1184));
366 DCHECK(content::BrowserThread::CurrentlyOn(content::BrowserThread::UI
));
369 content::BrowserThread::PostTaskAndReply(
370 content::BrowserThread::IO
,
372 base::Bind(&AdbClientSocketTest::StopMockAdbServer
,
373 base::Unretained(this)),
374 base::Bind(&AdbClientSocketTest::StopMessageLoop
,
375 base::Unretained(this)));
378 void StartMockAdbServer() {
379 DCHECK(content::BrowserThread::CurrentlyOn(content::BrowserThread::IO
));
380 net::IPAddressNumber address
;
381 net::ParseIPLiteralToNumber("127.0.0.1", &address
);
382 net::IPEndPoint
endpoint(address
, kAdbPort
);
384 adb_server_
.reset(new MockAdbServer(endpoint
, kBufferSize
));
387 void StopMockAdbServer() {
388 DCHECK(content::BrowserThread::CurrentlyOn(content::BrowserThread::IO
));
392 void StopMessageLoop() {
393 DCHECK(content::BrowserThread::CurrentlyOn(content::BrowserThread::UI
));
398 DCHECK(content::BrowserThread::CurrentlyOn(content::BrowserThread::UI
));
399 adb_bridge_
= DevToolsAdbBridge::Factory::GetForProfile(
400 browser()->profile());
402 DevToolsAdbBridge::DeviceProviders device_providers
;
403 device_providers
.push_back(AndroidDeviceProvider::GetAdbDeviceProvider());
405 adb_bridge_
->set_device_providers(device_providers
);
406 adb_bridge_
->AddListener(this);
410 scoped_refptr
<content::MessageLoopRunner
> runner
;
413 scoped_ptr
<MockAdbServer
> adb_server_
;
414 scoped_refptr
<DevToolsAdbBridge
> adb_bridge_
;
415 DevToolsAdbBridge::RemoteDevices devices_
;
418 IN_PROC_BROWSER_TEST_F(AdbClientSocketTest
, TestAdbClientSocket
) {
419 CHECK(content::BrowserThread::CurrentlyOn(content::BrowserThread::UI
));
420 runner
= new content::MessageLoopRunner
;