| Index: chrome/browser/extensions/extension_pref_store.h
|
| diff --git a/chrome/browser/extensions/extension_pref_store.h b/chrome/browser/extensions/extension_pref_store.h
|
| index f1767589a9aee4a67b2dec4814c4e19716e3803f..514979a2cb93800087fb5a00aca3498def868a3e 100644
|
| --- a/chrome/browser/extensions/extension_pref_store.h
|
| +++ b/chrome/browser/extensions/extension_pref_store.h
|
| @@ -6,15 +6,13 @@
|
| #define CHROME_BROWSER_EXTENSIONS_EXTENSION_PREF_STORE_H_
|
| #pragma once
|
|
|
| -#include <list>
|
| -#include <map>
|
| #include <string>
|
| #include <utility>
|
| #include <vector>
|
|
|
| #include "base/basictypes.h"
|
| +#include "base/gtest_prod_util.h"
|
| #include "base/scoped_ptr.h"
|
| -#include "base/stl_util-inl.h"
|
| #include "chrome/browser/prefs/pref_notifier.h"
|
| #include "chrome/common/notification_observer.h"
|
| #include "chrome/common/notification_registrar.h"
|
| @@ -22,6 +20,7 @@
|
|
|
| class DictionaryValue;
|
| class Extension;
|
| +class ListValue;
|
| class PrefService;
|
| class Profile;
|
| class Value;
|
| @@ -30,12 +29,22 @@ class Value;
|
| // proxy settings. A stack of relevant extensions is stored in order of
|
| // their addition to this PrefStore. For each preference, the last-added
|
| // enabled extension that tries to set it overrules any others.
|
| +//
|
| +// The preferences of extensions are embedded into the user's persistent pref
|
| +// store such that the precedence of extensions is persisted and such that
|
| +// preferences are available immediately after the browser is restarted (i.e.
|
| +// even before extensions have been started).
|
| +// The ExtensionPrefStore hooks into the "extension_preferences" branch of the
|
| +// persistent user prefs. The structure is as follows:
|
| +// extension_preferences: (
|
| +// { id: "extension_id_1", preferences: { ...preferences of ext. 1... } },
|
| +// { id: "extension_id_2", preferences: { ...preferences of ext. 2... } },
|
| +// ... )
|
| +// In this example, extension 1 has higher precendence than extension 2.
|
| +// Keys in extension_preferences/preferences are stored without path expansion.
|
| class ExtensionPrefStore : public PrefStore,
|
| public NotificationObserver {
|
| public:
|
| - // Maps preference paths to their values.
|
| - typedef std::map<const char*, Value*> PrefValueMap;
|
| -
|
| // The type passed as Details for an EXTENSION_PREF_CHANGED notification.
|
| // The nested pairs are <extension, <pref_path, pref_value> >. This is here,
|
| // rather than in (say) notification_type.h, to keep the dependency on
|
| @@ -43,6 +52,11 @@ class ExtensionPrefStore : public PrefStore,
|
| typedef std::pair<const Extension*, std::pair<const char*, Value*> >
|
| ExtensionPrefDetails;
|
|
|
| + // The local-state ExtensionPrefStore (shared among all profiles on a
|
| + // machine), indicated by a NULL |profile|, is prohibited from storing
|
| + // preferences. Since extensions are installed per profile, allowing them to
|
| + // control machine-wide settings could lead to unsolvable conflicts, and
|
| + // should not be necessary anyway.
|
| ExtensionPrefStore(Profile* profile, PrefNotifier::PrefStoreType type);
|
| virtual ~ExtensionPrefStore();
|
|
|
| @@ -58,33 +72,94 @@ class ExtensionPrefStore : public PrefStore,
|
| virtual void UninstallExtension(const Extension* extension);
|
|
|
| // PrefStore methods:
|
| - virtual DictionaryValue* prefs() const { return prefs_.get(); }
|
| -
|
| + virtual DictionaryValue* prefs() const;
|
| virtual PrefReadError ReadPrefs() { return PREF_READ_ERROR_NONE; }
|
|
|
| + static const char kExtensionPreferencesKey[];
|
| protected:
|
| // Returns a vector of the extension IDs in the extension_stack_.
|
| // This should only be accessed by subclasses for unit-testing.
|
| void GetExtensionIDs(std::vector<std::string>* result);
|
|
|
| - // Returns the applicable pref service from the profile (if we have one) or
|
| - // the browser's local state. This should only be accessed or overridden by
|
| - // subclasses for unit-testing.
|
| - virtual PrefService* GetPrefService();
|
| + // For PrefService injection for unit-testing
|
| + virtual PrefService* GetPrefService() const;
|
| +
|
| + // Schedules a write of the backing user-pref store.
|
| + virtual void SchedulePersist();
|
|
|
| private:
|
| - // Associates an extension with the prefs it sets. Owns the pref values.
|
| - struct ExtensionPrefs {
|
| - ExtensionPrefs(const Extension* extension, PrefValueMap* values);
|
| + // For visibility of helper classes:
|
| + FRIEND_TEST_ALL_PREFIXES(ExtensionPrefStoreTest, ExtensionPrefs);
|
| + FRIEND_TEST_ALL_PREFIXES(ExtensionPrefStoreTest, ExtensionStack);
|
| + FRIEND_TEST_ALL_PREFIXES(ExtensionPrefStoreTest, InstallRegisteredExtensions);
|
| + FRIEND_TEST_ALL_PREFIXES(ExtensionPrefStoreTest,
|
| + UninstallRegisteredExtensions);
|
| + FRIEND_TEST_ALL_PREFIXES(ExtensionPrefStoreTest,
|
| + ReadFromRegisteredExtensions);
|
| +
|
| + // Wrapper around one Dictionary embedded in the "extension_preferences"
|
| + // list. Contains an "id"->String and "preferences"->Dictionary entry.
|
| + class ExtensionPrefs {
|
| + public:
|
| + // Wraps itself around the DictionaryValue that is supposed to follow
|
| + // the predefined structure (see ExtensionPrefStore). The ownership
|
| + // of dict remains at the caller.
|
| + explicit ExtensionPrefs(DictionaryValue* dict);
|
| ~ExtensionPrefs();
|
|
|
| - const Extension* extension;
|
| - PrefValueMap* pref_values;
|
| + // Creates a DictionaryValue { id: |extension_id|, preferences: {} }
|
| + // and passes ownership to the caller.
|
| + static DictionaryValue* Create(const std::string& extension_id);
|
| +
|
| + // Returns the id of the extension managed by this ExtensionPrefs
|
| + std::string extension_id() const;
|
| +
|
| + // Returns a pointer to the preference values, the ownership remains
|
| + // at |dict| passed to the constructor.
|
| + DictionaryValue* pref_values();
|
| +
|
| + // Sets a |value| and passes ownership to |dict|. Deletes the value, if
|
| + // |value| is NULL.
|
| + void Set(const std::string& key, Value* value);
|
| +
|
| + // Retrieves a value but retains ownership at |dict|. Returns NULL if the
|
| + // key is not contained in the dictionary.
|
| + Value* Get(const std::string& key);
|
| +
|
| + // Keys in |dict_|:
|
| + static const char kIdKey[];
|
| + static const char kPreferencesKey[];
|
| + private:
|
| + // Weak reference
|
| + DictionaryValue* dict_;
|
| };
|
|
|
| - // A pseudo-stack of extensions and their preferences. Extensions are always
|
| - // added to the head, but may be removed from the middle.
|
| - typedef std::list<ExtensionPrefs*> ExtensionStack;
|
| + // Wrapper around a list in a persistent PrefStore in order to manage and
|
| + // persist preferences of extensions. Each individual extension's preferences
|
| + // are managed by ExtensionPrefs.
|
| + class ExtensionStack {
|
| + public:
|
| + // Preliminary construction, still requires init-call.
|
| + ExtensionStack();
|
| + ~ExtensionStack();
|
| +
|
| + // Attaches the |list| to this object without taking ownership.
|
| + void Init(ListValue* list);
|
| + bool IsInitialized() const;
|
| +
|
| + // returns number of registered extensions
|
| + size_t Size() const;
|
| + ExtensionPrefs Get(int index);
|
| + void Remove(int index);
|
| +
|
| + // Creates and registeres a new entry for the extension with |extension_id|.
|
| + // This extension gains highest precedence and can override other
|
| + // extensions' preference values.
|
| + ExtensionPrefs CreateEntry(const std::string& extension_id);
|
| + private:
|
| + // Weak reference
|
| + ListValue* list_;
|
| + };
|
|
|
| // Applies the highest-priority extension's setting for the given preference
|
| // path to the |prefs_| store, or clears the setting there if no extensions
|
| @@ -92,7 +167,7 @@ class ExtensionPrefStore : public PrefStore,
|
| void UpdateOnePref(const char* path);
|
|
|
| // Updates each preference in the key set of the |pref_values| map.
|
| - void UpdatePrefs(const PrefValueMap* pref_values);
|
| + void UpdatePrefs(const DictionaryValue* pref_values);
|
|
|
| // Registers this as an observer for relevant notifications.
|
| void RegisterObservers();
|
| @@ -102,11 +177,17 @@ class ExtensionPrefStore : public PrefStore,
|
| const NotificationSource& source,
|
| const NotificationDetails& details);
|
|
|
| + // Initializes the extension_stack_ with empty dictionaries for all
|
| + // extensions registered in the precedence list.
|
| + // Declared const for "DictionaryValue* prefs() const"
|
| + void LazyInit() const;
|
| +
|
| // A cache of the highest-priority values for each preference that any
|
| // extension is controlling, for quick read access. Owns the stored values.
|
| scoped_ptr<DictionaryValue> prefs_;
|
|
|
| - ExtensionStack extension_stack_;
|
| + // Accessor to preference values stored in user's pref store.
|
| + mutable ExtensionStack extension_stack_;
|
|
|
| NotificationRegistrar notification_registrar_;
|
|
|
| @@ -115,6 +196,10 @@ class ExtensionPrefStore : public PrefStore,
|
| // extensions.
|
| Profile* profile_;
|
|
|
| + // Empty list that is only used for "local state" (when there is no user pref
|
| + // store).
|
| + scoped_ptr<ListValue> dummy_prefs;
|
| +
|
| // My PrefStore type, assigned by the PrefValueStore.
|
| PrefNotifier::PrefStoreType type_;
|
|
|
|
|