| Index: chrome/browser/extensions/api/alarms/alarm_manager.cc
|
| diff --git a/chrome/browser/extensions/api/alarms/alarm_manager.cc b/chrome/browser/extensions/api/alarms/alarm_manager.cc
|
| deleted file mode 100644
|
| index bf688fcbae76e5c483c4c86e5af5bc2e7c63a0e7..0000000000000000000000000000000000000000
|
| --- a/chrome/browser/extensions/api/alarms/alarm_manager.cc
|
| +++ /dev/null
|
| @@ -1,470 +0,0 @@
|
| -// Copyright (c) 2012 The Chromium Authors. All rights reserved.
|
| -// Use of this source code is governed by a BSD-style license that can be
|
| -// found in the LICENSE file.
|
| -
|
| -#include "chrome/browser/extensions/api/alarms/alarm_manager.h"
|
| -
|
| -#include "base/bind.h"
|
| -#include "base/json/json_writer.h"
|
| -#include "base/lazy_instance.h"
|
| -#include "base/message_loop/message_loop.h"
|
| -#include "base/time/clock.h"
|
| -#include "base/time/default_clock.h"
|
| -#include "base/time/time.h"
|
| -#include "base/value_conversions.h"
|
| -#include "base/values.h"
|
| -#include "chrome/browser/extensions/extension_service.h"
|
| -#include "chrome/common/extensions/api/alarms.h"
|
| -#include "extensions/browser/event_router.h"
|
| -#include "extensions/browser/extension_registry.h"
|
| -#include "extensions/browser/extension_system.h"
|
| -#include "extensions/browser/state_store.h"
|
| -
|
| -namespace extensions {
|
| -
|
| -namespace alarms = api::alarms;
|
| -
|
| -namespace {
|
| -
|
| -// A list of alarms that this extension has set.
|
| -const char kRegisteredAlarms[] = "alarms";
|
| -const char kAlarmGranularity[] = "granularity";
|
| -
|
| -// The minimum period between polling for alarms to run.
|
| -const base::TimeDelta kDefaultMinPollPeriod() {
|
| - return base::TimeDelta::FromDays(1);
|
| -}
|
| -
|
| -class DefaultAlarmDelegate : public AlarmManager::Delegate {
|
| - public:
|
| - explicit DefaultAlarmDelegate(content::BrowserContext* context)
|
| - : browser_context_(context) {}
|
| - ~DefaultAlarmDelegate() override {}
|
| -
|
| - void OnAlarm(const std::string& extension_id, const Alarm& alarm) override {
|
| - scoped_ptr<base::ListValue> args(new base::ListValue());
|
| - args->Append(alarm.js_alarm->ToValue().release());
|
| - scoped_ptr<Event> event(new Event(alarms::OnAlarm::kEventName,
|
| - args.Pass()));
|
| - EventRouter::Get(browser_context_)
|
| - ->DispatchEventToExtension(extension_id, event.Pass());
|
| - }
|
| -
|
| - private:
|
| - content::BrowserContext* browser_context_;
|
| -};
|
| -
|
| -// Creates a TimeDelta from a delay as specified in the API.
|
| -base::TimeDelta TimeDeltaFromDelay(double delay_in_minutes) {
|
| - return base::TimeDelta::FromMicroseconds(
|
| - delay_in_minutes * base::Time::kMicrosecondsPerMinute);
|
| -}
|
| -
|
| -std::vector<Alarm> AlarmsFromValue(const base::ListValue* list) {
|
| - std::vector<Alarm> alarms;
|
| - for (size_t i = 0; i < list->GetSize(); ++i) {
|
| - const base::DictionaryValue* alarm_dict = NULL;
|
| - Alarm alarm;
|
| - if (list->GetDictionary(i, &alarm_dict) &&
|
| - api::alarms::Alarm::Populate(*alarm_dict, alarm.js_alarm.get())) {
|
| - const base::Value* time_value = NULL;
|
| - if (alarm_dict->Get(kAlarmGranularity, &time_value))
|
| - base::GetValueAsTimeDelta(*time_value, &alarm.granularity);
|
| - alarms.push_back(alarm);
|
| - }
|
| - }
|
| - return alarms;
|
| -}
|
| -
|
| -scoped_ptr<base::ListValue> AlarmsToValue(const std::vector<Alarm>& alarms) {
|
| - scoped_ptr<base::ListValue> list(new base::ListValue());
|
| - for (size_t i = 0; i < alarms.size(); ++i) {
|
| - scoped_ptr<base::DictionaryValue> alarm =
|
| - alarms[i].js_alarm->ToValue().Pass();
|
| - alarm->Set(kAlarmGranularity,
|
| - base::CreateTimeDeltaValue(alarms[i].granularity));
|
| - list->Append(alarm.release());
|
| - }
|
| - return list.Pass();
|
| -}
|
| -
|
| -} // namespace
|
| -
|
| -// AlarmManager
|
| -
|
| -AlarmManager::AlarmManager(content::BrowserContext* context)
|
| - : browser_context_(context),
|
| - clock_(new base::DefaultClock()),
|
| - delegate_(new DefaultAlarmDelegate(context)),
|
| - extension_registry_observer_(this) {
|
| - extension_registry_observer_.Add(ExtensionRegistry::Get(browser_context_));
|
| -
|
| - StateStore* storage = ExtensionSystem::Get(browser_context_)->state_store();
|
| - if (storage)
|
| - storage->RegisterKey(kRegisteredAlarms);
|
| -}
|
| -
|
| -AlarmManager::~AlarmManager() {
|
| -}
|
| -
|
| -void AlarmManager::AddAlarm(const std::string& extension_id,
|
| - const Alarm& alarm,
|
| - const AddAlarmCallback& callback) {
|
| - RunWhenReady(extension_id, base::Bind(
|
| - &AlarmManager::AddAlarmWhenReady, AsWeakPtr(), alarm, callback));
|
| -}
|
| -
|
| -void AlarmManager::GetAlarm(const std::string& extension_id,
|
| - const std::string& name,
|
| - const GetAlarmCallback& callback) {
|
| - RunWhenReady(extension_id, base::Bind(
|
| - &AlarmManager::GetAlarmWhenReady, AsWeakPtr(), name, callback));
|
| -}
|
| -
|
| -void AlarmManager::GetAllAlarms(
|
| - const std::string& extension_id, const GetAllAlarmsCallback& callback) {
|
| - RunWhenReady(extension_id, base::Bind(
|
| - &AlarmManager::GetAllAlarmsWhenReady, AsWeakPtr(), callback));
|
| -}
|
| -
|
| -void AlarmManager::RemoveAlarm(const std::string& extension_id,
|
| - const std::string& name,
|
| - const RemoveAlarmCallback& callback) {
|
| - RunWhenReady(extension_id, base::Bind(
|
| - &AlarmManager::RemoveAlarmWhenReady, AsWeakPtr(), name, callback));
|
| -}
|
| -
|
| -void AlarmManager::RemoveAllAlarms(const std::string& extension_id,
|
| - const RemoveAllAlarmsCallback& callback) {
|
| - RunWhenReady(extension_id, base::Bind(
|
| - &AlarmManager::RemoveAllAlarmsWhenReady, AsWeakPtr(), callback));
|
| -}
|
| -
|
| -void AlarmManager::AddAlarmWhenReady(const Alarm& alarm,
|
| - const AddAlarmCallback& callback,
|
| - const std::string& extension_id) {
|
| - AddAlarmImpl(extension_id, alarm);
|
| - WriteToStorage(extension_id);
|
| - callback.Run();
|
| -}
|
| -
|
| -void AlarmManager::GetAlarmWhenReady(const std::string& name,
|
| - const GetAlarmCallback& callback,
|
| - const std::string& extension_id) {
|
| - AlarmIterator it = GetAlarmIterator(extension_id, name);
|
| - callback.Run(it.first != alarms_.end() ? &*it.second : NULL);
|
| -}
|
| -
|
| -void AlarmManager::GetAllAlarmsWhenReady(const GetAllAlarmsCallback& callback,
|
| - const std::string& extension_id) {
|
| - AlarmMap::iterator list = alarms_.find(extension_id);
|
| - callback.Run(list != alarms_.end() ? &list->second : NULL);
|
| -}
|
| -
|
| -void AlarmManager::RemoveAlarmWhenReady(const std::string& name,
|
| - const RemoveAlarmCallback& callback,
|
| - const std::string& extension_id) {
|
| - AlarmIterator it = GetAlarmIterator(extension_id, name);
|
| - if (it.first == alarms_.end()) {
|
| - callback.Run(false);
|
| - return;
|
| - }
|
| -
|
| - RemoveAlarmIterator(it);
|
| - WriteToStorage(extension_id);
|
| - callback.Run(true);
|
| -}
|
| -
|
| -void AlarmManager::RemoveAllAlarmsWhenReady(
|
| - const RemoveAllAlarmsCallback& callback, const std::string& extension_id) {
|
| - AlarmMap::iterator list = alarms_.find(extension_id);
|
| - if (list != alarms_.end()) {
|
| - // Note: I'm using indices rather than iterators here because
|
| - // RemoveAlarmIterator will delete the list when it becomes empty.
|
| - for (size_t i = 0, size = list->second.size(); i < size; ++i)
|
| - RemoveAlarmIterator(AlarmIterator(list, list->second.begin()));
|
| -
|
| - CHECK(alarms_.find(extension_id) == alarms_.end());
|
| - WriteToStorage(extension_id);
|
| - }
|
| - callback.Run();
|
| -}
|
| -
|
| -AlarmManager::AlarmIterator AlarmManager::GetAlarmIterator(
|
| - const std::string& extension_id, const std::string& name) {
|
| - AlarmMap::iterator list = alarms_.find(extension_id);
|
| - if (list == alarms_.end())
|
| - return make_pair(alarms_.end(), AlarmList::iterator());
|
| -
|
| - for (AlarmList::iterator it = list->second.begin();
|
| - it != list->second.end(); ++it) {
|
| - if (it->js_alarm->name == name)
|
| - return make_pair(list, it);
|
| - }
|
| -
|
| - return make_pair(alarms_.end(), AlarmList::iterator());
|
| -}
|
| -
|
| -void AlarmManager::SetClockForTesting(base::Clock* clock) {
|
| - clock_.reset(clock);
|
| -}
|
| -
|
| -static base::LazyInstance<BrowserContextKeyedAPIFactory<AlarmManager> >
|
| - g_factory = LAZY_INSTANCE_INITIALIZER;
|
| -
|
| -// static
|
| -BrowserContextKeyedAPIFactory<AlarmManager>*
|
| -AlarmManager::GetFactoryInstance() {
|
| - return g_factory.Pointer();
|
| -}
|
| -
|
| -// static
|
| -AlarmManager* AlarmManager::Get(content::BrowserContext* browser_context) {
|
| - return BrowserContextKeyedAPIFactory<AlarmManager>::Get(browser_context);
|
| -}
|
| -
|
| -void AlarmManager::RemoveAlarmIterator(const AlarmIterator& iter) {
|
| - AlarmList& list = iter.first->second;
|
| - list.erase(iter.second);
|
| - if (list.empty())
|
| - alarms_.erase(iter.first);
|
| -
|
| - // Cancel the timer if there are no more alarms.
|
| - // We don't need to reschedule the poll otherwise, because in
|
| - // the worst case we would just poll one extra time.
|
| - if (alarms_.empty()) {
|
| - timer_.Stop();
|
| - next_poll_time_ = base::Time();
|
| - }
|
| -}
|
| -
|
| -void AlarmManager::OnAlarm(AlarmIterator it) {
|
| - CHECK(it.first != alarms_.end());
|
| - Alarm& alarm = *it.second;
|
| - std::string extension_id_copy(it.first->first);
|
| - delegate_->OnAlarm(extension_id_copy, alarm);
|
| -
|
| - // Update our scheduled time for the next alarm.
|
| - if (double* period_in_minutes =
|
| - alarm.js_alarm->period_in_minutes.get()) {
|
| - // Get the timer's delay in JS time (i.e., convert it from minutes to
|
| - // milliseconds).
|
| - double period_in_js_time =
|
| - *period_in_minutes * base::Time::kMicrosecondsPerMinute /
|
| - base::Time::kMicrosecondsPerMillisecond;
|
| - // Find out how many periods have transpired since the alarm last went off
|
| - // (it's possible that we missed some).
|
| - int transpired_periods =
|
| - (last_poll_time_.ToJsTime() - alarm.js_alarm->scheduled_time) /
|
| - period_in_js_time;
|
| - // Schedule the alarm for the next period that is in-line with the original
|
| - // scheduling.
|
| - alarm.js_alarm->scheduled_time +=
|
| - period_in_js_time * (transpired_periods + 1);
|
| - } else {
|
| - RemoveAlarmIterator(it);
|
| - }
|
| - WriteToStorage(extension_id_copy);
|
| -}
|
| -
|
| -void AlarmManager::AddAlarmImpl(const std::string& extension_id,
|
| - const Alarm& alarm) {
|
| - // Override any old alarm with the same name.
|
| - AlarmIterator old_alarm = GetAlarmIterator(extension_id,
|
| - alarm.js_alarm->name);
|
| - if (old_alarm.first != alarms_.end())
|
| - RemoveAlarmIterator(old_alarm);
|
| -
|
| - alarms_[extension_id].push_back(alarm);
|
| - base::Time alarm_time =
|
| - base::Time::FromJsTime(alarm.js_alarm->scheduled_time);
|
| - if (next_poll_time_.is_null() || alarm_time < next_poll_time_)
|
| - SetNextPollTime(alarm_time);
|
| -}
|
| -
|
| -void AlarmManager::WriteToStorage(const std::string& extension_id) {
|
| - StateStore* storage = ExtensionSystem::Get(browser_context_)->state_store();
|
| - if (!storage)
|
| - return;
|
| -
|
| - scoped_ptr<base::Value> alarms;
|
| - AlarmMap::iterator list = alarms_.find(extension_id);
|
| - if (list != alarms_.end())
|
| - alarms.reset(AlarmsToValue(list->second).release());
|
| - else
|
| - alarms.reset(AlarmsToValue(std::vector<Alarm>()).release());
|
| - storage->SetExtensionValue(extension_id, kRegisteredAlarms, alarms.Pass());
|
| -}
|
| -
|
| -void AlarmManager::ReadFromStorage(const std::string& extension_id,
|
| - scoped_ptr<base::Value> value) {
|
| - base::ListValue* list = NULL;
|
| - if (value.get() && value->GetAsList(&list)) {
|
| - std::vector<Alarm> alarm_states = AlarmsFromValue(list);
|
| - for (size_t i = 0; i < alarm_states.size(); ++i)
|
| - AddAlarmImpl(extension_id, alarm_states[i]);
|
| - }
|
| -
|
| - ReadyQueue& extension_ready_queue = ready_actions_[extension_id];
|
| - while (!extension_ready_queue.empty()) {
|
| - extension_ready_queue.front().Run(extension_id);
|
| - extension_ready_queue.pop();
|
| - }
|
| - ready_actions_.erase(extension_id);
|
| -}
|
| -
|
| -void AlarmManager::SetNextPollTime(const base::Time& time) {
|
| - next_poll_time_ = time;
|
| - timer_.Start(FROM_HERE,
|
| - std::max(base::TimeDelta::FromSeconds(0), time - clock_->Now()),
|
| - this,
|
| - &AlarmManager::PollAlarms);
|
| -}
|
| -
|
| -void AlarmManager::ScheduleNextPoll() {
|
| - // If there are no alarms, stop the timer.
|
| - if (alarms_.empty()) {
|
| - timer_.Stop();
|
| - next_poll_time_ = base::Time();
|
| - return;
|
| - }
|
| -
|
| - // Find the soonest alarm that is scheduled to run and the smallest
|
| - // granularity of any alarm.
|
| - // alarms_ guarantees that none of its contained lists are empty.
|
| - base::Time soonest_alarm_time = base::Time::FromJsTime(
|
| - alarms_.begin()->second.begin()->js_alarm->scheduled_time);
|
| - base::TimeDelta min_granularity = kDefaultMinPollPeriod();
|
| - for (AlarmMap::const_iterator m_it = alarms_.begin(), m_end = alarms_.end();
|
| - m_it != m_end; ++m_it) {
|
| - for (AlarmList::const_iterator l_it = m_it->second.begin();
|
| - l_it != m_it->second.end(); ++l_it) {
|
| - base::Time cur_alarm_time =
|
| - base::Time::FromJsTime(l_it->js_alarm->scheduled_time);
|
| - if (cur_alarm_time < soonest_alarm_time)
|
| - soonest_alarm_time = cur_alarm_time;
|
| - if (l_it->granularity < min_granularity)
|
| - min_granularity = l_it->granularity;
|
| - base::TimeDelta cur_alarm_delta = cur_alarm_time - last_poll_time_;
|
| - if (cur_alarm_delta < l_it->minimum_granularity)
|
| - cur_alarm_delta = l_it->minimum_granularity;
|
| - if (cur_alarm_delta < min_granularity)
|
| - min_granularity = cur_alarm_delta;
|
| - }
|
| - }
|
| -
|
| - base::Time next_poll(last_poll_time_ + min_granularity);
|
| - // If the next alarm is more than min_granularity in the future, wait for it.
|
| - // Otherwise, only poll as often as min_granularity.
|
| - // As a special case, if we've never checked for an alarm before
|
| - // (e.g. during startup), let alarms fire asap.
|
| - if (last_poll_time_.is_null() || next_poll < soonest_alarm_time)
|
| - next_poll = soonest_alarm_time;
|
| -
|
| - // Schedule the poll.
|
| - SetNextPollTime(next_poll);
|
| -}
|
| -
|
| -void AlarmManager::PollAlarms() {
|
| - last_poll_time_ = clock_->Now();
|
| -
|
| - // Run any alarms scheduled in the past. OnAlarm uses vector::erase to remove
|
| - // elements from the AlarmList, and map::erase to remove AlarmLists from the
|
| - // AlarmMap.
|
| - for (AlarmMap::iterator m_it = alarms_.begin(), m_end = alarms_.end();
|
| - m_it != m_end;) {
|
| - AlarmMap::iterator cur_extension = m_it++;
|
| -
|
| - // Iterate (a) backwards so that removing elements doesn't affect
|
| - // upcoming iterations, and (b) with indices so that if the last
|
| - // iteration destroys the AlarmList, I'm not about to use the end
|
| - // iterator that the destruction invalidates.
|
| - for (size_t i = cur_extension->second.size(); i > 0; --i) {
|
| - AlarmList::iterator cur_alarm = cur_extension->second.begin() + i - 1;
|
| - if (base::Time::FromJsTime(cur_alarm->js_alarm->scheduled_time) <=
|
| - last_poll_time_) {
|
| - OnAlarm(make_pair(cur_extension, cur_alarm));
|
| - }
|
| - }
|
| - }
|
| -
|
| - ScheduleNextPoll();
|
| -}
|
| -
|
| -static void RemoveAllOnUninstallCallback() {}
|
| -
|
| -void AlarmManager::RunWhenReady(
|
| - const std::string& extension_id, const ReadyAction& action) {
|
| - ReadyMap::iterator it = ready_actions_.find(extension_id);
|
| -
|
| - if (it == ready_actions_.end())
|
| - action.Run(extension_id);
|
| - else
|
| - it->second.push(action);
|
| -}
|
| -
|
| -void AlarmManager::OnExtensionLoaded(content::BrowserContext* browser_context,
|
| - const Extension* extension) {
|
| - StateStore* storage = ExtensionSystem::Get(browser_context_)->state_store();
|
| - if (storage) {
|
| - ready_actions_.insert(ReadyMap::value_type(extension->id(), ReadyQueue()));
|
| - storage->GetExtensionValue(
|
| - extension->id(),
|
| - kRegisteredAlarms,
|
| - base::Bind(
|
| - &AlarmManager::ReadFromStorage, AsWeakPtr(), extension->id()));
|
| - }
|
| -}
|
| -
|
| -void AlarmManager::OnExtensionUninstalled(
|
| - content::BrowserContext* browser_context,
|
| - const Extension* extension,
|
| - extensions::UninstallReason reason) {
|
| - RemoveAllAlarms(extension->id(), base::Bind(RemoveAllOnUninstallCallback));
|
| -}
|
| -
|
| -// AlarmManager::Alarm
|
| -
|
| -Alarm::Alarm()
|
| - : js_alarm(new api::alarms::Alarm()) {
|
| -}
|
| -
|
| -Alarm::Alarm(const std::string& name,
|
| - const api::alarms::AlarmCreateInfo& create_info,
|
| - base::TimeDelta min_granularity,
|
| - base::Time now)
|
| - : js_alarm(new api::alarms::Alarm()) {
|
| - js_alarm->name = name;
|
| - minimum_granularity = min_granularity;
|
| -
|
| - if (create_info.when.get()) {
|
| - // Absolute scheduling.
|
| - js_alarm->scheduled_time = *create_info.when;
|
| - granularity = base::Time::FromJsTime(js_alarm->scheduled_time) - now;
|
| - } else {
|
| - // Relative scheduling.
|
| - double* delay_in_minutes = create_info.delay_in_minutes.get();
|
| - if (delay_in_minutes == NULL)
|
| - delay_in_minutes = create_info.period_in_minutes.get();
|
| - CHECK(delay_in_minutes != NULL)
|
| - << "ValidateAlarmCreateInfo in alarms_api.cc should have "
|
| - << "prevented this call.";
|
| - base::TimeDelta delay = TimeDeltaFromDelay(*delay_in_minutes);
|
| - js_alarm->scheduled_time = (now + delay).ToJsTime();
|
| - granularity = delay;
|
| - }
|
| -
|
| - if (granularity < min_granularity)
|
| - granularity = min_granularity;
|
| -
|
| - // Check for repetition.
|
| - if (create_info.period_in_minutes.get()) {
|
| - js_alarm->period_in_minutes.reset(
|
| - new double(*create_info.period_in_minutes));
|
| - }
|
| -}
|
| -
|
| -Alarm::~Alarm() {
|
| -}
|
| -
|
| -} // namespace extensions
|
|
|