OLD | NEW |
1 // Copyright (c) 2012 The Chromium Authors. All rights reserved. | 1 // Copyright (c) 2012 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 "extensions/browser/event_router.h" | 5 #include "extensions/browser/event_router.h" |
6 | 6 |
7 #include <utility> | 7 #include <utility> |
8 | 8 |
9 #include "base/atomic_sequence_num.h" | 9 #include "base/atomic_sequence_num.h" |
10 #include "base/bind.h" | 10 #include "base/bind.h" |
11 #include "base/command_line.h" | 11 #include "base/command_line.h" |
12 #include "base/lazy_instance.h" | 12 #include "base/lazy_instance.h" |
13 #include "base/message_loop/message_loop.h" | 13 #include "base/message_loop/message_loop.h" |
14 #include "base/stl_util.h" | 14 #include "base/stl_util.h" |
15 #include "base/values.h" | 15 #include "base/values.h" |
16 #include "content/public/browser/notification_service.h" | 16 #include "content/public/browser/notification_service.h" |
17 #include "content/public/browser/render_process_host.h" | 17 #include "content/public/browser/render_process_host.h" |
18 #include "extensions/browser/api_activity_monitor.h" | 18 #include "extensions/browser/api_activity_monitor.h" |
19 #include "extensions/browser/event_router_factory.h" | 19 #include "extensions/browser/event_router_factory.h" |
20 #include "extensions/browser/extension_host.h" | 20 #include "extensions/browser/extension_host.h" |
21 #include "extensions/browser/extension_prefs.h" | 21 #include "extensions/browser/extension_prefs.h" |
22 #include "extensions/browser/extension_registry.h" | 22 #include "extensions/browser/extension_registry.h" |
23 #include "extensions/browser/extension_system.h" | 23 #include "extensions/browser/extension_system.h" |
24 #include "extensions/browser/extensions_browser_client.h" | 24 #include "extensions/browser/extensions_browser_client.h" |
25 #include "extensions/browser/lazy_background_task_queue.h" | 25 #include "extensions/browser/lazy_background_task_queue.h" |
26 #include "extensions/browser/notification_types.h" | 26 #include "extensions/browser/notification_types.h" |
27 #include "extensions/browser/process_manager.h" | 27 #include "extensions/browser/process_manager.h" |
28 #include "extensions/browser/process_map.h" | 28 #include "extensions/browser/process_map.h" |
| 29 #include "extensions/common/constants.h" |
29 #include "extensions/common/extension.h" | 30 #include "extensions/common/extension.h" |
30 #include "extensions/common/extension_api.h" | 31 #include "extensions/common/extension_api.h" |
31 #include "extensions/common/extension_messages.h" | 32 #include "extensions/common/extension_messages.h" |
32 #include "extensions/common/extension_urls.h" | 33 #include "extensions/common/extension_urls.h" |
33 #include "extensions/common/features/feature.h" | 34 #include "extensions/common/features/feature.h" |
34 #include "extensions/common/features/feature_provider.h" | 35 #include "extensions/common/features/feature_provider.h" |
35 #include "extensions/common/manifest_handlers/background_info.h" | 36 #include "extensions/common/manifest_handlers/background_info.h" |
36 #include "extensions/common/manifest_handlers/incognito_info.h" | 37 #include "extensions/common/manifest_handlers/incognito_info.h" |
37 #include "extensions/common/permissions/permissions_data.h" | 38 #include "extensions/common/permissions/permissions_data.h" |
38 | 39 |
(...skipping 101 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
140 return EventRouterFactory::GetForBrowserContext(browser_context); | 141 return EventRouterFactory::GetForBrowserContext(browser_context); |
141 } | 142 } |
142 | 143 |
143 // static | 144 // static |
144 std::string EventRouter::GetBaseEventName(const std::string& full_event_name) { | 145 std::string EventRouter::GetBaseEventName(const std::string& full_event_name) { |
145 size_t slash_sep = full_event_name.find('/'); | 146 size_t slash_sep = full_event_name.find('/'); |
146 return full_event_name.substr(0, slash_sep); | 147 return full_event_name.substr(0, slash_sep); |
147 } | 148 } |
148 | 149 |
149 // static | 150 // static |
150 void EventRouter::DispatchEvent(IPC::Sender* ipc_sender, | 151 void EventRouter::DispatchEventToSender(IPC::Sender* ipc_sender, |
151 void* browser_context_id, | 152 void* browser_context_id, |
152 const std::string& extension_id, | 153 const std::string& extension_id, |
153 const std::string& event_name, | 154 events::HistogramValue histogram_value, |
154 scoped_ptr<ListValue> event_args, | 155 const std::string& event_name, |
155 UserGestureState user_gesture, | 156 scoped_ptr<ListValue> event_args, |
156 const EventFilteringInfo& info) { | 157 UserGestureState user_gesture, |
| 158 const EventFilteringInfo& info) { |
157 int event_id = g_extension_event_id.GetNext(); | 159 int event_id = g_extension_event_id.GetNext(); |
158 | 160 |
159 if (!BrowserThread::CurrentlyOn(BrowserThread::UI)) { | 161 if (BrowserThread::CurrentlyOn(BrowserThread::UI)) { |
| 162 DoDispatchEventToSenderBookkeepingOnUI(browser_context_id, extension_id, |
| 163 event_id, histogram_value, |
| 164 event_name); |
| 165 } else { |
160 // This is called from WebRequest API. | 166 // This is called from WebRequest API. |
161 // TODO(lazyboy): Skip this entirely: http://crbug.com/488747. | 167 // TODO(lazyboy): Skip this entirely: http://crbug.com/488747. |
162 BrowserThread::PostTask( | 168 BrowserThread::PostTask( |
163 BrowserThread::UI, FROM_HERE, | 169 BrowserThread::UI, FROM_HERE, |
164 base::Bind(&EventRouter::IncrementInFlightEventsOnUI, | 170 base::Bind(&EventRouter::DoDispatchEventToSenderBookkeepingOnUI, |
165 browser_context_id, extension_id, event_id, event_name)); | 171 browser_context_id, extension_id, event_id, histogram_value, |
166 } else { | 172 event_name)); |
167 IncrementInFlightEventsOnUI(browser_context_id, extension_id, event_id, | |
168 event_name); | |
169 } | 173 } |
170 | 174 |
171 DispatchExtensionMessage(ipc_sender, browser_context_id, extension_id, | 175 DispatchExtensionMessage(ipc_sender, browser_context_id, extension_id, |
172 event_id, event_name, event_args.get(), user_gesture, | 176 event_id, event_name, event_args.get(), user_gesture, |
173 info); | 177 info); |
174 } | 178 } |
175 | 179 |
176 EventRouter::EventRouter(BrowserContext* browser_context, | 180 EventRouter::EventRouter(BrowserContext* browser_context, |
177 ExtensionPrefs* extension_prefs) | 181 ExtensionPrefs* extension_prefs) |
178 : browser_context_(browser_context), | 182 : browser_context_(browser_context), |
(...skipping 314 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
493 listeners_.GetEventListeners(*event)); | 497 listeners_.GetEventListeners(*event)); |
494 | 498 |
495 std::set<EventDispatchIdentifier> already_dispatched; | 499 std::set<EventDispatchIdentifier> already_dispatched; |
496 | 500 |
497 // We dispatch events for lazy background pages first because attempting to do | 501 // We dispatch events for lazy background pages first because attempting to do |
498 // so will cause those that are being suspended to cancel that suspension. | 502 // so will cause those that are being suspended to cancel that suspension. |
499 // As canceling a suspension entails sending an event to the affected | 503 // As canceling a suspension entails sending an event to the affected |
500 // background page, and as that event needs to be delivered before we dispatch | 504 // background page, and as that event needs to be delivered before we dispatch |
501 // the event we are dispatching here, we dispatch to the lazy listeners here | 505 // the event we are dispatching here, we dispatch to the lazy listeners here |
502 // first. | 506 // first. |
503 for (std::set<const EventListener*>::iterator it = listeners.begin(); | 507 for (const EventListener* listener : listeners) { |
504 it != listeners.end(); it++) { | |
505 const EventListener* listener = *it; | |
506 if (restrict_to_extension_id.empty() || | 508 if (restrict_to_extension_id.empty() || |
507 restrict_to_extension_id == listener->extension_id()) { | 509 restrict_to_extension_id == listener->extension_id()) { |
508 if (listener->IsLazy()) { | 510 if (listener->IsLazy()) { |
509 DispatchLazyEvent(listener->extension_id(), event, &already_dispatched, | 511 DispatchLazyEvent(listener->extension_id(), event, &already_dispatched, |
510 listener->filter()); | 512 listener->filter()); |
511 } | 513 } |
512 } | 514 } |
513 } | 515 } |
514 | 516 |
515 for (std::set<const EventListener*>::iterator it = listeners.begin(); | 517 for (const EventListener* listener : listeners) { |
516 it != listeners.end(); it++) { | |
517 const EventListener* listener = *it; | |
518 if (restrict_to_extension_id.empty() || | 518 if (restrict_to_extension_id.empty() || |
519 restrict_to_extension_id == listener->extension_id()) { | 519 restrict_to_extension_id == listener->extension_id()) { |
520 if (listener->process()) { | 520 if (listener->process()) { |
521 EventDispatchIdentifier dispatch_id(listener->GetBrowserContext(), | 521 EventDispatchIdentifier dispatch_id(listener->GetBrowserContext(), |
522 listener->extension_id()); | 522 listener->extension_id()); |
523 if (!ContainsKey(already_dispatched, dispatch_id)) { | 523 if (!ContainsKey(already_dispatched, dispatch_id)) { |
524 DispatchEventToProcess(listener->extension_id(), | 524 DispatchEventToProcess(listener->extension_id(), |
525 listener->listener_url(), listener->process(), | 525 listener->listener_url(), listener->process(), |
526 event, listener->filter()); | 526 event, listener->filter(), |
| 527 false /* did_enqueue */); |
527 } | 528 } |
528 } | 529 } |
529 } | 530 } |
530 } | 531 } |
531 } | 532 } |
532 | 533 |
533 void EventRouter::DispatchLazyEvent( | 534 void EventRouter::DispatchLazyEvent( |
534 const std::string& extension_id, | 535 const std::string& extension_id, |
535 const linked_ptr<Event>& event, | 536 const linked_ptr<Event>& event, |
536 std::set<EventDispatchIdentifier>* already_dispatched, | 537 std::set<EventDispatchIdentifier>* already_dispatched, |
(...skipping 23 matching lines...) Expand all Loading... |
560 std::make_pair(incognito_context, extension_id)); | 561 std::make_pair(incognito_context, extension_id)); |
561 } | 562 } |
562 } | 563 } |
563 } | 564 } |
564 | 565 |
565 void EventRouter::DispatchEventToProcess( | 566 void EventRouter::DispatchEventToProcess( |
566 const std::string& extension_id, | 567 const std::string& extension_id, |
567 const GURL& listener_url, | 568 const GURL& listener_url, |
568 content::RenderProcessHost* process, | 569 content::RenderProcessHost* process, |
569 const linked_ptr<Event>& event, | 570 const linked_ptr<Event>& event, |
570 const base::DictionaryValue* listener_filter) { | 571 const base::DictionaryValue* listener_filter, |
| 572 bool did_enqueue) { |
571 BrowserContext* listener_context = process->GetBrowserContext(); | 573 BrowserContext* listener_context = process->GetBrowserContext(); |
572 ProcessMap* process_map = ProcessMap::Get(listener_context); | 574 ProcessMap* process_map = ProcessMap::Get(listener_context); |
573 | 575 |
574 // NOTE: |extension| being NULL does not necessarily imply that this event | 576 // NOTE: |extension| being NULL does not necessarily imply that this event |
575 // shouldn't be dispatched. Events can be dispatched to WebUI and webviews as | 577 // shouldn't be dispatched. Events can be dispatched to WebUI and webviews as |
576 // well. It all depends on what GetMostLikelyContextType returns. | 578 // well. It all depends on what GetMostLikelyContextType returns. |
577 const Extension* extension = | 579 const Extension* extension = |
578 ExtensionRegistry::Get(browser_context_)->enabled_extensions().GetByID( | 580 ExtensionRegistry::Get(browser_context_)->enabled_extensions().GetByID( |
579 extension_id); | 581 extension_id); |
580 | 582 |
(...skipping 49 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
630 listener_filter)) { | 632 listener_filter)) { |
631 return; | 633 return; |
632 } | 634 } |
633 | 635 |
634 int event_id = g_extension_event_id.GetNext(); | 636 int event_id = g_extension_event_id.GetNext(); |
635 DispatchExtensionMessage(process, listener_context, extension_id, event_id, | 637 DispatchExtensionMessage(process, listener_context, extension_id, event_id, |
636 event->event_name, event->event_args.get(), | 638 event->event_name, event->event_args.get(), |
637 event->user_gesture, event->filter_info); | 639 event->user_gesture, event->filter_info); |
638 | 640 |
639 if (extension) { | 641 if (extension) { |
| 642 ReportEvent(event->histogram_value, extension, did_enqueue); |
640 IncrementInFlightEvents(listener_context, extension, event_id, | 643 IncrementInFlightEvents(listener_context, extension, event_id, |
641 event->event_name); | 644 event->event_name); |
642 } | 645 } |
643 } | 646 } |
644 | 647 |
645 bool EventRouter::CanDispatchEventToBrowserContext( | 648 bool EventRouter::CanDispatchEventToBrowserContext( |
646 BrowserContext* context, | 649 BrowserContext* context, |
647 const Extension* extension, | 650 const Extension* extension, |
648 const linked_ptr<Event>& event) { | 651 const linked_ptr<Event>& event) { |
649 // Is this event from a different browser context than the renderer (ie, an | 652 // Is this event from a different browser context than the renderer (ie, an |
(...skipping 36 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
686 queue->AddPendingTask(context, extension->id(), | 689 queue->AddPendingTask(context, extension->id(), |
687 base::Bind(&EventRouter::DispatchPendingEvent, | 690 base::Bind(&EventRouter::DispatchPendingEvent, |
688 base::Unretained(this), dispatched_event)); | 691 base::Unretained(this), dispatched_event)); |
689 return true; | 692 return true; |
690 } | 693 } |
691 | 694 |
692 return false; | 695 return false; |
693 } | 696 } |
694 | 697 |
695 // static | 698 // static |
696 void EventRouter::IncrementInFlightEventsOnUI(void* browser_context_id, | 699 void EventRouter::DoDispatchEventToSenderBookkeepingOnUI( |
697 const std::string& extension_id, | 700 void* browser_context_id, |
698 int event_id, | 701 const std::string& extension_id, |
699 const std::string& event_name) { | 702 int event_id, |
| 703 events::HistogramValue histogram_value, |
| 704 const std::string& event_name) { |
700 DCHECK_CURRENTLY_ON(BrowserThread::UI); | 705 DCHECK_CURRENTLY_ON(BrowserThread::UI); |
701 BrowserContext* browser_context = | 706 BrowserContext* browser_context = |
702 reinterpret_cast<BrowserContext*>(browser_context_id); | 707 reinterpret_cast<BrowserContext*>(browser_context_id); |
703 if (!ExtensionsBrowserClient::Get()->IsValidContext(browser_context)) | 708 if (!ExtensionsBrowserClient::Get()->IsValidContext(browser_context)) |
704 return; | 709 return; |
705 EventRouter* event_router = EventRouter::Get(browser_context); | |
706 if (!event_router) | |
707 return; | |
708 const Extension* extension = | 710 const Extension* extension = |
709 ExtensionRegistry::Get(browser_context)->enabled_extensions().GetByID( | 711 ExtensionRegistry::Get(browser_context)->enabled_extensions().GetByID( |
710 extension_id); | 712 extension_id); |
711 if (!extension) | 713 if (!extension) |
712 return; | 714 return; |
| 715 EventRouter* event_router = EventRouter::Get(browser_context); |
713 event_router->IncrementInFlightEvents(browser_context, extension, event_id, | 716 event_router->IncrementInFlightEvents(browser_context, extension, event_id, |
714 event_name); | 717 event_name); |
| 718 event_router->ReportEvent(histogram_value, extension, |
| 719 false /* did_enqueue */); |
715 } | 720 } |
716 | 721 |
717 void EventRouter::IncrementInFlightEvents(BrowserContext* context, | 722 void EventRouter::IncrementInFlightEvents(BrowserContext* context, |
718 const Extension* extension, | 723 const Extension* extension, |
719 int event_id, | 724 int event_id, |
720 const std::string& event_name) { | 725 const std::string& event_name) { |
721 // TODO(chirantan): Turn this on once crbug.com/464513 is fixed. | 726 // TODO(chirantan): Turn this on once crbug.com/464513 is fixed. |
722 // DCHECK_CURRENTLY_ON(BrowserThread::UI); | 727 // DCHECK_CURRENTLY_ON(BrowserThread::UI); |
723 | 728 |
724 // Only increment in-flight events if the lazy background page is active, | 729 // Only increment in-flight events if the lazy background page is active, |
(...skipping 15 matching lines...) Expand all Loading... |
740 // The event ACK is routed to the background host, so this should never be | 745 // The event ACK is routed to the background host, so this should never be |
741 // NULL. | 746 // NULL. |
742 CHECK(host); | 747 CHECK(host); |
743 // TODO(mpcomplete): We should never get this message unless | 748 // TODO(mpcomplete): We should never get this message unless |
744 // HasLazyBackgroundPage is true. Find out why we're getting it anyway. | 749 // HasLazyBackgroundPage is true. Find out why we're getting it anyway. |
745 if (host->extension() && | 750 if (host->extension() && |
746 BackgroundInfo::HasLazyBackgroundPage(host->extension())) | 751 BackgroundInfo::HasLazyBackgroundPage(host->extension())) |
747 pm->DecrementLazyKeepaliveCount(host->extension()); | 752 pm->DecrementLazyKeepaliveCount(host->extension()); |
748 } | 753 } |
749 | 754 |
| 755 void EventRouter::ReportEvent(events::HistogramValue histogram_value, |
| 756 const Extension* extension, |
| 757 bool did_enqueue) { |
| 758 DCHECK_CURRENTLY_ON(content::BrowserThread::UI); |
| 759 |
| 760 // TODO(kalman): UMA for dispatched event. |
| 761 // TODO(kalman): UMA specifically for component extensions. |
| 762 |
| 763 // Note: for these, all we know is that the extension *has* a persistent or |
| 764 // event page, not that the event is being dispatched *to* such a page. |
| 765 // |
| 766 // However, this is an academic distinction, since extensions with any |
| 767 // background page have that background page running (or in the case of |
| 768 // dormant event pages, must be started) regardless of where the event is |
| 769 // being dispatched. Events are dispatched to a *process* not a *frame*. |
| 770 if (BackgroundInfo::HasPersistentBackgroundPage(extension)) { |
| 771 // TODO(kalman): UMA for dispatched event to an extension with a persistent |
| 772 // background page. |
| 773 } else if (BackgroundInfo::HasLazyBackgroundPage(extension)) { |
| 774 if (did_enqueue) { |
| 775 // TODO(kalman): UMA for dispatched event to an extension with an event |
| 776 // page, and the event page was woken up to do so. |
| 777 } else { |
| 778 // TODO(kalman): UMA for dispatched event to an extension with an event |
| 779 // page that was already running. |
| 780 } |
| 781 } |
| 782 } |
| 783 |
750 void EventRouter::DispatchPendingEvent(const linked_ptr<Event>& event, | 784 void EventRouter::DispatchPendingEvent(const linked_ptr<Event>& event, |
751 ExtensionHost* host) { | 785 ExtensionHost* host) { |
752 if (!host) | 786 if (!host) |
753 return; | 787 return; |
754 | 788 |
755 if (listeners_.HasProcessListener(host->render_process_host(), | 789 if (listeners_.HasProcessListener(host->render_process_host(), |
756 host->extension()->id())) { | 790 host->extension()->id())) { |
757 // URL events cannot be lazy therefore can't be pending, hence the GURL(). | 791 DispatchEventToProcess(host->extension()->id(), host->GetURL(), |
758 DispatchEventToProcess(host->extension()->id(), GURL(), | 792 host->render_process_host(), event, nullptr, |
759 host->render_process_host(), event, nullptr); | 793 true /* did_enqueue */); |
760 } | 794 } |
761 } | 795 } |
762 | 796 |
763 void EventRouter::Observe(int type, | 797 void EventRouter::Observe(int type, |
764 const content::NotificationSource& source, | 798 const content::NotificationSource& source, |
765 const content::NotificationDetails& details) { | 799 const content::NotificationDetails& details) { |
766 switch (type) { | 800 switch (type) { |
767 case extensions::NOTIFICATION_EXTENSION_ENABLED: { | 801 case extensions::NOTIFICATION_EXTENSION_ENABLED: { |
768 // If the extension has a lazy background page, make sure it gets loaded | 802 // If the extension has a lazy background page, make sure it gets loaded |
769 // to register the events the extension is interested in. | 803 // to register the events the extension is interested in. |
(...skipping 85 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
855 const std::string& extension_id, | 889 const std::string& extension_id, |
856 const GURL& listener_url, | 890 const GURL& listener_url, |
857 content::BrowserContext* browser_context) | 891 content::BrowserContext* browser_context) |
858 : event_name(event_name), | 892 : event_name(event_name), |
859 extension_id(extension_id), | 893 extension_id(extension_id), |
860 listener_url(listener_url), | 894 listener_url(listener_url), |
861 browser_context(browser_context) { | 895 browser_context(browser_context) { |
862 } | 896 } |
863 | 897 |
864 } // namespace extensions | 898 } // namespace extensions |
OLD | NEW |