| OLD | NEW |
| (Empty) |
| 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 | |
| 3 // found in the LICENSE file. | |
| 4 | |
| 5 #ifndef CHROME_BROWSER_EXTENSIONS_API_ALARMS_ALARM_MANAGER_H__ | |
| 6 #define CHROME_BROWSER_EXTENSIONS_API_ALARMS_ALARM_MANAGER_H__ | |
| 7 | |
| 8 #include <map> | |
| 9 #include <queue> | |
| 10 #include <string> | |
| 11 #include <vector> | |
| 12 | |
| 13 #include "base/callback.h" | |
| 14 #include "base/gtest_prod_util.h" | |
| 15 #include "base/memory/weak_ptr.h" | |
| 16 #include "base/scoped_observer.h" | |
| 17 #include "base/timer/timer.h" | |
| 18 #include "chrome/common/extensions/api/alarms.h" | |
| 19 #include "extensions/browser/browser_context_keyed_api_factory.h" | |
| 20 #include "extensions/browser/extension_registry_observer.h" | |
| 21 | |
| 22 namespace base { | |
| 23 class Clock; | |
| 24 } // namespace base | |
| 25 | |
| 26 namespace content { | |
| 27 class BrowserContext; | |
| 28 } // namespace content | |
| 29 | |
| 30 namespace extensions { | |
| 31 class ExtensionAlarmsSchedulingTest; | |
| 32 class ExtensionRegistry; | |
| 33 | |
| 34 struct Alarm { | |
| 35 Alarm(); | |
| 36 Alarm(const std::string& name, | |
| 37 const api::alarms::AlarmCreateInfo& create_info, | |
| 38 base::TimeDelta min_granularity, | |
| 39 base::Time now); | |
| 40 ~Alarm(); | |
| 41 | |
| 42 linked_ptr<api::alarms::Alarm> js_alarm; | |
| 43 // The granularity isn't exposed to the extension's javascript, but we poll at | |
| 44 // least as often as the shortest alarm's granularity. It's initialized as | |
| 45 // the relative delay requested in creation, even if creation uses an absolute | |
| 46 // time. This will always be at least as large as the min_granularity | |
| 47 // constructor argument. | |
| 48 base::TimeDelta granularity; | |
| 49 // The minimum granularity is the minimum allowed polling rate. This stops | |
| 50 // alarms from polling too often. | |
| 51 base::TimeDelta minimum_granularity; | |
| 52 }; | |
| 53 | |
| 54 // Manages the currently pending alarms for every extension in a profile. | |
| 55 // There is one manager per virtual Profile. | |
| 56 class AlarmManager : public BrowserContextKeyedAPI, | |
| 57 public ExtensionRegistryObserver, | |
| 58 public base::SupportsWeakPtr<AlarmManager> { | |
| 59 public: | |
| 60 typedef std::vector<Alarm> AlarmList; | |
| 61 | |
| 62 class Delegate { | |
| 63 public: | |
| 64 virtual ~Delegate() {} | |
| 65 // Called when an alarm fires. | |
| 66 virtual void OnAlarm(const std::string& extension_id, | |
| 67 const Alarm& alarm) = 0; | |
| 68 }; | |
| 69 | |
| 70 explicit AlarmManager(content::BrowserContext* context); | |
| 71 ~AlarmManager() override; | |
| 72 | |
| 73 // Override the default delegate. Callee assumes onwership. Used for testing. | |
| 74 void set_delegate(Delegate* delegate) { delegate_.reset(delegate); } | |
| 75 | |
| 76 typedef base::Callback<void()> AddAlarmCallback; | |
| 77 // Adds |alarm| for the given extension, and starts the timer. Invokes | |
| 78 // |callback| when done. | |
| 79 void AddAlarm(const std::string& extension_id, | |
| 80 const Alarm& alarm, | |
| 81 const AddAlarmCallback& callback); | |
| 82 | |
| 83 typedef base::Callback<void(Alarm*)> GetAlarmCallback; | |
| 84 // Passes the alarm with the given name, or NULL if none exists, to | |
| 85 // |callback|. | |
| 86 void GetAlarm(const std::string& extension_id, | |
| 87 const std::string& name, | |
| 88 const GetAlarmCallback& callback); | |
| 89 | |
| 90 typedef base::Callback<void(const AlarmList*)> GetAllAlarmsCallback; | |
| 91 // Passes the list of pending alarms for the given extension, or | |
| 92 // NULL if none exist, to |callback|. | |
| 93 void GetAllAlarms( | |
| 94 const std::string& extension_id, const GetAllAlarmsCallback& callback); | |
| 95 | |
| 96 typedef base::Callback<void(bool)> RemoveAlarmCallback; | |
| 97 // Cancels and removes the alarm with the given name. Invokes |callback| when | |
| 98 // done. | |
| 99 void RemoveAlarm(const std::string& extension_id, | |
| 100 const std::string& name, | |
| 101 const RemoveAlarmCallback& callback); | |
| 102 | |
| 103 typedef base::Callback<void()> RemoveAllAlarmsCallback; | |
| 104 // Cancels and removes all alarms for the given extension. Invokes |callback| | |
| 105 // when done. | |
| 106 void RemoveAllAlarms( | |
| 107 const std::string& extension_id, const RemoveAllAlarmsCallback& callback); | |
| 108 | |
| 109 // Replaces AlarmManager's owned clock with |clock| and takes ownership of it. | |
| 110 void SetClockForTesting(base::Clock* clock); | |
| 111 | |
| 112 // BrowserContextKeyedAPI implementation. | |
| 113 static BrowserContextKeyedAPIFactory<AlarmManager>* GetFactoryInstance(); | |
| 114 | |
| 115 // Convenience method to get the AlarmManager for a content::BrowserContext. | |
| 116 static AlarmManager* Get(content::BrowserContext* browser_context); | |
| 117 | |
| 118 private: | |
| 119 friend void RunScheduleNextPoll(AlarmManager*); | |
| 120 friend class ExtensionAlarmsSchedulingTest; | |
| 121 FRIEND_TEST_ALL_PREFIXES(ExtensionAlarmsSchedulingTest, PollScheduling); | |
| 122 FRIEND_TEST_ALL_PREFIXES(ExtensionAlarmsSchedulingTest, | |
| 123 ReleasedExtensionPollsInfrequently); | |
| 124 FRIEND_TEST_ALL_PREFIXES(ExtensionAlarmsSchedulingTest, TimerRunning); | |
| 125 FRIEND_TEST_ALL_PREFIXES(ExtensionAlarmsSchedulingTest, MinimumGranularity); | |
| 126 FRIEND_TEST_ALL_PREFIXES(ExtensionAlarmsSchedulingTest, | |
| 127 DifferentMinimumGranularities); | |
| 128 FRIEND_TEST_ALL_PREFIXES(ExtensionAlarmsSchedulingTest, | |
| 129 RepeatingAlarmsScheduledPredictably); | |
| 130 friend class BrowserContextKeyedAPIFactory<AlarmManager>; | |
| 131 | |
| 132 typedef std::string ExtensionId; | |
| 133 typedef std::map<ExtensionId, AlarmList> AlarmMap; | |
| 134 | |
| 135 typedef base::Callback<void(const std::string&)> ReadyAction; | |
| 136 typedef std::queue<ReadyAction> ReadyQueue; | |
| 137 typedef std::map<ExtensionId, ReadyQueue> ReadyMap; | |
| 138 | |
| 139 // Iterator used to identify a particular alarm within the Map/List pair. | |
| 140 // "Not found" is represented by <alarms_.end(), invalid_iterator>. | |
| 141 typedef std::pair<AlarmMap::iterator, AlarmList::iterator> AlarmIterator; | |
| 142 | |
| 143 // Part of AddAlarm that is executed after alarms are loaded. | |
| 144 void AddAlarmWhenReady(const Alarm& alarm, | |
| 145 const AddAlarmCallback& callback, | |
| 146 const std::string& extension_id); | |
| 147 | |
| 148 // Part of GetAlarm that is executed after alarms are loaded. | |
| 149 void GetAlarmWhenReady(const std::string& name, | |
| 150 const GetAlarmCallback& callback, | |
| 151 const std::string& extension_id); | |
| 152 | |
| 153 // Part of GetAllAlarms that is executed after alarms are loaded. | |
| 154 void GetAllAlarmsWhenReady(const GetAllAlarmsCallback& callback, | |
| 155 const std::string& extension_id); | |
| 156 | |
| 157 // Part of RemoveAlarm that is executed after alarms are loaded. | |
| 158 void RemoveAlarmWhenReady(const std::string& name, | |
| 159 const RemoveAlarmCallback& callback, | |
| 160 const std::string& extension_id); | |
| 161 | |
| 162 // Part of RemoveAllAlarms that is executed after alarms are loaded. | |
| 163 void RemoveAllAlarmsWhenReady( | |
| 164 const RemoveAllAlarmsCallback& callback, const std::string& extension_id); | |
| 165 | |
| 166 // Helper to return the iterators within the AlarmMap and AlarmList for the | |
| 167 // matching alarm, or an iterator to the end of the AlarmMap if none were | |
| 168 // found. | |
| 169 AlarmIterator GetAlarmIterator(const std::string& extension_id, | |
| 170 const std::string& name); | |
| 171 | |
| 172 // Helper to cancel and remove the alarm at the given iterator. The iterator | |
| 173 // must be valid. | |
| 174 void RemoveAlarmIterator(const AlarmIterator& iter); | |
| 175 | |
| 176 // Callback for when an alarm fires. | |
| 177 void OnAlarm(AlarmIterator iter); | |
| 178 | |
| 179 // Internal helper to add an alarm and start the timer with the given delay. | |
| 180 void AddAlarmImpl(const std::string& extension_id, | |
| 181 const Alarm& alarm); | |
| 182 | |
| 183 // Syncs our alarm data for the given extension to/from the state storage. | |
| 184 void WriteToStorage(const std::string& extension_id); | |
| 185 void ReadFromStorage(const std::string& extension_id, | |
| 186 scoped_ptr<base::Value> value); | |
| 187 | |
| 188 // Set the timer to go off at the specified |time|, and set |next_poll_time| | |
| 189 // appropriately. | |
| 190 void SetNextPollTime(const base::Time& time); | |
| 191 | |
| 192 // Schedules the next poll of alarms for when the next soonest alarm runs, | |
| 193 // but not more often than the minimum granularity of all alarms. | |
| 194 void ScheduleNextPoll(); | |
| 195 | |
| 196 // Polls the alarms, running any that have elapsed. After running them and | |
| 197 // rescheduling repeating alarms, schedule the next poll. | |
| 198 void PollAlarms(); | |
| 199 | |
| 200 // Executes |action| for given extension, making sure that the extension's | |
| 201 // alarm data has been synced from the storage. | |
| 202 void RunWhenReady(const std::string& extension_id, const ReadyAction& action); | |
| 203 | |
| 204 // ExtensionRegistryObserver implementation. | |
| 205 void OnExtensionLoaded(content::BrowserContext* browser_context, | |
| 206 const Extension* extension) override; | |
| 207 void OnExtensionUninstalled(content::BrowserContext* browser_context, | |
| 208 const Extension* extension, | |
| 209 extensions::UninstallReason reason) override; | |
| 210 | |
| 211 // BrowserContextKeyedAPI implementation. | |
| 212 static const char* service_name() { | |
| 213 return "AlarmManager"; | |
| 214 } | |
| 215 static const bool kServiceHasOwnInstanceInIncognito = true; | |
| 216 | |
| 217 content::BrowserContext* const browser_context_; | |
| 218 scoped_ptr<base::Clock> clock_; | |
| 219 scoped_ptr<Delegate> delegate_; | |
| 220 | |
| 221 // Listen to extension load notifications. | |
| 222 ScopedObserver<ExtensionRegistry, ExtensionRegistryObserver> | |
| 223 extension_registry_observer_; | |
| 224 | |
| 225 // The timer for this alarm manager. | |
| 226 base::OneShotTimer<AlarmManager> timer_; | |
| 227 | |
| 228 // A map of our pending alarms, per extension. | |
| 229 // Invariant: None of the AlarmLists are empty. | |
| 230 AlarmMap alarms_; | |
| 231 | |
| 232 // A map of actions waiting for alarm data to be synced from storage, per | |
| 233 // extension. | |
| 234 ReadyMap ready_actions_; | |
| 235 | |
| 236 // The previous time that alarms were run. | |
| 237 base::Time last_poll_time_; | |
| 238 | |
| 239 // Next poll's time. | |
| 240 base::Time next_poll_time_; | |
| 241 | |
| 242 DISALLOW_COPY_AND_ASSIGN(AlarmManager); | |
| 243 }; | |
| 244 | |
| 245 } // namespace extensions | |
| 246 | |
| 247 #endif // CHROME_BROWSER_EXTENSIONS_API_ALARMS_ALARM_MANAGER_H__ | |
| OLD | NEW |