| 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 <stddef.h> | 7 #include <stddef.h> |
| 8 | 8 |
| 9 #include <utility> | 9 #include <utility> |
| 10 | 10 |
| (...skipping 54 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
| 65 BrowserContext* context = static_cast<BrowserContext*>(browser_context_id); | 65 BrowserContext* context = static_cast<BrowserContext*>(browser_context_id); |
| 66 activity_monitor::OnApiEventDispatched(context, extension_id, event_name, | 66 activity_monitor::OnApiEventDispatched(context, extension_id, event_name, |
| 67 args); | 67 args); |
| 68 } | 68 } |
| 69 | 69 |
| 70 // A global identifier used to distinguish extension events. | 70 // A global identifier used to distinguish extension events. |
| 71 base::StaticAtomicSequenceNumber g_extension_event_id; | 71 base::StaticAtomicSequenceNumber g_extension_event_id; |
| 72 | 72 |
| 73 } // namespace | 73 } // namespace |
| 74 | 74 |
| 75 const char EventRouter::kRegisteredEvents[] = "events"; | 75 const char EventRouter::kRegisteredLazyEvents[] = "events"; |
| 76 const char EventRouter::kRegisteredServiceWorkerEvents[] = |
| 77 "serviceworkerevents"; |
| 76 | 78 |
| 77 // static | 79 // static |
| 78 void EventRouter::DispatchExtensionMessage(IPC::Sender* ipc_sender, | 80 void EventRouter::DispatchExtensionMessage(IPC::Sender* ipc_sender, |
| 81 int worker_thread_id, |
| 79 void* browser_context_id, | 82 void* browser_context_id, |
| 80 const std::string& extension_id, | 83 const std::string& extension_id, |
| 81 int event_id, | 84 int event_id, |
| 82 const std::string& event_name, | 85 const std::string& event_name, |
| 83 ListValue* event_args, | 86 ListValue* event_args, |
| 84 UserGestureState user_gesture, | 87 UserGestureState user_gesture, |
| 85 const EventFilteringInfo& info) { | 88 const EventFilteringInfo& info) { |
| 86 NotifyEventDispatched(browser_context_id, extension_id, event_name, | 89 NotifyEventDispatched(browser_context_id, extension_id, event_name, |
| 87 *event_args); | 90 *event_args); |
| 88 | 91 |
| 89 ExtensionMsg_DispatchEvent_Params params; | 92 ExtensionMsg_DispatchEvent_Params params; |
| 93 params.worker_thread_id = worker_thread_id; |
| 90 params.extension_id = extension_id; | 94 params.extension_id = extension_id; |
| 91 params.event_name = event_name; | 95 params.event_name = event_name; |
| 92 params.event_id = event_id; | 96 params.event_id = event_id; |
| 93 params.is_user_gesture = user_gesture == USER_GESTURE_ENABLED; | 97 params.is_user_gesture = user_gesture == USER_GESTURE_ENABLED; |
| 94 params.filtering_info.Swap(info.AsValue().get()); | 98 params.filtering_info.Swap(info.AsValue().get()); |
| 95 | 99 |
| 96 ipc_sender->Send(new ExtensionMsg_DispatchEvent(params, *event_args)); | 100 ipc_sender->Send(new ExtensionMsg_DispatchEvent(params, *event_args)); |
| 97 } | 101 } |
| 98 | 102 |
| 99 // static | 103 // static |
| (...skipping 25 matching lines...) Expand all Loading... |
| 125 } else { | 129 } else { |
| 126 // This is called from WebRequest API. | 130 // This is called from WebRequest API. |
| 127 // TODO(lazyboy): Skip this entirely: http://crbug.com/488747. | 131 // TODO(lazyboy): Skip this entirely: http://crbug.com/488747. |
| 128 BrowserThread::PostTask( | 132 BrowserThread::PostTask( |
| 129 BrowserThread::UI, FROM_HERE, | 133 BrowserThread::UI, FROM_HERE, |
| 130 base::Bind(&EventRouter::DoDispatchEventToSenderBookkeepingOnUI, | 134 base::Bind(&EventRouter::DoDispatchEventToSenderBookkeepingOnUI, |
| 131 browser_context_id, extension_id, event_id, histogram_value, | 135 browser_context_id, extension_id, event_id, histogram_value, |
| 132 event_name)); | 136 event_name)); |
| 133 } | 137 } |
| 134 | 138 |
| 135 DispatchExtensionMessage(ipc_sender, browser_context_id, extension_id, | 139 DispatchExtensionMessage(ipc_sender, |
| 140 // TODO(lazyboy): |kNonWorkerThreadId| means these |
| 141 // will not work for extension SW. |
| 142 kNonWorkerThreadId, browser_context_id, extension_id, |
| 136 event_id, event_name, event_args.get(), user_gesture, | 143 event_id, event_name, event_args.get(), user_gesture, |
| 137 info); | 144 info); |
| 138 } | 145 } |
| 139 | 146 |
| 140 EventRouter::EventRouter(BrowserContext* browser_context, | 147 EventRouter::EventRouter(BrowserContext* browser_context, |
| 141 ExtensionPrefs* extension_prefs) | 148 ExtensionPrefs* extension_prefs) |
| 142 : browser_context_(browser_context), | 149 : browser_context_(browser_context), |
| 143 extension_prefs_(extension_prefs), | 150 extension_prefs_(extension_prefs), |
| 144 extension_registry_observer_(this), | 151 extension_registry_observer_(this), |
| 145 listeners_(this) { | 152 listeners_(this) { |
| 146 registrar_.Add(this, | 153 registrar_.Add(this, |
| 147 extensions::NOTIFICATION_EXTENSION_ENABLED, | 154 extensions::NOTIFICATION_EXTENSION_ENABLED, |
| 148 content::Source<BrowserContext>(browser_context_)); | 155 content::Source<BrowserContext>(browser_context_)); |
| 149 extension_registry_observer_.Add(ExtensionRegistry::Get(browser_context_)); | 156 extension_registry_observer_.Add(ExtensionRegistry::Get(browser_context_)); |
| 150 } | 157 } |
| 151 | 158 |
| 152 EventRouter::~EventRouter() { | 159 EventRouter::~EventRouter() { |
| 153 for (auto* process : observed_process_set_) | 160 for (auto* process : observed_process_set_) |
| 154 process->RemoveObserver(this); | 161 process->RemoveObserver(this); |
| 155 } | 162 } |
| 156 | 163 |
| 157 void EventRouter::AddEventListener(const std::string& event_name, | 164 void EventRouter::AddEventListener(const std::string& event_name, |
| 158 content::RenderProcessHost* process, | 165 content::RenderProcessHost* process, |
| 159 const std::string& extension_id) { | 166 const std::string& extension_id) { |
| 160 listeners_.AddListener(EventListener::ForExtension( | 167 listeners_.AddListener( |
| 161 event_name, extension_id, process, std::unique_ptr<DictionaryValue>())); | 168 EventListener::ForExtension(event_name, extension_id, process, nullptr)); |
| 169 } |
| 170 |
| 171 void EventRouter::AddServiceWorkerEventListener( |
| 172 const std::string& event_name, |
| 173 content::RenderProcessHost* process, |
| 174 const ExtensionId& extension_id, |
| 175 int worker_thread_id) { |
| 176 listeners_.AddListener(EventListener::ForExtensionServiceWorker( |
| 177 event_name, extension_id, process, worker_thread_id, nullptr)); |
| 162 } | 178 } |
| 163 | 179 |
| 164 void EventRouter::RemoveEventListener(const std::string& event_name, | 180 void EventRouter::RemoveEventListener(const std::string& event_name, |
| 165 content::RenderProcessHost* process, | 181 content::RenderProcessHost* process, |
| 166 const std::string& extension_id) { | 182 const std::string& extension_id) { |
| 167 std::unique_ptr<EventListener> listener = EventListener::ForExtension( | 183 std::unique_ptr<EventListener> listener = |
| 168 event_name, extension_id, process, std::unique_ptr<DictionaryValue>()); | 184 EventListener::ForExtension(event_name, extension_id, process, nullptr); |
| 185 listeners_.RemoveListener(listener.get()); |
| 186 } |
| 187 |
| 188 void EventRouter::RemoveServiceWorkerEventListener( |
| 189 const std::string& event_name, |
| 190 content::RenderProcessHost* process, |
| 191 const ExtensionId& extension_id, |
| 192 int worker_thread_id) { |
| 193 std::unique_ptr<EventListener> listener = |
| 194 EventListener::ForExtensionServiceWorker( |
| 195 event_name, extension_id, process, worker_thread_id, nullptr); |
| 169 listeners_.RemoveListener(listener.get()); | 196 listeners_.RemoveListener(listener.get()); |
| 170 } | 197 } |
| 171 | 198 |
| 172 void EventRouter::AddEventListenerForURL(const std::string& event_name, | 199 void EventRouter::AddEventListenerForURL(const std::string& event_name, |
| 173 content::RenderProcessHost* process, | 200 content::RenderProcessHost* process, |
| 174 const GURL& listener_url) { | 201 const GURL& listener_url) { |
| 175 listeners_.AddListener(EventListener::ForURL( | 202 listeners_.AddListener( |
| 176 event_name, listener_url, process, std::unique_ptr<DictionaryValue>())); | 203 EventListener::ForURL(event_name, listener_url, process, nullptr)); |
| 177 } | 204 } |
| 178 | 205 |
| 179 void EventRouter::RemoveEventListenerForURL(const std::string& event_name, | 206 void EventRouter::RemoveEventListenerForURL(const std::string& event_name, |
| 180 content::RenderProcessHost* process, | 207 content::RenderProcessHost* process, |
| 181 const GURL& listener_url) { | 208 const GURL& listener_url) { |
| 182 std::unique_ptr<EventListener> listener = EventListener::ForURL( | 209 std::unique_ptr<EventListener> listener = |
| 183 event_name, listener_url, process, std::unique_ptr<DictionaryValue>()); | 210 EventListener::ForURL(event_name, listener_url, process, nullptr); |
| 184 listeners_.RemoveListener(listener.get()); | 211 listeners_.RemoveListener(listener.get()); |
| 185 } | 212 } |
| 186 | 213 |
| 187 void EventRouter::RegisterObserver(Observer* observer, | 214 void EventRouter::RegisterObserver(Observer* observer, |
| 188 const std::string& event_name) { | 215 const std::string& event_name) { |
| 189 // Observing sub-event names like "foo.onBar/123" is not allowed. | 216 // Observing sub-event names like "foo.onBar/123" is not allowed. |
| 190 DCHECK(event_name.find('/') == std::string::npos); | 217 DCHECK(event_name.find('/') == std::string::npos); |
| 191 observers_[event_name] = observer; | 218 observers_[event_name] = observer; |
| 192 } | 219 } |
| 193 | 220 |
| (...skipping 43 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
| 237 host->RemoveObserver(this); | 264 host->RemoveObserver(this); |
| 238 } | 265 } |
| 239 | 266 |
| 240 void EventRouter::RenderProcessHostDestroyed(content::RenderProcessHost* host) { | 267 void EventRouter::RenderProcessHostDestroyed(content::RenderProcessHost* host) { |
| 241 listeners_.RemoveListenersForProcess(host); | 268 listeners_.RemoveListenersForProcess(host); |
| 242 observed_process_set_.erase(host); | 269 observed_process_set_.erase(host); |
| 243 host->RemoveObserver(this); | 270 host->RemoveObserver(this); |
| 244 } | 271 } |
| 245 | 272 |
| 246 void EventRouter::AddLazyEventListener(const std::string& event_name, | 273 void EventRouter::AddLazyEventListener(const std::string& event_name, |
| 247 const std::string& extension_id) { | 274 const ExtensionId& extension_id) { |
| 248 bool is_new = listeners_.AddListener(EventListener::ForExtension( | 275 AddLazyEventListenerImpl(event_name, extension_id, kNonWorkerThreadId); |
| 249 event_name, extension_id, NULL, std::unique_ptr<DictionaryValue>())); | |
| 250 | |
| 251 if (is_new) { | |
| 252 std::set<std::string> events = GetRegisteredEvents(extension_id); | |
| 253 bool prefs_is_new = events.insert(event_name).second; | |
| 254 if (prefs_is_new) | |
| 255 SetRegisteredEvents(extension_id, events); | |
| 256 } | |
| 257 } | 276 } |
| 258 | 277 |
| 259 void EventRouter::RemoveLazyEventListener(const std::string& event_name, | 278 void EventRouter::RemoveLazyEventListener(const std::string& event_name, |
| 260 const std::string& extension_id) { | 279 const ExtensionId& extension_id) { |
| 261 std::unique_ptr<EventListener> listener = EventListener::ForExtension( | 280 RemoveLazyEventListenerImpl(event_name, extension_id, kNonWorkerThreadId); |
| 262 event_name, extension_id, NULL, std::unique_ptr<DictionaryValue>()); | |
| 263 bool did_exist = listeners_.RemoveListener(listener.get()); | |
| 264 | |
| 265 if (did_exist) { | |
| 266 std::set<std::string> events = GetRegisteredEvents(extension_id); | |
| 267 bool prefs_did_exist = events.erase(event_name) > 0; | |
| 268 DCHECK(prefs_did_exist); | |
| 269 SetRegisteredEvents(extension_id, events); | |
| 270 } | |
| 271 } | 281 } |
| 272 | 282 |
| 283 void EventRouter::AddLazyServiceWorkerEventListener( |
| 284 const std::string& event_name, |
| 285 const ExtensionId& extension_id, |
| 286 int worker_thread_id) { |
| 287 AddLazyEventListenerImpl(event_name, extension_id, worker_thread_id); |
| 288 } |
| 289 |
| 290 void EventRouter::RemoveLazyServiceWorkerEventListener( |
| 291 const std::string& event_name, |
| 292 const ExtensionId& extension_id, |
| 293 int worker_thread_id) { |
| 294 RemoveLazyEventListenerImpl(event_name, extension_id, worker_thread_id); |
| 295 } |
| 296 |
| 297 // TODO(lazyboy): Support filters for extension SW events. |
| 273 void EventRouter::AddFilteredEventListener(const std::string& event_name, | 298 void EventRouter::AddFilteredEventListener(const std::string& event_name, |
| 274 content::RenderProcessHost* process, | 299 content::RenderProcessHost* process, |
| 275 const std::string& extension_id, | 300 const std::string& extension_id, |
| 276 const base::DictionaryValue& filter, | 301 const base::DictionaryValue& filter, |
| 277 bool add_lazy_listener) { | 302 bool add_lazy_listener) { |
| 278 listeners_.AddListener(EventListener::ForExtension( | 303 listeners_.AddListener(EventListener::ForExtension( |
| 279 event_name, extension_id, process, | 304 event_name, extension_id, process, |
| 280 std::unique_ptr<DictionaryValue>(filter.DeepCopy()))); | 305 std::unique_ptr<DictionaryValue>(filter.DeepCopy()))); |
| 281 | 306 |
| 282 if (!add_lazy_listener) | 307 if (!add_lazy_listener) |
| 283 return; | 308 return; |
| 284 | 309 |
| 285 bool added = listeners_.AddListener(EventListener::ForExtension( | 310 bool added = listeners_.AddListener(EventListener::ForExtension( |
| 286 event_name, extension_id, nullptr, | 311 event_name, extension_id, nullptr, |
| 287 std::unique_ptr<DictionaryValue>(filter.DeepCopy()))); | 312 std::unique_ptr<DictionaryValue>(filter.DeepCopy()))); |
| 288 if (added) | 313 if (added) |
| 289 AddFilterToEvent(event_name, extension_id, &filter); | 314 AddFilterToEvent(event_name, extension_id, &filter); |
| 290 } | 315 } |
| 291 | 316 |
| 317 // TODO(lazyboy): Support filters for extension SW events. |
| 292 void EventRouter::RemoveFilteredEventListener( | 318 void EventRouter::RemoveFilteredEventListener( |
| 293 const std::string& event_name, | 319 const std::string& event_name, |
| 294 content::RenderProcessHost* process, | 320 content::RenderProcessHost* process, |
| 295 const std::string& extension_id, | 321 const std::string& extension_id, |
| 296 const base::DictionaryValue& filter, | 322 const base::DictionaryValue& filter, |
| 297 bool remove_lazy_listener) { | 323 bool remove_lazy_listener) { |
| 298 std::unique_ptr<EventListener> listener = EventListener::ForExtension( | 324 std::unique_ptr<EventListener> listener = EventListener::ForExtension( |
| 299 event_name, extension_id, process, | 325 event_name, extension_id, process, |
| 300 std::unique_ptr<DictionaryValue>(filter.DeepCopy())); | 326 std::unique_ptr<DictionaryValue>(filter.DeepCopy())); |
| 301 | 327 |
| (...skipping 12 matching lines...) Expand all Loading... |
| 314 return listeners_.HasListenerForEvent(event_name); | 340 return listeners_.HasListenerForEvent(event_name); |
| 315 } | 341 } |
| 316 | 342 |
| 317 bool EventRouter::ExtensionHasEventListener( | 343 bool EventRouter::ExtensionHasEventListener( |
| 318 const std::string& extension_id, | 344 const std::string& extension_id, |
| 319 const std::string& event_name) const { | 345 const std::string& event_name) const { |
| 320 return listeners_.HasListenerForExtension(extension_id, event_name); | 346 return listeners_.HasListenerForExtension(extension_id, event_name); |
| 321 } | 347 } |
| 322 | 348 |
| 323 std::set<std::string> EventRouter::GetRegisteredEvents( | 349 std::set<std::string> EventRouter::GetRegisteredEvents( |
| 324 const std::string& extension_id) const { | 350 const std::string& extension_id, |
| 351 RegisteredEventType type) const { |
| 325 std::set<std::string> events; | 352 std::set<std::string> events; |
| 326 const ListValue* events_value = NULL; | 353 const ListValue* events_value = NULL; |
| 327 | 354 |
| 328 if (!extension_prefs_ || | 355 const char* pref_key = type == RegisteredEventType::kLazy |
| 329 !extension_prefs_->ReadPrefAsList( | 356 ? kRegisteredLazyEvents |
| 330 extension_id, kRegisteredEvents, &events_value)) { | 357 : kRegisteredServiceWorkerEvents; |
| 358 if (!extension_prefs_ || !extension_prefs_->ReadPrefAsList( |
| 359 extension_id, pref_key, &events_value)) { |
| 331 return events; | 360 return events; |
| 332 } | 361 } |
| 333 | 362 |
| 334 for (size_t i = 0; i < events_value->GetSize(); ++i) { | 363 for (size_t i = 0; i < events_value->GetSize(); ++i) { |
| 335 std::string event; | 364 std::string event; |
| 336 if (events_value->GetString(i, &event)) | 365 if (events_value->GetString(i, &event)) |
| 337 events.insert(event); | 366 events.insert(event); |
| 338 } | 367 } |
| 339 return events; | 368 return events; |
| 340 } | 369 } |
| 341 | 370 |
| 371 void EventRouter::ClearRegisteredEventsForTest( |
| 372 const ExtensionId& extension_id) { |
| 373 SetRegisteredEvents(extension_id, std::set<std::string>(), |
| 374 RegisteredEventType::kLazy); |
| 375 SetRegisteredEvents(extension_id, std::set<std::string>(), |
| 376 RegisteredEventType::kServiceWorker); |
| 377 } |
| 378 |
| 342 void EventRouter::RemoveFilterFromEvent(const std::string& event_name, | 379 void EventRouter::RemoveFilterFromEvent(const std::string& event_name, |
| 343 const std::string& extension_id, | 380 const std::string& extension_id, |
| 344 const DictionaryValue* filter) { | 381 const DictionaryValue* filter) { |
| 345 ExtensionPrefs::ScopedDictionaryUpdate update( | 382 ExtensionPrefs::ScopedDictionaryUpdate update( |
| 346 extension_prefs_, extension_id, kFilteredEvents); | 383 extension_prefs_, extension_id, kFilteredEvents); |
| 347 auto filtered_events = update.Create(); | 384 auto filtered_events = update.Create(); |
| 348 ListValue* filter_list = NULL; | 385 ListValue* filter_list = NULL; |
| 349 if (!filtered_events || | 386 if (!filtered_events || |
| 350 !filtered_events->GetListWithoutPathExpansion(event_name, &filter_list)) { | 387 !filtered_events->GetListWithoutPathExpansion(event_name, &filter_list)) { |
| 351 return; | 388 return; |
| (...skipping 38 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
| 390 if (!has_listener) | 427 if (!has_listener) |
| 391 RemoveLazyEventListener(event_name, extension_id); | 428 RemoveLazyEventListener(event_name, extension_id); |
| 392 } | 429 } |
| 393 | 430 |
| 394 void EventRouter::DispatchEventImpl(const std::string& restrict_to_extension_id, | 431 void EventRouter::DispatchEventImpl(const std::string& restrict_to_extension_id, |
| 395 const linked_ptr<Event>& event) { | 432 const linked_ptr<Event>& event) { |
| 396 // We don't expect to get events from a completely different browser context. | 433 // We don't expect to get events from a completely different browser context. |
| 397 DCHECK(!event->restrict_to_browser_context || | 434 DCHECK(!event->restrict_to_browser_context || |
| 398 ExtensionsBrowserClient::Get()->IsSameContext( | 435 ExtensionsBrowserClient::Get()->IsSameContext( |
| 399 browser_context_, event->restrict_to_browser_context)); | 436 browser_context_, event->restrict_to_browser_context)); |
| 400 | |
| 401 std::set<const EventListener*> listeners( | 437 std::set<const EventListener*> listeners( |
| 402 listeners_.GetEventListeners(*event)); | 438 listeners_.GetEventListeners(*event)); |
| 403 | 439 |
| 404 std::set<EventDispatchIdentifier> already_dispatched; | 440 std::set<EventDispatchIdentifier> already_dispatched; |
| 405 | 441 |
| 406 // We dispatch events for lazy background pages first because attempting to do | 442 // We dispatch events for lazy background pages first because attempting to do |
| 407 // so will cause those that are being suspended to cancel that suspension. | 443 // so will cause those that are being suspended to cancel that suspension. |
| 408 // As canceling a suspension entails sending an event to the affected | 444 // As canceling a suspension entails sending an event to the affected |
| 409 // background page, and as that event needs to be delivered before we dispatch | 445 // background page, and as that event needs to be delivered before we dispatch |
| 410 // the event we are dispatching here, we dispatch to the lazy listeners here | 446 // the event we are dispatching here, we dispatch to the lazy listeners here |
| 411 // first. | 447 // first. |
| 412 for (const EventListener* listener : listeners) { | 448 for (const EventListener* listener : listeners) { |
| 413 if (restrict_to_extension_id.empty() || | 449 if (restrict_to_extension_id.empty() || |
| 414 restrict_to_extension_id == listener->extension_id()) { | 450 restrict_to_extension_id == listener->extension_id()) { |
| 415 if (listener->IsLazy()) { | 451 // TODO(lazyboy): Support lazy listeners for extension SW events. |
| 452 if (listener->IsLazy() && !listener->IsForServiceWorker()) { |
| 416 DispatchLazyEvent(listener->extension_id(), event, &already_dispatched, | 453 DispatchLazyEvent(listener->extension_id(), event, &already_dispatched, |
| 417 listener->filter()); | 454 listener->filter()); |
| 418 } | 455 } |
| 419 } | 456 } |
| 420 } | 457 } |
| 421 | 458 |
| 422 for (const EventListener* listener : listeners) { | 459 for (const EventListener* listener : listeners) { |
| 423 if (restrict_to_extension_id.empty() || | 460 if (restrict_to_extension_id.empty() || |
| 424 restrict_to_extension_id == listener->extension_id()) { | 461 restrict_to_extension_id == listener->extension_id()) { |
| 425 if (listener->process()) { | 462 if (listener->process()) { |
| 426 EventDispatchIdentifier dispatch_id(listener->GetBrowserContext(), | 463 EventDispatchIdentifier dispatch_id(listener->GetBrowserContext(), |
| 427 listener->extension_id()); | 464 listener->extension_id(), |
| 465 listener->worker_thread_id()); |
| 428 if (!base::ContainsKey(already_dispatched, dispatch_id)) { | 466 if (!base::ContainsKey(already_dispatched, dispatch_id)) { |
| 429 DispatchEventToProcess(listener->extension_id(), | 467 DispatchEventToProcess(listener->extension_id(), |
| 430 listener->listener_url(), listener->process(), | 468 listener->listener_url(), listener->process(), |
| 431 event, listener->filter(), | 469 listener->worker_thread_id(), event, |
| 432 false /* did_enqueue */); | 470 listener->filter(), false /* did_enqueue */); |
| 433 } | 471 } |
| 434 } | 472 } |
| 435 } | 473 } |
| 436 } | 474 } |
| 437 } | 475 } |
| 438 | 476 |
| 439 void EventRouter::DispatchLazyEvent( | 477 void EventRouter::DispatchLazyEvent( |
| 440 const std::string& extension_id, | 478 const std::string& extension_id, |
| 441 const linked_ptr<Event>& event, | 479 const linked_ptr<Event>& event, |
| 442 std::set<EventDispatchIdentifier>* already_dispatched, | 480 std::set<EventDispatchIdentifier>* already_dispatched, |
| 443 const base::DictionaryValue* listener_filter) { | 481 const base::DictionaryValue* listener_filter) { |
| 444 // Check both the original and the incognito browser context to see if we | 482 // Check both the original and the incognito browser context to see if we |
| 445 // should load a lazy bg page to handle the event. The latter case | 483 // should load a lazy bg page to handle the event. The latter case |
| 446 // occurs in the case of split-mode extensions. | 484 // occurs in the case of split-mode extensions. |
| 447 const Extension* extension = | 485 const Extension* extension = |
| 448 ExtensionRegistry::Get(browser_context_)->enabled_extensions().GetByID( | 486 ExtensionRegistry::Get(browser_context_)->enabled_extensions().GetByID( |
| 449 extension_id); | 487 extension_id); |
| 450 if (!extension) | 488 if (!extension) |
| 451 return; | 489 return; |
| 452 | 490 |
| 453 if (MaybeLoadLazyBackgroundPageToDispatchEvent(browser_context_, extension, | 491 if (MaybeLoadLazyBackgroundPageToDispatchEvent(browser_context_, extension, |
| 454 event, listener_filter)) { | 492 event, listener_filter)) { |
| 455 already_dispatched->insert(std::make_pair(browser_context_, extension_id)); | 493 already_dispatched->insert( |
| 494 std::make_tuple(browser_context_, extension_id, kNonWorkerThreadId)); |
| 456 } | 495 } |
| 457 | 496 |
| 458 ExtensionsBrowserClient* browser_client = ExtensionsBrowserClient::Get(); | 497 ExtensionsBrowserClient* browser_client = ExtensionsBrowserClient::Get(); |
| 459 if (browser_client->HasOffTheRecordContext(browser_context_) && | 498 if (browser_client->HasOffTheRecordContext(browser_context_) && |
| 460 IncognitoInfo::IsSplitMode(extension)) { | 499 IncognitoInfo::IsSplitMode(extension)) { |
| 461 BrowserContext* incognito_context = | 500 BrowserContext* incognito_context = |
| 462 browser_client->GetOffTheRecordContext(browser_context_); | 501 browser_client->GetOffTheRecordContext(browser_context_); |
| 463 if (MaybeLoadLazyBackgroundPageToDispatchEvent(incognito_context, extension, | 502 if (MaybeLoadLazyBackgroundPageToDispatchEvent(incognito_context, extension, |
| 464 event, listener_filter)) { | 503 event, listener_filter)) { |
| 465 already_dispatched->insert( | 504 already_dispatched->insert( |
| 466 std::make_pair(incognito_context, extension_id)); | 505 std::make_tuple(incognito_context, extension_id, kNonWorkerThreadId)); |
| 467 } | 506 } |
| 468 } | 507 } |
| 469 } | 508 } |
| 470 | 509 |
| 471 void EventRouter::DispatchEventToProcess( | 510 void EventRouter::DispatchEventToProcess( |
| 472 const std::string& extension_id, | 511 const std::string& extension_id, |
| 473 const GURL& listener_url, | 512 const GURL& listener_url, |
| 474 content::RenderProcessHost* process, | 513 content::RenderProcessHost* process, |
| 514 int worker_thread_id, |
| 475 const linked_ptr<Event>& event, | 515 const linked_ptr<Event>& event, |
| 476 const base::DictionaryValue* listener_filter, | 516 const base::DictionaryValue* listener_filter, |
| 477 bool did_enqueue) { | 517 bool did_enqueue) { |
| 478 BrowserContext* listener_context = process->GetBrowserContext(); | 518 BrowserContext* listener_context = process->GetBrowserContext(); |
| 479 ProcessMap* process_map = ProcessMap::Get(listener_context); | 519 ProcessMap* process_map = ProcessMap::Get(listener_context); |
| 480 | 520 |
| 481 // NOTE: |extension| being NULL does not necessarily imply that this event | 521 // NOTE: |extension| being NULL does not necessarily imply that this event |
| 482 // shouldn't be dispatched. Events can be dispatched to WebUI and webviews as | 522 // shouldn't be dispatched. Events can be dispatched to WebUI and webviews as |
| 483 // well. It all depends on what GetMostLikelyContextType returns. | 523 // well. It all depends on what GetMostLikelyContextType returns. |
| 484 const Extension* extension = | 524 const Extension* extension = |
| (...skipping 47 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
| 532 return; | 572 return; |
| 533 } | 573 } |
| 534 | 574 |
| 535 if (!event->will_dispatch_callback.is_null() && | 575 if (!event->will_dispatch_callback.is_null() && |
| 536 !event->will_dispatch_callback.Run(listener_context, extension, | 576 !event->will_dispatch_callback.Run(listener_context, extension, |
| 537 event.get(), listener_filter)) { | 577 event.get(), listener_filter)) { |
| 538 return; | 578 return; |
| 539 } | 579 } |
| 540 | 580 |
| 541 int event_id = g_extension_event_id.GetNext(); | 581 int event_id = g_extension_event_id.GetNext(); |
| 542 DispatchExtensionMessage(process, listener_context, extension_id, event_id, | 582 DispatchExtensionMessage(process, worker_thread_id, listener_context, |
| 543 event->event_name, event->event_args.get(), | 583 extension_id, event_id, event->event_name, |
| 544 event->user_gesture, event->filter_info); | 584 event->event_args.get(), event->user_gesture, |
| 585 event->filter_info); |
| 545 | 586 |
| 546 if (extension) { | 587 if (extension) { |
| 547 ReportEvent(event->histogram_value, extension, did_enqueue); | 588 ReportEvent(event->histogram_value, extension, did_enqueue); |
| 548 IncrementInFlightEvents(listener_context, extension, event_id, | 589 IncrementInFlightEvents(listener_context, extension, event_id, |
| 549 event->event_name); | 590 event->event_name); |
| 550 } | 591 } |
| 551 } | 592 } |
| 552 | 593 |
| 553 bool EventRouter::CanDispatchEventToBrowserContext( | 594 bool EventRouter::CanDispatchEventToBrowserContext( |
| 554 BrowserContext* context, | 595 BrowserContext* context, |
| (...skipping 150 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
| 705 } | 746 } |
| 706 } | 747 } |
| 707 } | 748 } |
| 708 | 749 |
| 709 void EventRouter::DispatchPendingEvent(const linked_ptr<Event>& event, | 750 void EventRouter::DispatchPendingEvent(const linked_ptr<Event>& event, |
| 710 ExtensionHost* host) { | 751 ExtensionHost* host) { |
| 711 if (!host) | 752 if (!host) |
| 712 return; | 753 return; |
| 713 | 754 |
| 714 if (listeners_.HasProcessListener(host->render_process_host(), | 755 if (listeners_.HasProcessListener(host->render_process_host(), |
| 756 kNonWorkerThreadId, |
| 715 host->extension()->id())) { | 757 host->extension()->id())) { |
| 716 DispatchEventToProcess(host->extension()->id(), host->GetURL(), | 758 DispatchEventToProcess(host->extension()->id(), host->GetURL(), |
| 717 host->render_process_host(), event, nullptr, | 759 host->render_process_host(), kNonWorkerThreadId, |
| 718 true /* did_enqueue */); | 760 event, nullptr, true /* did_enqueue */); |
| 719 } | 761 } |
| 720 } | 762 } |
| 721 | 763 |
| 722 void EventRouter::SetRegisteredEvents(const std::string& extension_id, | 764 void EventRouter::SetRegisteredEvents(const std::string& extension_id, |
| 723 const std::set<std::string>& events) { | 765 const std::set<std::string>& events, |
| 766 RegisteredEventType type) { |
| 724 auto events_value = base::MakeUnique<base::ListValue>(); | 767 auto events_value = base::MakeUnique<base::ListValue>(); |
| 725 for (std::set<std::string>::const_iterator iter = events.begin(); | 768 for (std::set<std::string>::const_iterator iter = events.begin(); |
| 726 iter != events.end(); ++iter) { | 769 iter != events.end(); ++iter) { |
| 727 events_value->AppendString(*iter); | 770 events_value->AppendString(*iter); |
| 728 } | 771 } |
| 729 extension_prefs_->UpdateExtensionPref(extension_id, kRegisteredEvents, | 772 const char* pref_key = type == RegisteredEventType::kLazy |
| 773 ? kRegisteredLazyEvents |
| 774 : kRegisteredServiceWorkerEvents; |
| 775 extension_prefs_->UpdateExtensionPref(extension_id, pref_key, |
| 730 std::move(events_value)); | 776 std::move(events_value)); |
| 731 } | 777 } |
| 732 | 778 |
| 733 void EventRouter::AddFilterToEvent(const std::string& event_name, | 779 void EventRouter::AddFilterToEvent(const std::string& event_name, |
| 734 const std::string& extension_id, | 780 const std::string& extension_id, |
| 735 const DictionaryValue* filter) { | 781 const DictionaryValue* filter) { |
| 736 ExtensionPrefs::ScopedDictionaryUpdate update(extension_prefs_, extension_id, | 782 ExtensionPrefs::ScopedDictionaryUpdate update(extension_prefs_, extension_id, |
| 737 kFilteredEvents); | 783 kFilteredEvents); |
| 738 auto filtered_events = update.Create(); | 784 auto filtered_events = update.Create(); |
| 739 | 785 |
| (...skipping 18 matching lines...) Expand all Loading... |
| 758 LazyBackgroundTaskQueue* queue = | 804 LazyBackgroundTaskQueue* queue = |
| 759 LazyBackgroundTaskQueue::Get(browser_context_); | 805 LazyBackgroundTaskQueue::Get(browser_context_); |
| 760 queue->AddPendingTask(browser_context_, extension->id(), | 806 queue->AddPendingTask(browser_context_, extension->id(), |
| 761 base::Bind(&DoNothing)); | 807 base::Bind(&DoNothing)); |
| 762 } | 808 } |
| 763 } | 809 } |
| 764 | 810 |
| 765 void EventRouter::OnExtensionLoaded(content::BrowserContext* browser_context, | 811 void EventRouter::OnExtensionLoaded(content::BrowserContext* browser_context, |
| 766 const Extension* extension) { | 812 const Extension* extension) { |
| 767 // Add all registered lazy listeners to our cache. | 813 // Add all registered lazy listeners to our cache. |
| 814 // TODO(lazyboy): Load extension SW lazy events. |
| 768 std::set<std::string> registered_events = | 815 std::set<std::string> registered_events = |
| 769 GetRegisteredEvents(extension->id()); | 816 GetRegisteredEvents(extension->id(), RegisteredEventType::kLazy); |
| 770 listeners_.LoadUnfilteredLazyListeners(extension->id(), registered_events); | 817 listeners_.LoadUnfilteredLazyListeners(extension->id(), registered_events); |
| 818 // TODO(lazyboy): Load extension SW filtered events when they are available. |
| 771 const DictionaryValue* filtered_events = GetFilteredEvents(extension->id()); | 819 const DictionaryValue* filtered_events = GetFilteredEvents(extension->id()); |
| 772 if (filtered_events) | 820 if (filtered_events) |
| 773 listeners_.LoadFilteredLazyListeners(extension->id(), *filtered_events); | 821 listeners_.LoadFilteredLazyListeners(extension->id(), *filtered_events); |
| 774 } | 822 } |
| 775 | 823 |
| 776 void EventRouter::OnExtensionUnloaded(content::BrowserContext* browser_context, | 824 void EventRouter::OnExtensionUnloaded(content::BrowserContext* browser_context, |
| 777 const Extension* extension, | 825 const Extension* extension, |
| 778 UnloadedExtensionReason reason) { | 826 UnloadedExtensionReason reason) { |
| 779 // Remove all registered listeners from our cache. | 827 // Remove all registered listeners from our cache. |
| 780 listeners_.RemoveListenersForExtension(extension->id()); | 828 listeners_.RemoveListenersForExtension(extension->id()); |
| 781 } | 829 } |
| 782 | 830 |
| 831 void EventRouter::AddLazyEventListenerImpl(const std::string& event_name, |
| 832 const ExtensionId& extension_id, |
| 833 int worker_thread_id) { |
| 834 bool is_for_service_worker = worker_thread_id != kNonWorkerThreadId; |
| 835 bool is_new = listeners_.AddListener( |
| 836 is_for_service_worker |
| 837 ? EventListener::ForExtensionServiceWorker( |
| 838 event_name, extension_id, nullptr, worker_thread_id, nullptr) |
| 839 : EventListener::ForExtension(event_name, extension_id, nullptr, |
| 840 nullptr)); |
| 841 |
| 842 if (is_new) { |
| 843 RegisteredEventType type = is_for_service_worker |
| 844 ? RegisteredEventType::kServiceWorker |
| 845 : RegisteredEventType::kLazy; |
| 846 std::set<std::string> events = GetRegisteredEvents(extension_id, type); |
| 847 bool prefs_is_new = events.insert(event_name).second; |
| 848 if (prefs_is_new) |
| 849 SetRegisteredEvents(extension_id, events, type); |
| 850 } |
| 851 } |
| 852 |
| 853 void EventRouter::RemoveLazyEventListenerImpl(const std::string& event_name, |
| 854 const ExtensionId& extension_id, |
| 855 int worker_thread_id) { |
| 856 bool is_for_service_worker = worker_thread_id != kNonWorkerThreadId; |
| 857 std::unique_ptr<EventListener> listener = |
| 858 is_for_service_worker |
| 859 ? EventListener::ForExtensionServiceWorker( |
| 860 event_name, extension_id, nullptr, worker_thread_id, nullptr) |
| 861 : EventListener::ForExtension(event_name, extension_id, nullptr, |
| 862 nullptr); |
| 863 bool did_exist = listeners_.RemoveListener(listener.get()); |
| 864 |
| 865 if (did_exist) { |
| 866 RegisteredEventType type = is_for_service_worker |
| 867 ? RegisteredEventType::kServiceWorker |
| 868 : RegisteredEventType::kLazy; |
| 869 std::set<std::string> events = GetRegisteredEvents(extension_id, type); |
| 870 bool prefs_did_exist = events.erase(event_name) > 0; |
| 871 DCHECK(prefs_did_exist); |
| 872 SetRegisteredEvents(extension_id, events, type); |
| 873 } |
| 874 } |
| 875 |
| 783 Event::Event(events::HistogramValue histogram_value, | 876 Event::Event(events::HistogramValue histogram_value, |
| 784 const std::string& event_name, | 877 const std::string& event_name, |
| 785 std::unique_ptr<base::ListValue> event_args) | 878 std::unique_ptr<base::ListValue> event_args) |
| 786 : Event(histogram_value, event_name, std::move(event_args), nullptr) {} | 879 : Event(histogram_value, event_name, std::move(event_args), nullptr) {} |
| 787 | 880 |
| 788 Event::Event(events::HistogramValue histogram_value, | 881 Event::Event(events::HistogramValue histogram_value, |
| 789 const std::string& event_name, | 882 const std::string& event_name, |
| 790 std::unique_ptr<base::ListValue> event_args, | 883 std::unique_ptr<base::ListValue> event_args, |
| 791 BrowserContext* restrict_to_browser_context) | 884 BrowserContext* restrict_to_browser_context) |
| 792 : Event(histogram_value, | 885 : Event(histogram_value, |
| (...skipping 42 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
| 835 const std::string& extension_id, | 928 const std::string& extension_id, |
| 836 const GURL& listener_url, | 929 const GURL& listener_url, |
| 837 content::BrowserContext* browser_context) | 930 content::BrowserContext* browser_context) |
| 838 : event_name(event_name), | 931 : event_name(event_name), |
| 839 extension_id(extension_id), | 932 extension_id(extension_id), |
| 840 listener_url(listener_url), | 933 listener_url(listener_url), |
| 841 browser_context(browser_context) { | 934 browser_context(browser_context) { |
| 842 } | 935 } |
| 843 | 936 |
| 844 } // namespace extensions | 937 } // namespace extensions |
| OLD | NEW |