OLD | NEW |
1 // Copyright 2014 The Chromium Authors. All rights reserved. | 1 // Copyright 2014 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 "chrome/browser/devtools/device/devtools_android_bridge.h" | 5 #include "chrome/browser/devtools/device/devtools_android_bridge.h" |
6 | 6 |
7 #include <stddef.h> | 7 #include <stddef.h> |
8 #include <algorithm> | 8 #include <algorithm> |
9 #include <map> | 9 #include <map> |
10 #include <set> | 10 #include <set> |
(...skipping 37 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
48 #include "content/public/browser/user_metrics.h" | 48 #include "content/public/browser/user_metrics.h" |
49 #include "net/base/escape.h" | 49 #include "net/base/escape.h" |
50 #include "net/base/host_port_pair.h" | 50 #include "net/base/host_port_pair.h" |
51 #include "net/base/net_errors.h" | 51 #include "net/base/net_errors.h" |
52 | 52 |
53 #if defined(ENABLE_SERVICE_DISCOVERY) | 53 #if defined(ENABLE_SERVICE_DISCOVERY) |
54 #include "chrome/browser/devtools/device/cast_device_provider.h" | 54 #include "chrome/browser/devtools/device/cast_device_provider.h" |
55 #endif | 55 #endif |
56 | 56 |
57 using content::BrowserThread; | 57 using content::BrowserThread; |
| 58 using content::DevToolsAgentHost; |
58 | 59 |
59 namespace { | 60 namespace { |
60 | 61 |
61 const char kPageListRequest[] = "/json"; | 62 const char kPageListRequest[] = "/json"; |
62 const char kVersionRequest[] = "/json/version"; | 63 const char kVersionRequest[] = "/json/version"; |
63 const char kClosePageRequest[] = "/json/close/%s"; | 64 const char kClosePageRequest[] = "/json/close/%s"; |
64 const char kNewPageRequestWithURL[] = "/json/new?%s"; | 65 const char kNewPageRequestWithURL[] = "/json/new?%s"; |
65 const char kActivatePageRequest[] = "/json/activate/%s"; | 66 const char kActivatePageRequest[] = "/json/activate/%s"; |
66 const char kBrowserTargetSocket[] = "/devtools/browser"; | 67 const char kBrowserTargetSocket[] = "/devtools/browser"; |
67 const int kAdbPollingIntervalMs = 1000; | 68 const int kAdbPollingIntervalMs = 1000; |
68 | 69 |
69 const char kPageReloadCommand[] = "Page.reload"; | 70 const char kPageReloadCommand[] = "Page.reload"; |
70 | 71 |
71 const char kWebViewSocketPrefix[] = "webview_devtools_remote"; | 72 const char kWebViewSocketPrefix[] = "webview_devtools_remote"; |
72 | 73 |
73 bool BrowserIdFromString(const std::string& browser_id_str, | 74 bool BrowserIdFromString(const std::string& browser_id_str, |
74 DevToolsAndroidBridge::BrowserId* browser_id) { | 75 DevToolsAndroidBridge::BrowserId* browser_id) { |
75 size_t colon_pos = browser_id_str.find(':'); | 76 size_t colon_pos = browser_id_str.find(':'); |
76 if (colon_pos == std::string::npos) | 77 if (colon_pos == std::string::npos) |
77 return false; | 78 return false; |
78 browser_id->first = browser_id_str.substr(0, colon_pos); | 79 browser_id->first = browser_id_str.substr(0, colon_pos); |
79 browser_id->second = browser_id_str.substr(colon_pos + 1); | 80 browser_id->second = browser_id_str.substr(colon_pos + 1); |
80 return true; | 81 return true; |
81 } | 82 } |
82 | 83 |
| 84 static void NoOp(int, const std::string&) {} |
| 85 |
83 } // namespace | 86 } // namespace |
84 | 87 |
85 // DiscoveryRequest ----------------------------------------------------- | 88 // DiscoveryRequest ----------------------------------------------------- |
86 | 89 |
87 class DevToolsAndroidBridge::DiscoveryRequest | 90 class DevToolsAndroidBridge::DiscoveryRequest |
88 : public base::RefCountedThreadSafe<DiscoveryRequest, | 91 : public base::RefCountedThreadSafe<DiscoveryRequest, |
89 BrowserThread::DeleteOnUIThread> { | 92 BrowserThread::DeleteOnUIThread> { |
90 public: | 93 public: |
91 DiscoveryRequest(AndroidDeviceManager* device_manager, | 94 DiscoveryRequest(AndroidDeviceManager* device_manager, |
92 const DeviceListCallback& callback); | 95 const DeviceListCallback& callback); |
(...skipping 190 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
283 } | 286 } |
284 | 287 |
285 // AgentHostDelegate ---------------------------------------------------------- | 288 // AgentHostDelegate ---------------------------------------------------------- |
286 | 289 |
287 class DevToolsAndroidBridge::AgentHostDelegate | 290 class DevToolsAndroidBridge::AgentHostDelegate |
288 : public content::DevToolsExternalAgentProxyDelegate, | 291 : public content::DevToolsExternalAgentProxyDelegate, |
289 public AndroidDeviceManager::AndroidWebSocket::Delegate { | 292 public AndroidDeviceManager::AndroidWebSocket::Delegate { |
290 public: | 293 public: |
291 static scoped_refptr<content::DevToolsAgentHost> GetOrCreateAgentHost( | 294 static scoped_refptr<content::DevToolsAgentHost> GetOrCreateAgentHost( |
292 DevToolsAndroidBridge* bridge, | 295 DevToolsAndroidBridge* bridge, |
293 const std::string& id, | |
294 const BrowserId& browser_id, | 296 const BrowserId& browser_id, |
295 const std::string& target_path); | 297 const std::string& local_id, |
| 298 const std::string& target_path, |
| 299 const std::string& type, |
| 300 base::DictionaryValue* value); |
296 | 301 |
297 private: | 302 private: |
298 AgentHostDelegate( | 303 AgentHostDelegate( |
299 DevToolsAndroidBridge* bridge, | 304 DevToolsAndroidBridge* bridge, |
300 const std::string& id, | |
301 const BrowserId& browser_id, | 305 const BrowserId& browser_id, |
302 const std::string& target_path); | 306 const std::string& local_id, |
| 307 const std::string& target_path, |
| 308 const std::string& type, |
| 309 base::DictionaryValue* value); |
303 ~AgentHostDelegate() override; | 310 ~AgentHostDelegate() override; |
| 311 // DevToolsExternalAgentProxyDelegate overrides. |
304 void Attach(content::DevToolsExternalAgentProxy* proxy) override; | 312 void Attach(content::DevToolsExternalAgentProxy* proxy) override; |
305 void Detach() override; | 313 void Detach() override; |
| 314 std::string GetId() override; |
| 315 std::string GetType() override; |
| 316 std::string GetTitle() override; |
| 317 std::string GetDescription() override; |
| 318 GURL GetURL() override; |
| 319 GURL GetFaviconURL() override; |
| 320 bool Activate() override; |
| 321 bool Inspect() override; |
| 322 void Reload() override; |
| 323 bool Close() override; |
306 void SendMessageToBackend(const std::string& message) override; | 324 void SendMessageToBackend(const std::string& message) override; |
| 325 |
307 void OnSocketOpened() override; | 326 void OnSocketOpened() override; |
308 void OnFrameRead(const std::string& message) override; | 327 void OnFrameRead(const std::string& message) override; |
309 void OnSocketClosed() override; | 328 void OnSocketClosed() override; |
310 | 329 |
311 std::string id_; | |
312 base::WeakPtr<DevToolsAndroidBridge> bridge_; | 330 base::WeakPtr<DevToolsAndroidBridge> bridge_; |
313 BrowserId browser_id_; | 331 BrowserId browser_id_; |
| 332 std::string local_id_; |
314 std::string target_path_; | 333 std::string target_path_; |
| 334 std::string remote_type_; |
| 335 std::string remote_id_; |
| 336 std::string frontend_url_; |
| 337 std::string title_; |
| 338 std::string description_; |
| 339 GURL url_; |
| 340 GURL favicon_url_; |
315 bool socket_opened_; | 341 bool socket_opened_; |
316 std::vector<std::string> pending_messages_; | 342 std::vector<std::string> pending_messages_; |
317 scoped_refptr<AndroidDeviceManager::Device> device_; | 343 scoped_refptr<AndroidDeviceManager::Device> device_; |
318 std::unique_ptr<AndroidDeviceManager::AndroidWebSocket> web_socket_; | 344 std::unique_ptr<AndroidDeviceManager::AndroidWebSocket> web_socket_; |
319 content::DevToolsAgentHost* agent_host_; | 345 content::DevToolsAgentHost* agent_host_; |
320 content::DevToolsExternalAgentProxy* proxy_; | 346 content::DevToolsExternalAgentProxy* proxy_; |
321 DISALLOW_COPY_AND_ASSIGN(AgentHostDelegate); | 347 DISALLOW_COPY_AND_ASSIGN(AgentHostDelegate); |
322 }; | 348 }; |
323 | 349 |
| 350 static std::string GetStringProperty(base::DictionaryValue* value, |
| 351 const std::string& name) { |
| 352 std::string result; |
| 353 value->GetString(name, &result); |
| 354 return result; |
| 355 } |
| 356 |
| 357 static std::string BuildUniqueTargetId( |
| 358 const DevToolsAndroidBridge::BrowserId& browser_id, |
| 359 base::DictionaryValue* value) { |
| 360 return base::StringPrintf("%s:%s:%s", browser_id.first.c_str(), |
| 361 browser_id.second.c_str(), GetStringProperty(value, "id").c_str()); |
| 362 } |
| 363 |
| 364 static std::string GetFrontendURL(base::DictionaryValue* value) { |
| 365 std::string frontend_url = GetStringProperty(value, "devtoolsFrontendUrl"); |
| 366 size_t ws_param = frontend_url.find("?ws"); |
| 367 if (ws_param != std::string::npos) |
| 368 frontend_url = frontend_url.substr(0, ws_param); |
| 369 if (base::StartsWith(frontend_url, "http:", base::CompareCase::SENSITIVE)) |
| 370 frontend_url = "https:" + frontend_url.substr(5); |
| 371 return frontend_url; |
| 372 } |
| 373 |
| 374 static std::string GetTargetPath(base::DictionaryValue* value) { |
| 375 std::string target_path = GetStringProperty(value, "webSocketDebuggerUrl"); |
| 376 |
| 377 if (base::StartsWith(target_path, "ws://", base::CompareCase::SENSITIVE)) { |
| 378 size_t pos = target_path.find("/", 5); |
| 379 if (pos == std::string::npos) |
| 380 pos = 5; |
| 381 target_path = target_path.substr(pos); |
| 382 } else { |
| 383 target_path = std::string(); |
| 384 } |
| 385 return target_path; |
| 386 } |
| 387 |
324 // static | 388 // static |
325 scoped_refptr<content::DevToolsAgentHost> | 389 scoped_refptr<content::DevToolsAgentHost> |
326 DevToolsAndroidBridge::AgentHostDelegate::GetOrCreateAgentHost( | 390 DevToolsAndroidBridge::AgentHostDelegate::GetOrCreateAgentHost( |
327 DevToolsAndroidBridge* bridge, | 391 DevToolsAndroidBridge* bridge, |
328 const std::string& id, | |
329 const BrowserId& browser_id, | 392 const BrowserId& browser_id, |
330 const std::string& target_path) { | 393 const std::string& local_id, |
| 394 const std::string& target_path, |
| 395 const std::string& type, |
| 396 base::DictionaryValue* value) { |
331 DCHECK_CURRENTLY_ON(BrowserThread::UI); | 397 DCHECK_CURRENTLY_ON(BrowserThread::UI); |
332 AgentHostDelegates::iterator it = bridge->host_delegates_.find(id); | 398 AgentHostDelegates::iterator it = bridge->host_delegates_.find(local_id); |
333 if (it != bridge->host_delegates_.end()) | 399 if (it != bridge->host_delegates_.end()) |
334 return it->second->agent_host_; | 400 return it->second->agent_host_; |
335 | 401 |
336 AgentHostDelegate* delegate = | 402 AgentHostDelegate* delegate = new AgentHostDelegate( |
337 new AgentHostDelegate(bridge, id, browser_id, target_path); | 403 bridge, browser_id, local_id, target_path, type, value); |
338 scoped_refptr<content::DevToolsAgentHost> result = | 404 scoped_refptr<content::DevToolsAgentHost> result = |
339 content::DevToolsAgentHost::Create(delegate); | 405 content::DevToolsAgentHost::Create(delegate); |
340 delegate->agent_host_ = result.get(); | 406 delegate->agent_host_ = result.get(); |
341 return result; | 407 return result; |
342 } | 408 } |
343 | 409 |
344 DevToolsAndroidBridge::AgentHostDelegate::AgentHostDelegate( | 410 DevToolsAndroidBridge::AgentHostDelegate::AgentHostDelegate( |
345 DevToolsAndroidBridge* bridge, | 411 DevToolsAndroidBridge* bridge, |
346 const std::string& id, | |
347 const BrowserId& browser_id, | 412 const BrowserId& browser_id, |
348 const std::string& target_path) | 413 const std::string& local_id, |
349 : id_(id), | 414 const std::string& target_path, |
350 bridge_(bridge->AsWeakPtr()), | 415 const std::string& type, |
| 416 base::DictionaryValue* value) |
| 417 : bridge_(bridge->AsWeakPtr()), |
351 browser_id_(browser_id), | 418 browser_id_(browser_id), |
| 419 local_id_(local_id), |
352 target_path_(target_path), | 420 target_path_(target_path), |
| 421 remote_type_(type), |
| 422 remote_id_(value ? GetStringProperty(value, "id") : ""), |
| 423 frontend_url_(value ? GetFrontendURL(value) : ""), |
| 424 title_(value ? base::UTF16ToUTF8(net::UnescapeForHTML(base::UTF8ToUTF16( |
| 425 GetStringProperty(value, "title")))) : ""), |
| 426 description_(value ? GetStringProperty(value, "description") : ""), |
| 427 url_(GURL(value ? GetStringProperty(value, "url") : "")), |
| 428 favicon_url_(GURL(value ? GetStringProperty(value, "faviconUrl") : "")), |
353 socket_opened_(false), | 429 socket_opened_(false), |
354 agent_host_(NULL), | 430 agent_host_(NULL), |
355 proxy_(NULL) { | 431 proxy_(NULL) { |
356 bridge_->host_delegates_[id] = this; | 432 bridge_->host_delegates_[local_id_] = this; |
357 } | 433 } |
358 | 434 |
359 DevToolsAndroidBridge::AgentHostDelegate::~AgentHostDelegate() { | 435 DevToolsAndroidBridge::AgentHostDelegate::~AgentHostDelegate() { |
360 if (bridge_) | 436 if (bridge_) |
361 bridge_->host_delegates_.erase(id_); | 437 bridge_->host_delegates_.erase(local_id_); |
362 } | 438 } |
363 | 439 |
364 void DevToolsAndroidBridge::AgentHostDelegate::Attach( | 440 void DevToolsAndroidBridge::AgentHostDelegate::Attach( |
365 content::DevToolsExternalAgentProxy* proxy) { | 441 content::DevToolsExternalAgentProxy* proxy) { |
366 proxy_ = proxy; | 442 proxy_ = proxy; |
367 content::RecordAction( | 443 content::RecordAction( |
368 base::StartsWith(browser_id_.second, kWebViewSocketPrefix, | 444 base::StartsWith(browser_id_.second, kWebViewSocketPrefix, |
369 base::CompareCase::SENSITIVE) | 445 base::CompareCase::SENSITIVE) |
370 ? base::UserMetricsAction("DevTools_InspectAndroidWebView") | 446 ? base::UserMetricsAction("DevTools_InspectAndroidWebView") |
371 : base::UserMetricsAction("DevTools_InspectAndroidPage")); | 447 : base::UserMetricsAction("DevTools_InspectAndroidPage")); |
372 | 448 |
373 // Retain the device so it's not released until AgentHost is detached. | 449 // Retain the device so it's not released until AgentHost is detached. |
374 if (bridge_) | 450 if (bridge_) |
375 device_ = bridge_->FindDevice(browser_id_.first); | 451 device_ = bridge_->FindDevice(browser_id_.first); |
376 if (!device_.get()) | 452 if (!device_.get()) |
377 return; | 453 return; |
378 | 454 |
379 web_socket_.reset( | 455 web_socket_.reset( |
380 device_->CreateWebSocket(browser_id_.second, target_path_, this)); | 456 device_->CreateWebSocket(browser_id_.second, target_path_, this)); |
381 } | 457 } |
382 | 458 |
383 void DevToolsAndroidBridge::AgentHostDelegate::Detach() { | 459 void DevToolsAndroidBridge::AgentHostDelegate::Detach() { |
384 web_socket_.reset(); | 460 web_socket_.reset(); |
385 device_ = nullptr; | 461 device_ = nullptr; |
386 proxy_ = nullptr; | 462 proxy_ = nullptr; |
387 } | 463 } |
388 | 464 |
389 void DevToolsAndroidBridge::AgentHostDelegate::SendMessageToBackend( | 465 std::string DevToolsAndroidBridge::AgentHostDelegate::GetId() { |
390 const std::string& message) { | |
391 // We could have detached due to physical connection being closed. | |
392 if (!proxy_) | |
393 return; | |
394 if (socket_opened_) | |
395 web_socket_->SendFrame(message); | |
396 else | |
397 pending_messages_.push_back(message); | |
398 } | |
399 | |
400 void DevToolsAndroidBridge::AgentHostDelegate::OnSocketOpened() { | |
401 socket_opened_ = true; | |
402 for (std::vector<std::string>::iterator it = pending_messages_.begin(); | |
403 it != pending_messages_.end(); ++it) { | |
404 SendMessageToBackend(*it); | |
405 } | |
406 pending_messages_.clear(); | |
407 } | |
408 | |
409 void DevToolsAndroidBridge::AgentHostDelegate::OnFrameRead( | |
410 const std::string& message) { | |
411 if (proxy_) | |
412 proxy_->DispatchOnClientHost(message); | |
413 } | |
414 | |
415 void DevToolsAndroidBridge::AgentHostDelegate::OnSocketClosed() { | |
416 if (proxy_) { | |
417 std::string message = "{ \"method\": \"Inspector.detached\", " | |
418 "\"params\": { \"reason\": \"Connection lost.\"} }"; | |
419 proxy_->DispatchOnClientHost(message); | |
420 Detach(); | |
421 } | |
422 } | |
423 | |
424 //// RemotePageTarget ---------------------------------------------- | |
425 | |
426 class DevToolsAndroidBridge::RemotePageTarget : public DevToolsTargetImpl { | |
427 public: | |
428 RemotePageTarget(DevToolsAndroidBridge* bridge, | |
429 const BrowserId& browser_id, | |
430 const base::DictionaryValue& value); | |
431 ~RemotePageTarget() override; | |
432 | |
433 // DevToolsTargetImpl overrides. | |
434 std::string GetId() const override; | |
435 bool IsAttached() const override; | |
436 bool Activate() const override; | |
437 bool Close() const override; | |
438 void Inspect(Profile* profile) const override; | |
439 void Reload() const override; | |
440 | |
441 private: | |
442 base::WeakPtr<DevToolsAndroidBridge> bridge_; | |
443 BrowserId browser_id_; | |
444 std::string target_path_; | |
445 std::string frontend_url_; | |
446 std::string remote_id_; | |
447 std::string remote_type_; | |
448 std::string local_id_; | |
449 DISALLOW_COPY_AND_ASSIGN(RemotePageTarget); | |
450 }; | |
451 | |
452 static std::string GetStringProperty(const base::DictionaryValue& value, | |
453 const std::string& name) { | |
454 std::string result; | |
455 value.GetString(name, &result); | |
456 return result; | |
457 } | |
458 | |
459 static std::string BuildUniqueTargetId( | |
460 const DevToolsAndroidBridge::BrowserId& browser_id, | |
461 const base::DictionaryValue& value) { | |
462 return base::StringPrintf("%s:%s:%s", browser_id.first.c_str(), | |
463 browser_id.second.c_str(), GetStringProperty(value, "id").c_str()); | |
464 } | |
465 | |
466 static std::string GetFrontendURL(const base::DictionaryValue& value) { | |
467 std::string frontend_url = GetStringProperty(value, "devtoolsFrontendUrl"); | |
468 size_t ws_param = frontend_url.find("?ws"); | |
469 if (ws_param != std::string::npos) | |
470 frontend_url = frontend_url.substr(0, ws_param); | |
471 if (base::StartsWith(frontend_url, "http:", base::CompareCase::SENSITIVE)) | |
472 frontend_url = "https:" + frontend_url.substr(5); | |
473 return frontend_url; | |
474 } | |
475 | |
476 static std::string GetTargetPath(const base::DictionaryValue& value) { | |
477 std::string target_path = GetStringProperty(value, "webSocketDebuggerUrl"); | |
478 | |
479 if (base::StartsWith(target_path, "ws://", base::CompareCase::SENSITIVE)) { | |
480 size_t pos = target_path.find("/", 5); | |
481 if (pos == std::string::npos) | |
482 pos = 5; | |
483 target_path = target_path.substr(pos); | |
484 } else { | |
485 target_path = std::string(); | |
486 } | |
487 return target_path; | |
488 } | |
489 | |
490 DevToolsAndroidBridge::RemotePageTarget::RemotePageTarget( | |
491 DevToolsAndroidBridge* bridge, | |
492 const BrowserId& browser_id, | |
493 const base::DictionaryValue& value) | |
494 : DevToolsTargetImpl(AgentHostDelegate::GetOrCreateAgentHost( | |
495 bridge, | |
496 BuildUniqueTargetId(browser_id, value), | |
497 browser_id, | |
498 GetTargetPath(value))), | |
499 bridge_(bridge->AsWeakPtr()), | |
500 browser_id_(browser_id), | |
501 target_path_(GetTargetPath(value)), | |
502 frontend_url_(GetFrontendURL(value)), | |
503 remote_id_(GetStringProperty(value, "id")), | |
504 remote_type_(GetStringProperty(value, "type")), | |
505 local_id_(BuildUniqueTargetId(browser_id, value)) { | |
506 set_type("adb_page"); | |
507 set_url(GURL(GetStringProperty(value, "url"))); | |
508 set_title(base::UTF16ToUTF8(net::UnescapeForHTML(base::UTF8ToUTF16( | |
509 GetStringProperty(value, "title"))))); | |
510 set_description(GetStringProperty(value, "description")); | |
511 set_favicon_url(GURL(GetStringProperty(value, "faviconUrl"))); | |
512 target_path_ = GetTargetPath(value); | |
513 } | |
514 | |
515 DevToolsAndroidBridge::RemotePageTarget::~RemotePageTarget() { | |
516 } | |
517 | |
518 std::string DevToolsAndroidBridge::RemotePageTarget::GetId() const { | |
519 return local_id_; | 466 return local_id_; |
520 } | 467 } |
521 | 468 |
522 bool DevToolsAndroidBridge::RemotePageTarget::IsAttached() const { | 469 std::string DevToolsAndroidBridge::AgentHostDelegate::GetType() { |
523 return target_path_.empty(); | 470 return remote_type_; |
524 } | 471 } |
525 | 472 |
526 static void NoOp(int, const std::string&) {} | 473 std::string DevToolsAndroidBridge::AgentHostDelegate::GetTitle() { |
527 | 474 return title_; |
528 void DevToolsAndroidBridge::RemotePageTarget::Inspect(Profile* profile) const { | |
529 Activate(); | |
530 bool is_worker = remote_type_ == kTargetTypeWorker || | |
531 remote_type_ == kTargetTypeServiceWorker; | |
532 bool is_v8_only = remote_type_ == kTargetTypeNode; | |
533 DevToolsWindow::OpenExternalFrontend(profile, frontend_url_, GetAgentHost(), | |
534 is_worker, is_v8_only); | |
535 } | 475 } |
536 | 476 |
537 bool DevToolsAndroidBridge::RemotePageTarget::Activate() const { | 477 std::string DevToolsAndroidBridge::AgentHostDelegate::GetDescription() { |
| 478 return description_; |
| 479 } |
| 480 |
| 481 GURL DevToolsAndroidBridge::AgentHostDelegate::GetURL() { |
| 482 return url_; |
| 483 } |
| 484 |
| 485 GURL DevToolsAndroidBridge::AgentHostDelegate::GetFaviconURL() { |
| 486 return favicon_url_; |
| 487 } |
| 488 |
| 489 bool DevToolsAndroidBridge::AgentHostDelegate::Activate() { |
538 if (!bridge_) | 490 if (!bridge_) |
539 return false; | 491 return false; |
540 | 492 |
541 std::string request = base::StringPrintf(kActivatePageRequest, | 493 std::string request = base::StringPrintf(kActivatePageRequest, |
542 remote_id_.c_str()); | 494 remote_id_.c_str()); |
543 bridge_->SendJsonRequest(browser_id_, request, base::Bind(&NoOp)); | 495 bridge_->SendJsonRequest(browser_id_, request, base::Bind(&NoOp)); |
544 return true; | 496 return true; |
545 } | 497 } |
546 | 498 |
547 bool DevToolsAndroidBridge::RemotePageTarget::Close() const { | 499 bool DevToolsAndroidBridge::AgentHostDelegate::Inspect() { |
| 500 Activate(); |
| 501 bool is_worker = remote_type_ == DevToolsAgentHost::kTypeServiceWorker || |
| 502 remote_type_ == DevToolsAgentHost::kTypeSharedWorker; |
| 503 bool is_v8_only = remote_type_ == "node"; |
| 504 DevToolsWindow::OpenExternalFrontend(bridge_->profile_, frontend_url_, |
| 505 agent_host_, is_worker, is_v8_only); |
| 506 return true; |
| 507 } |
| 508 |
| 509 void DevToolsAndroidBridge::AgentHostDelegate::Reload() { |
| 510 if (!bridge_) |
| 511 return; |
| 512 |
| 513 bridge_->SendProtocolCommand(browser_id_, target_path_, kPageReloadCommand, |
| 514 nullptr, base::Closure()); |
| 515 } |
| 516 |
| 517 bool DevToolsAndroidBridge::AgentHostDelegate::Close() { |
548 if (!bridge_) | 518 if (!bridge_) |
549 return false; | 519 return false; |
550 | 520 |
551 std::string request = base::StringPrintf(kClosePageRequest, | 521 std::string request = base::StringPrintf(kClosePageRequest, |
552 remote_id_.c_str()); | 522 remote_id_.c_str()); |
553 bridge_->SendJsonRequest(browser_id_, request, base::Bind(&NoOp)); | 523 bridge_->SendJsonRequest(browser_id_, request, base::Bind(&NoOp)); |
554 return true; | 524 return true; |
555 } | 525 } |
556 | 526 |
557 void DevToolsAndroidBridge::RemotePageTarget::Reload() const { | 527 void DevToolsAndroidBridge::AgentHostDelegate::SendMessageToBackend( |
558 if (!bridge_) | 528 const std::string& message) { |
| 529 // We could have detached due to physical connection being closed. |
| 530 if (!proxy_) |
559 return; | 531 return; |
| 532 if (socket_opened_) |
| 533 web_socket_->SendFrame(message); |
| 534 else |
| 535 pending_messages_.push_back(message); |
| 536 } |
560 | 537 |
561 bridge_->SendProtocolCommand(browser_id_, target_path_, kPageReloadCommand, | 538 void DevToolsAndroidBridge::AgentHostDelegate::OnSocketOpened() { |
562 NULL, base::Closure()); | 539 socket_opened_ = true; |
| 540 for (std::vector<std::string>::iterator it = pending_messages_.begin(); |
| 541 it != pending_messages_.end(); ++it) { |
| 542 SendMessageToBackend(*it); |
| 543 } |
| 544 pending_messages_.clear(); |
| 545 } |
| 546 |
| 547 void DevToolsAndroidBridge::AgentHostDelegate::OnFrameRead( |
| 548 const std::string& message) { |
| 549 if (proxy_) |
| 550 proxy_->DispatchOnClientHost(message); |
| 551 } |
| 552 |
| 553 void DevToolsAndroidBridge::AgentHostDelegate::OnSocketClosed() { |
| 554 if (proxy_) { |
| 555 std::string message = "{ \"method\": \"Inspector.detached\", " |
| 556 "\"params\": { \"reason\": \"Connection lost.\"} }"; |
| 557 proxy_->DispatchOnClientHost(message); |
| 558 Detach(); |
| 559 } |
563 } | 560 } |
564 | 561 |
565 // DevToolsAndroidBridge::RemotePage ------------------------------------------ | 562 // DevToolsAndroidBridge::RemotePage ------------------------------------------ |
566 | 563 |
567 DevToolsAndroidBridge::RemotePage::RemotePage(const BrowserId& browser_id, | 564 DevToolsAndroidBridge::RemotePage::RemotePage(const BrowserId& browser_id, |
568 const base::DictionaryValue& dict) | 565 const base::DictionaryValue& dict) |
569 : browser_id_(browser_id), | 566 : browser_id_(browser_id), |
570 frontend_url_(GetFrontendURL(dict)), | |
571 dict_(dict.DeepCopy()) { | 567 dict_(dict.DeepCopy()) { |
572 } | 568 } |
573 | 569 |
574 DevToolsAndroidBridge::RemotePage::~RemotePage() { | 570 DevToolsAndroidBridge::RemotePage::~RemotePage() { |
575 } | 571 } |
576 | 572 |
577 // DevToolsAndroidBridge::RemoteBrowser --------------------------------------- | 573 // DevToolsAndroidBridge::RemoteBrowser --------------------------------------- |
578 | 574 |
579 DevToolsAndroidBridge::RemoteBrowser::RemoteBrowser( | 575 DevToolsAndroidBridge::RemoteBrowser::RemoteBrowser( |
580 const std::string& serial, | 576 const std::string& serial, |
(...skipping 20 matching lines...) Expand all Loading... |
601 version_, ".", base::KEEP_WHITESPACE, base::SPLIT_WANT_NONEMPTY)) { | 597 version_, ".", base::KEEP_WHITESPACE, base::SPLIT_WANT_NONEMPTY)) { |
602 int value = 0; | 598 int value = 0; |
603 base::StringToInt(part, &value); | 599 base::StringToInt(part, &value); |
604 result.push_back(value); | 600 result.push_back(value); |
605 } | 601 } |
606 return result; | 602 return result; |
607 } | 603 } |
608 | 604 |
609 DevToolsTargetImpl* | 605 DevToolsTargetImpl* |
610 DevToolsAndroidBridge::CreatePageTarget(scoped_refptr<RemotePage> page) { | 606 DevToolsAndroidBridge::CreatePageTarget(scoped_refptr<RemotePage> page) { |
611 return new RemotePageTarget(this, page->browser_id_, *page->dict_); | 607 std::string local_id = BuildUniqueTargetId(page->browser_id_, |
| 608 page->dict_.get()); |
| 609 std::string target_path = GetTargetPath(page->dict_.get()); |
| 610 std::string type = GetStringProperty(page->dict_.get(), "type"); |
| 611 scoped_refptr<DevToolsAgentHost> host = |
| 612 AgentHostDelegate::GetOrCreateAgentHost(this, page->browser_id_, local_id, |
| 613 target_path, type, |
| 614 page->dict_.get()); |
| 615 return new DevToolsTargetImpl(host); |
612 } | 616 } |
613 | 617 |
614 void DevToolsAndroidBridge::SendJsonRequest( | 618 void DevToolsAndroidBridge::SendJsonRequest( |
615 const BrowserId& browser_id, | 619 const BrowserId& browser_id, |
616 const std::string& request, | 620 const std::string& request, |
617 const JsonRequestCallback& callback) { | 621 const JsonRequestCallback& callback) { |
618 scoped_refptr<AndroidDeviceManager::Device> device( | 622 scoped_refptr<AndroidDeviceManager::Device> device( |
619 FindDevice(browser_id.first)); | 623 FindDevice(browser_id.first)); |
620 if (!device.get()) { | 624 if (!device.get()) { |
621 callback.Run(net::ERR_FAILED, std::string()); | 625 callback.Run(net::ERR_FAILED, std::string()); |
(...skipping 21 matching lines...) Expand all Loading... |
643 device, browser_id.second, target_path, | 647 device, browser_id.second, target_path, |
644 DevToolsProtocol::SerializeCommand(1, method, std::move(params)), | 648 DevToolsProtocol::SerializeCommand(1, method, std::move(params)), |
645 callback); | 649 callback); |
646 } | 650 } |
647 | 651 |
648 scoped_refptr<content::DevToolsAgentHost> | 652 scoped_refptr<content::DevToolsAgentHost> |
649 DevToolsAndroidBridge::GetBrowserAgentHost( | 653 DevToolsAndroidBridge::GetBrowserAgentHost( |
650 scoped_refptr<RemoteBrowser> browser) { | 654 scoped_refptr<RemoteBrowser> browser) { |
651 return AgentHostDelegate::GetOrCreateAgentHost( | 655 return AgentHostDelegate::GetOrCreateAgentHost( |
652 this, | 656 this, |
| 657 browser->browser_id_, |
653 "adb:" + browser->serial() + ":" + browser->socket(), | 658 "adb:" + browser->serial() + ":" + browser->socket(), |
654 browser->browser_id_, | 659 kBrowserTargetSocket, DevToolsAgentHost::kTypeBrowser, nullptr); |
655 kBrowserTargetSocket); | |
656 } | 660 } |
657 | 661 |
658 void DevToolsAndroidBridge::SendJsonRequest( | 662 void DevToolsAndroidBridge::SendJsonRequest( |
659 const std::string& browser_id_str, | 663 const std::string& browser_id_str, |
660 const std::string& url, | 664 const std::string& url, |
661 const JsonRequestCallback& callback) { | 665 const JsonRequestCallback& callback) { |
662 BrowserId browser_id; | 666 BrowserId browser_id; |
663 if (!BrowserIdFromString(browser_id_str, &browser_id)) { | 667 if (!BrowserIdFromString(browser_id_str, &browser_id)) { |
664 callback.Run(net::ERR_FAILED, std::string()); | 668 callback.Run(net::ERR_FAILED, std::string()); |
665 return; | 669 return; |
(...skipping 316 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
982 StopDeviceListPolling(); | 986 StopDeviceListPolling(); |
983 StartDeviceListPolling(); | 987 StartDeviceListPolling(); |
984 } | 988 } |
985 } | 989 } |
986 | 990 |
987 void DevToolsAndroidBridge::set_tcp_provider_callback_for_test( | 991 void DevToolsAndroidBridge::set_tcp_provider_callback_for_test( |
988 TCPProviderCallback callback) { | 992 TCPProviderCallback callback) { |
989 tcp_provider_callback_ = callback; | 993 tcp_provider_callback_ = callback; |
990 CreateDeviceProviders(); | 994 CreateDeviceProviders(); |
991 } | 995 } |
OLD | NEW |