diff options
author | cmumford <cmumford@chromium.org> | 2016-03-24 13:35:27 -0700 |
---|---|---|
committer | Commit bot <commit-bot@chromium.org> | 2016-03-24 20:36:29 +0000 |
commit | 6ae8d465f961bb3c6fa9ed285d877710fb69ae12 (patch) | |
tree | 2ef3a31a1254b3d2bc9fa84f942f5a4be023ab4a /extensions | |
parent | 7fad54c907e98f5c574cbf12420ab072eecf81ad (diff) | |
download | chromium_src-6ae8d465f961bb3c6fa9ed285d877710fb69ae12.zip chromium_src-6ae8d465f961bb3c6fa9ed285d877710fb69ae12.tar.gz chromium_src-6ae8d465f961bb3c6fa9ed285d877710fb69ae12.tar.bz2 |
Extensions: Using common ValueStoreFactory for all value stores.
1. Introduce new ValueStoreFactory interface used for the creation of
ValueStore's in all namespaces (local, sync, and managed), and for all
types (extension and application).
2. Delete SettingsStorageFactory/LeveldbSettingsStorageFactory,
and switched to ValueStoreFactory.
3. Created a new TestValueStoreFactory (for testing). This combines
settings_sync_unittest.cc:TestingValueStoreFactory and
ScopedSettingsStorageFactory.
4. ValueStoreFrontend::Backend always lazilily initializes using the
ValueStoreFactory. This makes unnecessary StateStore's deferred
initialization mechanism - which will be removed in an upcoming CL.
5. A new ValueStoreFactoryImpl to mint new ValueStore's for Chrome.
This currently delegates to a new LegacyValueStoreFactory which
creates new LeveldbValueStore. An upcoming CL will add a second
delegated factory (currently called ProfileValueStoreFactory) to
support a unified (per-profile) extensions database.
6. Removed memcheck suppression for SettingsStorageFactory as this
class is now deleted (crbug.com/163922).
BUG=453946,163922
Review URL: https://codereview.chromium.org/1803193002
Cr-Commit-Position: refs/heads/master@{#383137}
Diffstat (limited to 'extensions')
35 files changed, 994 insertions, 327 deletions
diff --git a/extensions/browser/api/extensions_api_client.cc b/extensions/browser/api/extensions_api_client.cc index 7eb174f..bfc08d3 100644 --- a/extensions/browser/api/extensions_api_client.cc +++ b/extensions/browser/api/extensions_api_client.cc @@ -28,7 +28,7 @@ ExtensionsAPIClient* ExtensionsAPIClient::Get() { return g_instance; } void ExtensionsAPIClient::AddAdditionalValueStoreCaches( content::BrowserContext* context, - const scoped_refptr<SettingsStorageFactory>& factory, + const scoped_refptr<ValueStoreFactory>& factory, const scoped_refptr<base::ObserverListThreadSafe<SettingsObserver>>& observers, std::map<settings_namespace::Namespace, ValueStoreCache*>* caches) {} diff --git a/extensions/browser/api/extensions_api_client.h b/extensions/browser/api/extensions_api_client.h index ca50ba8..d68174d 100644 --- a/extensions/browser/api/extensions_api_client.h +++ b/extensions/browser/api/extensions_api_client.h @@ -40,8 +40,8 @@ class MimeHandlerViewGuest; class MimeHandlerViewGuestDelegate; class RulesCacheDelegate; class SettingsObserver; -class SettingsStorageFactory; class ValueStoreCache; +class ValueStoreFactory; class VirtualKeyboardDelegate; class WebRequestEventRouterDelegate; class WebViewGuest; @@ -69,7 +69,7 @@ class ExtensionsAPIClient { // to |caches|. By default adds nothing. virtual void AddAdditionalValueStoreCaches( content::BrowserContext* context, - const scoped_refptr<SettingsStorageFactory>& factory, + const scoped_refptr<ValueStoreFactory>& factory, const scoped_refptr<base::ObserverListThreadSafe<SettingsObserver>>& observers, std::map<settings_namespace::Namespace, ValueStoreCache*>* caches); diff --git a/extensions/browser/api/storage/leveldb_settings_storage_factory.cc b/extensions/browser/api/storage/leveldb_settings_storage_factory.cc deleted file mode 100644 index ef7d302..0000000 --- a/extensions/browser/api/storage/leveldb_settings_storage_factory.cc +++ /dev/null @@ -1,43 +0,0 @@ -// Copyright 2014 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 "extensions/browser/api/storage/leveldb_settings_storage_factory.h" - -#include "base/files/file_util.h" -#include "base/logging.h" -#include "extensions/browser/value_store/leveldb_value_store.h" - -namespace extensions { - -namespace { - -// Statistics are logged to UMA with this string as part of histogram name. They -// can all be found under Extensions.Database.Open.<client>. Changing this needs -// to synchronize with histograms.xml, AND will also become incompatible with -// older browsers still reporting the previous values. -const char kDatabaseUMAClientName[] = "Settings"; - -base::FilePath GetDatabasePath(const base::FilePath& base_path, - const std::string& extension_id) { - return base_path.AppendASCII(extension_id); -} - -} // namespace - -ValueStore* LeveldbSettingsStorageFactory::Create( - const base::FilePath& base_path, - const std::string& extension_id) { - return new LeveldbValueStore(kDatabaseUMAClientName, - GetDatabasePath(base_path, extension_id)); -} - -void LeveldbSettingsStorageFactory::DeleteDatabaseIfExists( - const base::FilePath& base_path, - const std::string& extension_id) { - base::FilePath path = GetDatabasePath(base_path, extension_id); - if (base::PathExists(path)) - base::DeleteFile(path, true /* recursive */); -} - -} // namespace extensions diff --git a/extensions/browser/api/storage/leveldb_settings_storage_factory.h b/extensions/browser/api/storage/leveldb_settings_storage_factory.h deleted file mode 100644 index 6f9cc56..0000000 --- a/extensions/browser/api/storage/leveldb_settings_storage_factory.h +++ /dev/null @@ -1,28 +0,0 @@ -// Copyright 2014 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. - -#ifndef EXTENSIONS_BROWSER_API_STORAGE_LEVELDB_SETTINGS_STORAGE_FACTORY_H_ -#define EXTENSIONS_BROWSER_API_STORAGE_LEVELDB_SETTINGS_STORAGE_FACTORY_H_ - -#include "extensions/browser/api/storage/settings_storage_factory.h" - -namespace extensions { - -// Factory for creating LeveldbValueStore instances. -class LeveldbSettingsStorageFactory : public SettingsStorageFactory { - public: - ValueStore* Create(const base::FilePath& base_path, - const std::string& extension_id) override; - - void DeleteDatabaseIfExists(const base::FilePath& base_path, - const std::string& extension_id) override; - - private: - // SettingsStorageFactory is refcounted. - ~LeveldbSettingsStorageFactory() override {} -}; - -} // namespace extensions - -#endif // EXTENSIONS_BROWSER_API_STORAGE_LEVELDB_SETTINGS_STORAGE_FACTORY_H_ diff --git a/extensions/browser/api/storage/local_value_store_cache.cc b/extensions/browser/api/storage/local_value_store_cache.cc index 9c9ab15..920d130 100644 --- a/extensions/browser/api/storage/local_value_store_cache.cc +++ b/extensions/browser/api/storage/local_value_store_cache.cc @@ -8,18 +8,10 @@ #include <limits> -#include "base/bind.h" -#include "base/callback.h" -#include "base/files/file_path.h" #include "content/public/browser/browser_thread.h" -#include "extensions/browser/api/storage/settings_storage_factory.h" -#include "extensions/browser/api/storage/settings_storage_quota_enforcer.h" #include "extensions/browser/api/storage/weak_unlimited_settings_storage.h" -#include "extensions/browser/value_store/value_store.h" +#include "extensions/browser/value_store/value_store_factory.h" #include "extensions/common/api/storage.h" -#include "extensions/common/constants.h" -#include "extensions/common/extension.h" -#include "extensions/common/permissions/api_permission.h" #include "extensions/common/permissions/permissions_data.h" using content::BrowserThread; @@ -40,13 +32,8 @@ SettingsStorageQuotaEnforcer::Limits GetLocalQuotaLimits() { } // namespace LocalValueStoreCache::LocalValueStoreCache( - const scoped_refptr<SettingsStorageFactory>& factory, - const base::FilePath& profile_path) - : storage_factory_(factory), - extension_base_path_( - profile_path.AppendASCII(kLocalExtensionSettingsDirectoryName)), - app_base_path_(profile_path.AppendASCII(kLocalAppSettingsDirectoryName)), - quota_(GetLocalQuotaLimits()) { + const scoped_refptr<ValueStoreFactory>& factory) + : storage_factory_(factory), quota_(GetLocalQuotaLimits()) { DCHECK_CURRENTLY_ON(BrowserThread::UI); } @@ -59,7 +46,7 @@ void LocalValueStoreCache::RunWithValueStoreForExtension( scoped_refptr<const Extension> extension) { DCHECK_CURRENTLY_ON(BrowserThread::FILE); - ValueStore* storage = GetStorage(extension); + ValueStore* storage = GetStorage(extension.get()); // A neat way to implement unlimited storage; if the extension has the // unlimited storage permission, force through all calls to Set(). @@ -75,21 +62,26 @@ void LocalValueStoreCache::RunWithValueStoreForExtension( void LocalValueStoreCache::DeleteStorageSoon(const std::string& extension_id) { DCHECK_CURRENTLY_ON(BrowserThread::FILE); storage_map_.erase(extension_id); - storage_factory_->DeleteDatabaseIfExists(app_base_path_, extension_id); - storage_factory_->DeleteDatabaseIfExists(extension_base_path_, extension_id); + storage_factory_->DeleteSettings(settings_namespace::LOCAL, + ValueStoreFactory::ModelType::APP, + extension_id); + storage_factory_->DeleteSettings(settings_namespace::LOCAL, + ValueStoreFactory::ModelType::EXTENSION, + extension_id); } -ValueStore* LocalValueStoreCache::GetStorage( - scoped_refptr<const Extension> extension) { +ValueStore* LocalValueStoreCache::GetStorage(const Extension* extension) { StorageMap::iterator iter = storage_map_.find(extension->id()); if (iter != storage_map_.end()) return iter->second.get(); - const base::FilePath& file_path = - extension->is_app() ? app_base_path_ : extension_base_path_; + ValueStoreFactory::ModelType model_type = + extension->is_app() ? ValueStoreFactory::ModelType::APP + : ValueStoreFactory::ModelType::EXTENSION; + scoped_ptr<ValueStore> store = storage_factory_->CreateSettingsStore( + settings_namespace::LOCAL, model_type, extension->id()); linked_ptr<SettingsStorageQuotaEnforcer> storage( - new SettingsStorageQuotaEnforcer( - quota_, storage_factory_->Create(file_path, extension->id()))); + new SettingsStorageQuotaEnforcer(quota_, std::move(store))); DCHECK(storage.get()); storage_map_[extension->id()] = storage; diff --git a/extensions/browser/api/storage/local_value_store_cache.h b/extensions/browser/api/storage/local_value_store_cache.h index 6385ff1..c3303a5 100644 --- a/extensions/browser/api/storage/local_value_store_cache.h +++ b/extensions/browser/api/storage/local_value_store_cache.h @@ -6,7 +6,6 @@ #define EXTENSIONS_BROWSER_API_STORAGE_LOCAL_VALUE_STORE_CACHE_H_ #include "base/compiler_specific.h" -#include "base/files/file_path.h" #include "base/macros.h" #include "base/memory/linked_ptr.h" #include "base/memory/ref_counted.h" @@ -16,14 +15,14 @@ namespace extensions { -class SettingsStorageFactory; +class ValueStoreFactory; // ValueStoreCache for the LOCAL namespace. It owns a backend for apps and // another for extensions. Each backend takes care of persistence. class LocalValueStoreCache : public ValueStoreCache { public: - LocalValueStoreCache(const scoped_refptr<SettingsStorageFactory>& factory, - const base::FilePath& profile_path); + explicit LocalValueStoreCache( + const scoped_refptr<ValueStoreFactory>& factory); ~LocalValueStoreCache() override; // ValueStoreCache implementation: @@ -35,16 +34,10 @@ class LocalValueStoreCache : public ValueStoreCache { private: typedef std::map<std::string, linked_ptr<ValueStore> > StorageMap; - ValueStore* GetStorage(scoped_refptr<const Extension> extension); + ValueStore* GetStorage(const Extension* extension); // The Factory to use for creating new ValueStores. - const scoped_refptr<SettingsStorageFactory> storage_factory_; - - // The base path to use for extensions when creating new ValueStores. - const base::FilePath extension_base_path_; - - // The base path to use for apps when creating new ValueStores. - const base::FilePath app_base_path_; + const scoped_refptr<ValueStoreFactory> storage_factory_; // Quota limits (see SettingsStorageQuotaEnforcer). const SettingsStorageQuotaEnforcer::Limits quota_; diff --git a/extensions/browser/api/storage/settings_quota_unittest.cc b/extensions/browser/api/storage/settings_quota_unittest.cc index c89669c..3681401 100644 --- a/extensions/browser/api/storage/settings_quota_unittest.cc +++ b/extensions/browser/api/storage/settings_quota_unittest.cc @@ -54,7 +54,8 @@ class ExtensionSettingsQuotaTest : public testing::Test { ASSERT_TRUE(storage_.get() == NULL); SettingsStorageQuotaEnforcer::Limits limits = { quota_bytes, quota_bytes_per_item, max_items }; - storage_.reset(new SettingsStorageQuotaEnforcer(limits, delegate_)); + storage_.reset( + new SettingsStorageQuotaEnforcer(limits, make_scoped_ptr(delegate_))); } // Returns whether the settings in |storage_| and |delegate_| are the same as diff --git a/extensions/browser/api/storage/settings_storage_factory.h b/extensions/browser/api/storage/settings_storage_factory.h deleted file mode 100644 index 14f6187..0000000 --- a/extensions/browser/api/storage/settings_storage_factory.h +++ /dev/null @@ -1,42 +0,0 @@ -// Copyright 2014 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. - -#ifndef EXTENSIONS_BROWSER_API_STORAGE_SETTINGS_STORAGE_FACTORY_H_ -#define EXTENSIONS_BROWSER_API_STORAGE_SETTINGS_STORAGE_FACTORY_H_ - -#include <string> - -#include "base/files/file_path.h" -#include "base/memory/ref_counted.h" - -class ValueStore; - -namespace extensions { - -// Factory for creating SettingStorage instances. -// -// Refcouted because it's just too messy to distribute these objects between -// ValueStoreCache instances any other way. -class SettingsStorageFactory - : public base::RefCountedThreadSafe<SettingsStorageFactory> { - public: - // Creates a new ValueStore area for an extension under |base_path|. - // Return NULL to indicate failure. Must be called on the FILE thread. - virtual ValueStore* Create(const base::FilePath& base_path, - const std::string& extension_id) = 0; - - // Deletes the database for the extension, if one exists. - // Note: it is important to delete references to the database if any are - // held, because ValueStores will create themselves if there is no file. - virtual void DeleteDatabaseIfExists(const base::FilePath& base_path, - const std::string& extension_id) = 0; - - protected: - friend class base::RefCountedThreadSafe<SettingsStorageFactory>; - virtual ~SettingsStorageFactory() {} -}; - -} // namespace extensions - -#endif // EXTENSIONS_BROWSER_API_STORAGE_SETTINGS_STORAGE_FACTORY_H_ diff --git a/extensions/browser/api/storage/settings_storage_quota_enforcer.cc b/extensions/browser/api/storage/settings_storage_quota_enforcer.cc index 9a9e9b7..7d3703c 100644 --- a/extensions/browser/api/storage/settings_storage_quota_enforcer.cc +++ b/extensions/browser/api/storage/settings_storage_quota_enforcer.cc @@ -70,10 +70,11 @@ ValueStore::Status QuotaExceededError(Resource resource) { } // namespace -SettingsStorageQuotaEnforcer::SettingsStorageQuotaEnforcer(const Limits& limits, - ValueStore* delegate) +SettingsStorageQuotaEnforcer::SettingsStorageQuotaEnforcer( + const Limits& limits, + scoped_ptr<ValueStore> delegate) : limits_(limits), - delegate_(delegate), + delegate_(std::move(delegate)), used_total_(0), usage_calculated_(false) {} diff --git a/extensions/browser/api/storage/settings_storage_quota_enforcer.h b/extensions/browser/api/storage/settings_storage_quota_enforcer.h index 48a5e89..33de5b7 100644 --- a/extensions/browser/api/storage/settings_storage_quota_enforcer.h +++ b/extensions/browser/api/storage/settings_storage_quota_enforcer.h @@ -13,6 +13,7 @@ #include "base/compiler_specific.h" #include "base/macros.h" +#include "base/memory/scoped_ptr.h" #include "base/memory/weak_ptr.h" #include "extensions/browser/value_store/value_store.h" @@ -33,7 +34,8 @@ class SettingsStorageQuotaEnforcer : public ValueStore { size_t max_items; }; - SettingsStorageQuotaEnforcer(const Limits& limits, ValueStore* delegate); + SettingsStorageQuotaEnforcer(const Limits& limits, + scoped_ptr<ValueStore> delegate); ~SettingsStorageQuotaEnforcer() override; diff --git a/extensions/browser/api/storage/settings_test_util.cc b/extensions/browser/api/storage/settings_test_util.cc index c3dc4a5..4442541 100644 --- a/extensions/browser/api/storage/settings_test_util.cc +++ b/extensions/browser/api/storage/settings_test_util.cc @@ -118,34 +118,6 @@ scoped_refptr<const Extension> AddExtensionWithIdAndPermissions( return extension; } -// ScopedSettingsFactory - -ScopedSettingsStorageFactory::ScopedSettingsStorageFactory() {} - -ScopedSettingsStorageFactory::ScopedSettingsStorageFactory( - const scoped_refptr<SettingsStorageFactory>& delegate) - : delegate_(delegate) {} - -ScopedSettingsStorageFactory::~ScopedSettingsStorageFactory() {} - -void ScopedSettingsStorageFactory::Reset( - const scoped_refptr<SettingsStorageFactory>& delegate) { - delegate_ = delegate; -} - -ValueStore* ScopedSettingsStorageFactory::Create( - const base::FilePath& base_path, - const std::string& extension_id) { - DCHECK(delegate_.get()); - return delegate_->Create(base_path, extension_id); -} - -void ScopedSettingsStorageFactory::DeleteDatabaseIfExists( - const base::FilePath& base_path, - const std::string& extension_id) { - delegate_->DeleteDatabaseIfExists(base_path, extension_id); -} - } // namespace settings_test_util } // namespace extensions diff --git a/extensions/browser/api/storage/settings_test_util.h b/extensions/browser/api/storage/settings_test_util.h index 0b23f60..07a1d52 100644 --- a/extensions/browser/api/storage/settings_test_util.h +++ b/extensions/browser/api/storage/settings_test_util.h @@ -14,9 +14,9 @@ #include "base/memory/scoped_ptr.h" #include "chrome/test/base/testing_profile.h" #include "extensions/browser/api/storage/settings_namespace.h" -#include "extensions/browser/api/storage/settings_storage_factory.h" #include "extensions/browser/event_router.h" #include "extensions/browser/mock_extension_system.h" +#include "extensions/browser/value_store/value_store_factory.h" #include "extensions/common/extension.h" class ValueStore; @@ -56,30 +56,6 @@ scoped_refptr<const Extension> AddExtensionWithIdAndPermissions( Manifest::Type type, const std::set<std::string>& permissions); -// SettingsStorageFactory which acts as a wrapper for other factories. -class ScopedSettingsStorageFactory : public SettingsStorageFactory { - public: - ScopedSettingsStorageFactory(); - - explicit ScopedSettingsStorageFactory( - const scoped_refptr<SettingsStorageFactory>& delegate); - - // Sets the delegate factory (equivalent to scoped_ptr::reset). - void Reset(const scoped_refptr<SettingsStorageFactory>& delegate); - - // SettingsStorageFactory implementation. - ValueStore* Create(const base::FilePath& base_path, - const std::string& extension_id) override; - void DeleteDatabaseIfExists(const base::FilePath& base_path, - const std::string& extension_id) override; - - private: - // SettingsStorageFactory is refcounted. - ~ScopedSettingsStorageFactory() override; - - scoped_refptr<SettingsStorageFactory> delegate_; -}; - } // namespace settings_test_util } // namespace extensions diff --git a/extensions/browser/api/storage/storage_api_unittest.cc b/extensions/browser/api/storage/storage_api_unittest.cc index f987db2..4a013db 100644 --- a/extensions/browser/api/storage/storage_api_unittest.cc +++ b/extensions/browser/api/storage/storage_api_unittest.cc @@ -9,7 +9,6 @@ #include "base/strings/stringprintf.h" #include "content/public/test/test_browser_context.h" #include "extensions/browser/api/extensions_api_client.h" -#include "extensions/browser/api/storage/leveldb_settings_storage_factory.h" #include "extensions/browser/api/storage/settings_storage_quota_enforcer.h" #include "extensions/browser/api/storage/settings_test_util.h" #include "extensions/browser/api/storage/storage_api.h" @@ -20,6 +19,7 @@ #include "extensions/browser/test_extensions_browser_client.h" #include "extensions/browser/value_store/leveldb_value_store.h" #include "extensions/browser/value_store/value_store.h" +#include "extensions/browser/value_store/value_store_factory_impl.h" #include "extensions/common/manifest.h" #include "extensions/common/test_util.h" #include "third_party/leveldatabase/src/include/leveldb/db.h" @@ -32,8 +32,9 @@ namespace { // Caller owns the returned object. scoped_ptr<KeyedService> CreateStorageFrontendForTesting( content::BrowserContext* context) { - return StorageFrontend::CreateForTesting(new LeveldbSettingsStorageFactory(), - context); + scoped_refptr<ValueStoreFactory> factory = + new ValueStoreFactoryImpl(context->GetPath()); + return StorageFrontend::CreateForTesting(factory, context); } scoped_ptr<KeyedService> BuildEventRouter(content::BrowserContext* context) { @@ -105,6 +106,8 @@ TEST_F(StorageApiUnittest, RestoreCorruptedStorage) { settings_namespace::LOCAL, StorageFrontend::Get(browser_context())); ASSERT_TRUE(store); + // TODO(cmumford): Modify test as this requires that the factory always + // creates instances of LeveldbValueStore. SettingsStorageQuotaEnforcer* quota_store = static_cast<SettingsStorageQuotaEnforcer*>(store); LeveldbValueStore* leveldb_store = diff --git a/extensions/browser/api/storage/storage_frontend.cc b/extensions/browser/api/storage/storage_frontend.cc index 26d6ec8..911a492 100644 --- a/extensions/browser/api/storage/storage_frontend.cc +++ b/extensions/browser/api/storage/storage_frontend.cc @@ -16,10 +16,11 @@ #include "content/public/browser/browser_context.h" #include "content/public/browser/browser_thread.h" #include "extensions/browser/api/extensions_api_client.h" -#include "extensions/browser/api/storage/leveldb_settings_storage_factory.h" #include "extensions/browser/api/storage/local_value_store_cache.h" #include "extensions/browser/event_router.h" #include "extensions/browser/extension_registry.h" +#include "extensions/browser/extension_system.h" +#include "extensions/browser/value_store/value_store_factory.h" #include "extensions/common/api/storage.h" using content::BrowserContext; @@ -69,25 +70,23 @@ StorageFrontend* StorageFrontend::Get(BrowserContext* context) { // static scoped_ptr<StorageFrontend> StorageFrontend::CreateForTesting( - const scoped_refptr<SettingsStorageFactory>& storage_factory, + const scoped_refptr<ValueStoreFactory>& storage_factory, BrowserContext* context) { return make_scoped_ptr(new StorageFrontend(storage_factory, context)); } StorageFrontend::StorageFrontend(BrowserContext* context) - : browser_context_(context) { - Init(new LeveldbSettingsStorageFactory()); + : StorageFrontend(ExtensionSystem::Get(context)->store_factory(), context) { } StorageFrontend::StorageFrontend( - const scoped_refptr<SettingsStorageFactory>& factory, + const scoped_refptr<ValueStoreFactory>& factory, BrowserContext* context) : browser_context_(context) { Init(factory); } -void StorageFrontend::Init( - const scoped_refptr<SettingsStorageFactory>& factory) { +void StorageFrontend::Init(const scoped_refptr<ValueStoreFactory>& factory) { TRACE_EVENT0("browser,startup", "StorageFrontend::Init") SCOPED_UMA_HISTOGRAM_TIMER("Extensions.StorageFrontendInitTime"); @@ -98,8 +97,7 @@ void StorageFrontend::Init( observers_->AddObserver(browser_context_observer_.get()); - caches_[settings_namespace::LOCAL] = - new LocalValueStoreCache(factory, browser_context_->GetPath()); + caches_[settings_namespace::LOCAL] = new LocalValueStoreCache(factory); // Add any additional caches the embedder supports (for example, caches // for chrome.storage.managed and chrome.storage.sync). diff --git a/extensions/browser/api/storage/storage_frontend.h b/extensions/browser/api/storage/storage_frontend.h index 372fd78..6d3427f 100644 --- a/extensions/browser/api/storage/storage_frontend.h +++ b/extensions/browser/api/storage/storage_frontend.h @@ -13,7 +13,6 @@ #include "base/memory/scoped_ptr.h" #include "extensions/browser/api/storage/settings_namespace.h" #include "extensions/browser/api/storage/settings_observer.h" -#include "extensions/browser/api/storage/settings_storage_factory.h" #include "extensions/browser/api/storage/value_store_cache.h" #include "extensions/browser/browser_context_keyed_api_factory.h" @@ -23,6 +22,8 @@ class BrowserContext; namespace extensions { +class ValueStoreFactory; + // The component of the Storage API which runs on the UI thread. class StorageFrontend : public BrowserContextKeyedAPI { public: @@ -31,7 +32,7 @@ class StorageFrontend : public BrowserContextKeyedAPI { // Creates with a specific |storage_factory|. static scoped_ptr<StorageFrontend> CreateForTesting( - const scoped_refptr<SettingsStorageFactory>& storage_factory, + const scoped_refptr<ValueStoreFactory>& storage_factory, content::BrowserContext* context); // Public so tests can create and delete their own instances. @@ -74,10 +75,10 @@ class StorageFrontend : public BrowserContextKeyedAPI { explicit StorageFrontend(content::BrowserContext* context); // Constructor for tests. - StorageFrontend(const scoped_refptr<SettingsStorageFactory>& storage_factory, + StorageFrontend(const scoped_refptr<ValueStoreFactory>& storage_factory, content::BrowserContext* context); - void Init(const scoped_refptr<SettingsStorageFactory>& storage_factory); + void Init(const scoped_refptr<ValueStoreFactory>& storage_factory); // The (non-incognito) browser context this Frontend belongs to. content::BrowserContext* const browser_context_; diff --git a/extensions/browser/api/storage/storage_frontend_unittest.cc b/extensions/browser/api/storage/storage_frontend_unittest.cc index a434a3c..5995ceb 100644 --- a/extensions/browser/api/storage/storage_frontend_unittest.cc +++ b/extensions/browser/api/storage/storage_frontend_unittest.cc @@ -12,12 +12,12 @@ #include "content/public/test/test_browser_context.h" #include "content/public/test/test_browser_thread.h" #include "extensions/browser/api/extensions_api_client.h" -#include "extensions/browser/api/storage/leveldb_settings_storage_factory.h" #include "extensions/browser/api/storage/settings_namespace.h" #include "extensions/browser/api/storage/settings_test_util.h" #include "extensions/browser/api/storage/storage_frontend.h" #include "extensions/browser/extensions_test.h" #include "extensions/browser/value_store/value_store.h" +#include "extensions/browser/value_store/value_store_factory_impl.h" #include "testing/gtest/include/gtest/gtest.h" using content::BrowserThread; @@ -40,13 +40,13 @@ const ValueStore::WriteOptions DEFAULTS = ValueStore::DEFAULTS; class ExtensionSettingsFrontendTest : public ExtensionsTest { public: ExtensionSettingsFrontendTest() - : storage_factory_(new util::ScopedSettingsStorageFactory()), - ui_thread_(BrowserThread::UI, base::MessageLoop::current()), + : ui_thread_(BrowserThread::UI, base::MessageLoop::current()), file_thread_(BrowserThread::FILE, base::MessageLoop::current()) {} void SetUp() override { ExtensionsTest::SetUp(); ASSERT_TRUE(temp_dir_.CreateUniqueTempDir()); + storage_factory_ = new ValueStoreFactoryImpl(temp_dir_.path()); ResetFrontend(); } @@ -59,14 +59,13 @@ class ExtensionSettingsFrontendTest : public ExtensionsTest { protected: void ResetFrontend() { - storage_factory_->Reset(new LeveldbSettingsStorageFactory()); frontend_ = StorageFrontend::CreateForTesting(storage_factory_, browser_context()); } base::ScopedTempDir temp_dir_; scoped_ptr<StorageFrontend> frontend_; - scoped_refptr<util::ScopedSettingsStorageFactory> storage_factory_; + scoped_refptr<ValueStoreFactoryImpl> storage_factory_; private: base::MessageLoop message_loop_; diff --git a/extensions/browser/extension_system.h b/extensions/browser/extension_system.h index bbd8ae5..5382d3f 100644 --- a/extensions/browser/extension_system.h +++ b/extensions/browser/extension_system.h @@ -43,6 +43,7 @@ class RuntimeData; class ServiceWorkerManager; class SharedUserScriptMaster; class StateStore; +class ValueStoreFactory; // ExtensionSystem manages the lifetime of many of the services used by the // extensions and apps system, and it handles startup and shutdown as needed. @@ -85,6 +86,9 @@ class ExtensionSystem : public KeyedService { // The rules store is created at startup. virtual StateStore* rules_store() = 0; + // Returns the |ValueStore| factory created at startup. + virtual scoped_refptr<ValueStoreFactory> store_factory() = 0; + // Returns the IO-thread-accessible extension data. virtual InfoMap* info_map() = 0; diff --git a/extensions/browser/mock_extension_system.cc b/extensions/browser/mock_extension_system.cc index efcc546..6658832 100644 --- a/extensions/browser/mock_extension_system.cc +++ b/extensions/browser/mock_extension_system.cc @@ -4,6 +4,7 @@ #include "extensions/browser/mock_extension_system.h" +#include "extensions/browser/value_store/value_store_factory.h" #include "extensions/common/extension_set.h" namespace extensions { @@ -46,6 +47,10 @@ StateStore* MockExtensionSystem::rules_store() { return nullptr; } +scoped_refptr<ValueStoreFactory> MockExtensionSystem::store_factory() { + return nullptr; +} + InfoMap* MockExtensionSystem::info_map() { return nullptr; } diff --git a/extensions/browser/mock_extension_system.h b/extensions/browser/mock_extension_system.h index 3151edb..b926baa 100644 --- a/extensions/browser/mock_extension_system.h +++ b/extensions/browser/mock_extension_system.h @@ -35,6 +35,7 @@ class MockExtensionSystem : public ExtensionSystem { SharedUserScriptMaster* shared_user_script_master() override; StateStore* state_store() override; StateStore* rules_store() override; + scoped_refptr<ValueStoreFactory> store_factory() override; InfoMap* info_map() override; QuotaService* quota_service() override; AppSorting* app_sorting() override; diff --git a/extensions/browser/state_store.cc b/extensions/browser/state_store.cc index ff4ca02..2c8c9f9 100644 --- a/extensions/browser/state_store.cc +++ b/extensions/browser/state_store.cc @@ -14,6 +14,7 @@ #include "content/public/browser/notification_service.h" #include "content/public/browser/notification_types.h" #include "extensions/browser/extension_registry.h" +#include "extensions/browser/value_store/value_store_factory.h" #include "extensions/common/extension.h" namespace { @@ -69,11 +70,10 @@ void StateStore::DelayedTaskQueue::SetReady() { } StateStore::StateStore(content::BrowserContext* context, - const std::string& uma_client_name, - const base::FilePath& db_path, + const scoped_refptr<ValueStoreFactory>& store_factory, + ValueStoreFrontend::BackendType backend_type, bool deferred_load) - : db_path_(db_path), - uma_client_name_(uma_client_name), + : store_(new ValueStoreFrontend(store_factory, backend_type)), task_queue_(new DelayedTaskQueue()), extension_registry_observer_(this) { extension_registry_observer_.Add(ExtensionRegistry::Get(context)); @@ -90,17 +90,6 @@ StateStore::StateStore(content::BrowserContext* context, } } -StateStore::StateStore(content::BrowserContext* context, - scoped_ptr<ValueStore> value_store) - : store_(std::move(value_store)), - task_queue_(new DelayedTaskQueue()), - extension_registry_observer_(this) { - extension_registry_observer_.Add(ExtensionRegistry::Get(context)); - - // This constructor is for testing. No need to delay Init. - Init(); -} - StateStore::~StateStore() { } @@ -115,25 +104,23 @@ void StateStore::RegisterKey(const std::string& key) { void StateStore::GetExtensionValue(const std::string& extension_id, const std::string& key, ReadCallback callback) { - task_queue_->InvokeWhenReady(base::Bind(&ValueStoreFrontend::Get, - base::Unretained(&store_), - GetFullKey(extension_id, key), - callback)); + task_queue_->InvokeWhenReady( + base::Bind(&ValueStoreFrontend::Get, base::Unretained(store_.get()), + GetFullKey(extension_id, key), callback)); } void StateStore::SetExtensionValue(const std::string& extension_id, const std::string& key, scoped_ptr<base::Value> value) { - task_queue_->InvokeWhenReady(base::Bind(&ValueStoreFrontend::Set, - base::Unretained(&store_), - GetFullKey(extension_id, key), - base::Passed(&value))); + task_queue_->InvokeWhenReady( + base::Bind(&ValueStoreFrontend::Set, base::Unretained(store_.get()), + GetFullKey(extension_id, key), base::Passed(&value))); } void StateStore::RemoveExtensionValue(const std::string& extension_id, const std::string& key) { task_queue_->InvokeWhenReady(base::Bind(&ValueStoreFrontend::Remove, - base::Unretained(&store_), + base::Unretained(store_.get()), GetFullKey(extension_id, key))); } @@ -170,8 +157,9 @@ void StateStore::Init() { if (IsInitialized()) return; - if (!db_path_.empty()) - store_.Init(uma_client_name_, db_path_); + // TODO(cmumford): The store now always lazily initializes upon first access. + // A follow-on CL will remove this deferred initialization implementation + // which is now vestigial. task_queue_->SetReady(); } @@ -190,7 +178,7 @@ void StateStore::RemoveKeysForExtension(const std::string& extension_id) { key != registered_keys_.end(); ++key) { task_queue_->InvokeWhenReady(base::Bind(&ValueStoreFrontend::Remove, - base::Unretained(&store_), + base::Unretained(store_.get()), GetFullKey(extension_id, *key))); } } diff --git a/extensions/browser/state_store.h b/extensions/browser/state_store.h index 3809c1e3..5cc0037 100644 --- a/extensions/browser/state_store.h +++ b/extensions/browser/state_store.h @@ -23,6 +23,7 @@ class BrowserContext; namespace extensions { class ExtensionRegistry; +class ValueStoreFactory; // A storage area for per-extension state that needs to be persisted to disk. class StateStore : public base::SupportsWeakPtr<StateStore>, @@ -34,8 +35,8 @@ class StateStore : public base::SupportsWeakPtr<StateStore>, // If |deferred_load| is true, we won't load the database until the first // page has been loaded. StateStore(content::BrowserContext* context, - const std::string& uma_client_name, - const base::FilePath& db_path, + const scoped_refptr<ValueStoreFactory>& store_factory, + ValueStoreFrontend::BackendType backend_type, bool deferred_load); // This variant is useful for testing (using a mock ValueStore). StateStore(content::BrowserContext* context, scoped_ptr<ValueStore> store); @@ -94,14 +95,8 @@ class StateStore : public base::SupportsWeakPtr<StateStore>, bool is_update, const std::string& old_name) override; - // Path to our database, on disk. Empty during testing. - base::FilePath db_path_; - - // Database client name used for UMA logging by database backend. - const std::string uma_client_name_; - // The store that holds our key/values. - ValueStoreFrontend store_; + scoped_ptr<ValueStoreFrontend> store_; // List of all known keys. They will be cleared for each extension when it is // (un)installed. diff --git a/extensions/browser/value_store/legacy_value_store_factory.cc b/extensions/browser/value_store/legacy_value_store_factory.cc new file mode 100644 index 0000000..bafb93e --- /dev/null +++ b/extensions/browser/value_store/legacy_value_store_factory.cc @@ -0,0 +1,264 @@ +// Copyright 2016 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 "extensions/browser/value_store/legacy_value_store_factory.h" + +#include <string> + +#include "base/files/file_enumerator.h" +#include "base/files/file_util.h" +#include "content/public/browser/browser_thread.h" +#include "extensions/browser/value_store/leveldb_value_store.h" +#include "extensions/common/constants.h" +#include "extensions/common/extension.h" + +using base::AutoLock; +using content::BrowserThread; + +namespace { + +// Statistics are logged to UMA with these strings as part of histogram name. +// They can all be found under Extensions.Database.Open.<client>. Changing this +// needs to synchronize with histograms.xml, AND will also become incompatible +// with older browsers still reporting the previous values. +const char kSettingsDatabaseUMAClientName[] = "Settings"; +const char kRulesDatabaseUMAClientName[] = "Rules"; +const char kStateDatabaseUMAClientName[] = "State"; + +bool ValidDBExists(const base::FilePath& path) { + // TODO(cmumford): Enhance to detect if dir contains valid database. + return base::DirectoryExists(path); +} + +} // namespace + +namespace extensions { + +// +// ModelSettings +// +LegacyValueStoreFactory::ModelSettings::ModelSettings( + const base::FilePath& data_path) + : data_path_(data_path) {} + +base::FilePath LegacyValueStoreFactory::ModelSettings::GetDBPath( + const ExtensionId& extension_id) const { + return data_path_.AppendASCII(extension_id); +} + +bool LegacyValueStoreFactory::ModelSettings::DeleteData( + const ExtensionId& extension_id) { + return base::DeleteFile(GetDBPath(extension_id), true /* recursive */); +} + +bool LegacyValueStoreFactory::ModelSettings::DataExists( + const ExtensionId& extension_id) const { + return ValidDBExists(GetDBPath(extension_id)); +} + +std::set<ExtensionId> +LegacyValueStoreFactory::ModelSettings::GetKnownExtensionIDs() const { + DCHECK_CURRENTLY_ON(BrowserThread::FILE); + std::set<ExtensionId> result; + + // Leveldb databases are directories inside |base_path_|. + base::FileEnumerator extension_dirs(data_path_, false, + base::FileEnumerator::DIRECTORIES); + while (!extension_dirs.Next().empty()) { + base::FilePath extension_dir = extension_dirs.GetInfo().GetName(); + DCHECK(!extension_dir.IsAbsolute()); + // Extension ID's are 'a'..'p', so any directory within this folder will + // either be ASCII, or created by some other application and safe to ignore. + std::string maybe_as_ascii(extension_dir.MaybeAsASCII()); + if (!maybe_as_ascii.empty()) { + result.insert(maybe_as_ascii); + } + } + + return result; +} + +// +// SettingsRoot +// + +LegacyValueStoreFactory::SettingsRoot::SettingsRoot( + const base::FilePath& base_path, + const std::string& extension_dirname, + const std::string& app_dirname) { + if (!extension_dirname.empty()) + extensions_.reset( + new ModelSettings(base_path.AppendASCII(extension_dirname))); + + if (!app_dirname.empty()) + apps_.reset(new ModelSettings(base_path.AppendASCII(app_dirname))); +} + +LegacyValueStoreFactory::SettingsRoot::~SettingsRoot() = default; + +const LegacyValueStoreFactory::ModelSettings* +LegacyValueStoreFactory::SettingsRoot::GetModel(ModelType model_type) const { + switch (model_type) { + case ValueStoreFactory::ModelType::APP: + DCHECK(apps_ != nullptr); + return apps_.get(); + case ValueStoreFactory::ModelType::EXTENSION: + DCHECK(extensions_ != nullptr); + return extensions_.get(); + } + NOTREACHED(); + return nullptr; +} + +LegacyValueStoreFactory::ModelSettings* +LegacyValueStoreFactory::SettingsRoot::GetModel(ModelType model_type) { + switch (model_type) { + case ValueStoreFactory::ModelType::APP: + DCHECK(apps_ != nullptr); + return apps_.get(); + case ValueStoreFactory::ModelType::EXTENSION: + DCHECK(extensions_ != nullptr); + return extensions_.get(); + } + NOTREACHED(); + return nullptr; +} + +std::set<ExtensionId> +LegacyValueStoreFactory::SettingsRoot::GetKnownExtensionIDs( + ModelType model_type) const { + DCHECK_CURRENTLY_ON(BrowserThread::FILE); + switch (model_type) { + case ValueStoreFactory::ModelType::APP: + DCHECK(apps_ != nullptr); + return apps_->GetKnownExtensionIDs(); + case ValueStoreFactory::ModelType::EXTENSION: + DCHECK(extensions_ != nullptr); + return extensions_->GetKnownExtensionIDs(); + } + NOTREACHED(); + return std::set<ExtensionId>(); +} + +// +// LegacyValueStoreFactory +// + +LegacyValueStoreFactory::LegacyValueStoreFactory( + const base::FilePath& profile_path) + : profile_path_(profile_path), + local_settings_(profile_path, + kLocalExtensionSettingsDirectoryName, + kLocalAppSettingsDirectoryName), + sync_settings_(profile_path, + kSyncExtensionSettingsDirectoryName, + kSyncAppSettingsDirectoryName), + // Currently no such thing as a managed app - only an extension. + managed_settings_(profile_path, kManagedSettingsDirectoryName, "") {} + +LegacyValueStoreFactory::~LegacyValueStoreFactory() = default; + +bool LegacyValueStoreFactory::RulesDBExists() const { + return ValidDBExists(GetRulesDBPath()); +} + +bool LegacyValueStoreFactory::StateDBExists() const { + return ValidDBExists(GetStateDBPath()); +} + +scoped_ptr<ValueStore> LegacyValueStoreFactory::CreateRulesStore() { + return make_scoped_ptr( + new LeveldbValueStore(kRulesDatabaseUMAClientName, GetRulesDBPath())); +} + +scoped_ptr<ValueStore> LegacyValueStoreFactory::CreateStateStore() { + return make_scoped_ptr( + new LeveldbValueStore(kStateDatabaseUMAClientName, GetStateDBPath())); +} + +scoped_ptr<ValueStore> LegacyValueStoreFactory::CreateSettingsStore( + settings_namespace::Namespace settings_namespace, + ModelType model_type, + const ExtensionId& extension_id) { + const ModelSettings* settings_root = + GetSettingsRoot(settings_namespace).GetModel(model_type); + DCHECK(settings_root != nullptr); + return make_scoped_ptr(new LeveldbValueStore( + kSettingsDatabaseUMAClientName, settings_root->GetDBPath(extension_id))); +} + +void LegacyValueStoreFactory::DeleteSettings( + settings_namespace::Namespace settings_namespace, + ModelType model_type, + const ExtensionId& extension_id) { + // TODO(cmumford): Verify that we always need to be called on FILE thread. + DCHECK_CURRENTLY_ON(BrowserThread::FILE); + ModelSettings* model_settings = + GetSettingsRoot(settings_namespace).GetModel(model_type); + if (model_settings == nullptr) { + NOTREACHED(); + return; + } + model_settings->DeleteData(extension_id); +} + +bool LegacyValueStoreFactory::HasSettings( + settings_namespace::Namespace settings_namespace, + ModelType model_type, + const ExtensionId& extension_id) { + const ModelSettings* model_settings = + GetSettingsRoot(settings_namespace).GetModel(model_type); + if (model_settings == nullptr) + return false; + return model_settings->DataExists(extension_id); +} + +std::set<ExtensionId> LegacyValueStoreFactory::GetKnownExtensionIDs( + settings_namespace::Namespace settings_type, + ModelType model_type) const { + return GetSettingsRoot(settings_type).GetKnownExtensionIDs(model_type); +} + +const LegacyValueStoreFactory::SettingsRoot& +LegacyValueStoreFactory::GetSettingsRoot( + settings_namespace::Namespace settings_namespace) const { + switch (settings_namespace) { + case settings_namespace::LOCAL: + return local_settings_; + case settings_namespace::SYNC: + return sync_settings_; + case settings_namespace::MANAGED: + return managed_settings_; + case settings_namespace::INVALID: + break; + } + NOTREACHED(); + return local_settings_; +} + +LegacyValueStoreFactory::SettingsRoot& LegacyValueStoreFactory::GetSettingsRoot( + settings_namespace::Namespace settings_namespace) { + switch (settings_namespace) { + case settings_namespace::LOCAL: + return local_settings_; + case settings_namespace::SYNC: + return sync_settings_; + case settings_namespace::MANAGED: + return managed_settings_; + case settings_namespace::INVALID: + break; + } + NOTREACHED(); + return local_settings_; +} + +base::FilePath LegacyValueStoreFactory::GetRulesDBPath() const { + return profile_path_.AppendASCII(kRulesStoreName); +} + +base::FilePath LegacyValueStoreFactory::GetStateDBPath() const { + return profile_path_.AppendASCII(kStateStoreName); +} + +} // namespace extensions diff --git a/extensions/browser/value_store/legacy_value_store_factory.h b/extensions/browser/value_store/legacy_value_store_factory.h new file mode 100644 index 0000000..65db027 --- /dev/null +++ b/extensions/browser/value_store/legacy_value_store_factory.h @@ -0,0 +1,111 @@ +// Copyright 2016 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. + +#ifndef EXTENSIONS_BROWSER_VALUE_STORE_LEGACY_VALUE_STORE_FACTORY_H_ +#define EXTENSIONS_BROWSER_VALUE_STORE_LEGACY_VALUE_STORE_FACTORY_H_ + +#include <set> +#include <string> + +#include "base/files/file_path.h" +#include "base/memory/ref_counted.h" +#include "base/memory/scoped_ptr.h" +#include "extensions/browser/value_store/value_store.h" +#include "extensions/browser/value_store/value_store_factory.h" +#include "extensions/common/extension.h" + +namespace extensions { + +// A factory to create legacy ValueStore instances for storing extension +// state/rules/settings. "legacy" refers to the initial storage implementation +// which created a settings database per extension. +class LegacyValueStoreFactory : public ValueStoreFactory { + public: + explicit LegacyValueStoreFactory(const base::FilePath& profile_path); + + bool RulesDBExists() const; + bool StateDBExists() const; + + // ValueStoreFactory: + scoped_ptr<ValueStore> CreateRulesStore() override; + scoped_ptr<ValueStore> CreateStateStore() override; + scoped_ptr<ValueStore> CreateSettingsStore( + settings_namespace::Namespace settings_namespace, + ModelType model_type, + const ExtensionId& extension_id) override; + + void DeleteSettings(settings_namespace::Namespace settings_namespace, + ModelType model_type, + const ExtensionId& extension_id) override; + bool HasSettings(settings_namespace::Namespace settings_namespace, + ModelType model_type, + const ExtensionId& extension_id) override; + std::set<ExtensionId> GetKnownExtensionIDs( + settings_namespace::Namespace settings_namespace, + ModelType model_type) const override; + + private: + friend class base::RefCounted<LegacyValueStoreFactory>; + + // Manages a collection of legacy settings databases all within a common + // directory. + class ModelSettings { + public: + explicit ModelSettings(const base::FilePath& data_path); + + base::FilePath GetDBPath(const ExtensionId& extension_id) const; + bool DeleteData(const ExtensionId& extension_id); + bool DataExists(const ExtensionId& extension_id) const; + std::set<ExtensionId> GetKnownExtensionIDs() const; + + private: + // The path containing all settings databases under this root. + const base::FilePath data_path_; + + DISALLOW_COPY_AND_ASSIGN(ModelSettings); + }; + + // Manages two collections of legacy settings databases (apps & extensions) + // within a common base directory. + class SettingsRoot { + public: + // If either |extension_dirname| or |app_dirname| are empty then that + // ModelSetting will *not* be created. + SettingsRoot(const base::FilePath& base_path, + const std::string& extension_dirname, + const std::string& app_dirname); + ~SettingsRoot(); + + std::set<ExtensionId> GetKnownExtensionIDs(ModelType model_type) const; + const ModelSettings* GetModel(ModelType model_type) const; + ModelSettings* GetModel(ModelType model_type); + + private: + scoped_ptr<ModelSettings> extensions_; + scoped_ptr<ModelSettings> apps_; + + DISALLOW_COPY_AND_ASSIGN(SettingsRoot); + }; + + ~LegacyValueStoreFactory() override; + + const SettingsRoot& GetSettingsRoot( + settings_namespace::Namespace settings_namespace) const; + SettingsRoot& GetSettingsRoot( + settings_namespace::Namespace settings_namespace); + + base::FilePath GetRulesDBPath() const; + base::FilePath GetStateDBPath() const; + + const base::FilePath profile_path_; + SettingsRoot local_settings_; + SettingsRoot sync_settings_; + SettingsRoot managed_settings_; + + DISALLOW_COPY_AND_ASSIGN(LegacyValueStoreFactory); +}; + +} // namespace extensions + +#endif // EXTENSIONS_BROWSER_VALUE_STORE_LEGACY_VALUE_STORE_FACTORY_H_ diff --git a/extensions/browser/value_store/test_value_store_factory.cc b/extensions/browser/value_store/test_value_store_factory.cc new file mode 100644 index 0000000..f5135df --- /dev/null +++ b/extensions/browser/value_store/test_value_store_factory.cc @@ -0,0 +1,190 @@ +// Copyright 2016 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 "extensions/browser/value_store/test_value_store_factory.h" + +#include "extensions/browser/value_store/leveldb_value_store.h" +#include "extensions/browser/value_store/testing_value_store.h" + +namespace { + +const char kUMAClientName[] = "Test"; + +} // namespace + +namespace extensions { + +using SettingsNamespace = settings_namespace::Namespace; + +TestValueStoreFactory::StorageHelper::StorageHelper() = default; + +TestValueStoreFactory::StorageHelper::~StorageHelper() = default; + +std::set<ExtensionId> +TestValueStoreFactory::StorageHelper::GetKnownExtensionIDs( + ModelType model_type) const { + std::set<ExtensionId> ids; + switch (model_type) { + case ValueStoreFactory::ModelType::APP: + for (const auto& key : app_stores_) + ids.insert(key.first); + break; + case ValueStoreFactory::ModelType::EXTENSION: + for (const auto& key : extension_stores_) + ids.insert(key.first); + break; + } + return ids; +} + +void TestValueStoreFactory::StorageHelper::Reset() { + app_stores_.clear(); + extension_stores_.clear(); +} + +ValueStore* TestValueStoreFactory::StorageHelper::AddValueStore( + const ExtensionId& extension_id, + ValueStore* value_store, + ModelType model_type) { + if (model_type == ValueStoreFactory::ModelType::APP) { + DCHECK(app_stores_.find(extension_id) == app_stores_.end()); + app_stores_[extension_id] = value_store; + } else { + DCHECK(extension_stores_.find(extension_id) == extension_stores_.end()); + extension_stores_[extension_id] = value_store; + } + return value_store; +} + +void TestValueStoreFactory::StorageHelper::DeleteSettings( + const ExtensionId& extension_id, + ModelType model_type) { + switch (model_type) { + case ValueStoreFactory::ModelType::APP: + app_stores_.erase(extension_id); + break; + case ValueStoreFactory::ModelType::EXTENSION: + extension_stores_.erase(extension_id); + break; + } +} + +bool TestValueStoreFactory::StorageHelper::HasSettings( + const ExtensionId& extension_id, + ModelType model_type) const { + switch (model_type) { + case ValueStoreFactory::ModelType::APP: + return app_stores_.find(extension_id) != app_stores_.end(); + case ValueStoreFactory::ModelType::EXTENSION: + return extension_stores_.find(extension_id) != extension_stores_.end(); + } + NOTREACHED(); + return false; +} + +ValueStore* TestValueStoreFactory::StorageHelper::GetExisting( + const ExtensionId& extension_id) const { + auto it = app_stores_.find(extension_id); + if (it != app_stores_.end()) + return it->second; + it = extension_stores_.find(extension_id); + if (it != extension_stores_.end()) + return it->second; + return nullptr; +} + +TestValueStoreFactory::TestValueStoreFactory() = default; + +TestValueStoreFactory::TestValueStoreFactory(const base::FilePath& db_path) + : db_path_(db_path) {} + +TestValueStoreFactory::~TestValueStoreFactory() {} + +scoped_ptr<ValueStore> TestValueStoreFactory::CreateRulesStore() { + if (db_path_.empty()) + last_created_store_ = new TestingValueStore(); + else + last_created_store_ = new LeveldbValueStore(kUMAClientName, db_path_); + return make_scoped_ptr(last_created_store_); +} + +scoped_ptr<ValueStore> TestValueStoreFactory::CreateStateStore() { + return CreateRulesStore(); +} + +TestValueStoreFactory::StorageHelper& TestValueStoreFactory::GetStorageHelper( + SettingsNamespace settings_namespace) { + switch (settings_namespace) { + case settings_namespace::LOCAL: + return local_helper_; + case settings_namespace::SYNC: + return sync_helper_; + case settings_namespace::MANAGED: + return managed_helper_; + case settings_namespace::INVALID: + break; + } + NOTREACHED(); + return local_helper_; +} + +scoped_ptr<ValueStore> TestValueStoreFactory::CreateSettingsStore( + SettingsNamespace settings_namespace, + ModelType model_type, + const ExtensionId& extension_id) { + scoped_ptr<ValueStore> settings_store(CreateRulesStore()); + // Note: This factory is purposely keeping the raw pointers to each ValueStore + // created. Tests using TestValueStoreFactory must be careful to keep + // those ValueStore's alive for the duration of their test. + GetStorageHelper(settings_namespace) + .AddValueStore(extension_id, settings_store.get(), model_type); + return settings_store; +} + +ValueStore* TestValueStoreFactory::LastCreatedStore() const { + return last_created_store_; +} + +void TestValueStoreFactory::DeleteSettings(SettingsNamespace settings_namespace, + ModelType model_type, + const ExtensionId& extension_id) { + GetStorageHelper(settings_namespace).DeleteSettings(extension_id, model_type); +} + +bool TestValueStoreFactory::HasSettings(SettingsNamespace settings_namespace, + ModelType model_type, + const ExtensionId& extension_id) { + return GetStorageHelper(settings_namespace) + .HasSettings(extension_id, model_type); +} + +std::set<ExtensionId> TestValueStoreFactory::GetKnownExtensionIDs( + SettingsNamespace settings_namespace, + ModelType model_type) const { + return const_cast<TestValueStoreFactory*>(this) + ->GetStorageHelper(settings_namespace) + .GetKnownExtensionIDs(model_type); +} + +ValueStore* TestValueStoreFactory::GetExisting( + const ExtensionId& extension_id) const { + ValueStore* existing_store = local_helper_.GetExisting(extension_id); + if (existing_store) + return existing_store; + existing_store = sync_helper_.GetExisting(extension_id); + if (existing_store) + return existing_store; + existing_store = managed_helper_.GetExisting(extension_id); + DCHECK(existing_store != nullptr); + return existing_store; +} + +void TestValueStoreFactory::Reset() { + last_created_store_ = nullptr; + local_helper_.Reset(); + sync_helper_.Reset(); + managed_helper_.Reset(); +} + +} // namespace extensions diff --git a/extensions/browser/value_store/test_value_store_factory.h b/extensions/browser/value_store/test_value_store_factory.h new file mode 100644 index 0000000..86fbee77 --- /dev/null +++ b/extensions/browser/value_store/test_value_store_factory.h @@ -0,0 +1,95 @@ +// Copyright 2016 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. + +#ifndef EXTENSIONS_BROWSER_VALUE_STORE_TEST_VALUE_STORE_FACTORY_H_ +#define EXTENSIONS_BROWSER_VALUE_STORE_TEST_VALUE_STORE_FACTORY_H_ + +#include <map> +#include <set> + +#include "base/memory/scoped_ptr.h" +#include "extensions/browser/value_store/value_store_factory.h" +#include "extensions/common/extension.h" + +class ValueStore; + +namespace extensions { + +// Used for tests when a new test ValueStore is required. Will either open a +// database on disk (if path provided) returning a |LeveldbValueStore|. +// Otherwise a new |TestingValueStore| instance will be returned. +class TestValueStoreFactory : public ValueStoreFactory { + public: + TestValueStoreFactory(); + explicit TestValueStoreFactory(const base::FilePath& db_path); + + // ValueStoreFactory + scoped_ptr<ValueStore> CreateRulesStore() override; + scoped_ptr<ValueStore> CreateStateStore() override; + scoped_ptr<ValueStore> CreateSettingsStore( + settings_namespace::Namespace settings_namespace, + ModelType model_type, + const ExtensionId& extension_id) override; + void DeleteSettings(settings_namespace::Namespace settings_namespace, + ModelType model_type, + const ExtensionId& extension_id) override; + bool HasSettings(settings_namespace::Namespace settings_namespace, + ModelType model_type, + const ExtensionId& extension_id) override; + std::set<ExtensionId> GetKnownExtensionIDs( + settings_namespace::Namespace settings_namespace, + ModelType model_type) const override; + + // Return the last created |ValueStore|. Use with caution as this may return + // a dangling pointer since the creator now owns the ValueStore which can be + // deleted at any time. + ValueStore* LastCreatedStore() const; + // Return a previously created |ValueStore| for an extension. + ValueStore* GetExisting(const ExtensionId& extension_id) const; + // Reset this class (as if just created). + void Reset(); + + private: + // Manages a collection of |ValueStore|'s created for an app/extension. + // One of these exists for each setting type. + class StorageHelper { + public: + StorageHelper(); + ~StorageHelper(); + std::set<ExtensionId> GetKnownExtensionIDs(ModelType model_type) const; + ValueStore* AddValueStore(const ExtensionId& extension_id, + ValueStore* value_store, + ModelType model_type); + void DeleteSettings(const ExtensionId& extension_id, ModelType model_type); + bool HasSettings(const ExtensionId& extension_id, + ModelType model_type) const; + void Reset(); + ValueStore* GetExisting(const ExtensionId& extension_id) const; + + private: + std::map<ExtensionId, ValueStore*> app_stores_; + std::map<ExtensionId, ValueStore*> extension_stores_; + + DISALLOW_COPY_AND_ASSIGN(StorageHelper); + }; + + StorageHelper& GetStorageHelper( + settings_namespace::Namespace settings_namespace); + + ~TestValueStoreFactory() override; + base::FilePath db_path_; + ValueStore* last_created_store_ = nullptr; + + // None of these value stores are owned by this factory, so care must be + // taken when calling GetExisting. + StorageHelper local_helper_; + StorageHelper sync_helper_; + StorageHelper managed_helper_; + + DISALLOW_COPY_AND_ASSIGN(TestValueStoreFactory); +}; + +} // namespace extensions + +#endif // EXTENSIONS_BROWSER_VALUE_STORE_TEST_VALUE_STORE_FACTORY_H_ diff --git a/extensions/browser/value_store/value_store_factory.h b/extensions/browser/value_store/value_store_factory.h new file mode 100644 index 0000000..ec43547 --- /dev/null +++ b/extensions/browser/value_store/value_store_factory.h @@ -0,0 +1,67 @@ +// Copyright 2016 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. + +#ifndef EXTENSIONS_BROWSER_VALUE_STORE_VALUE_STORE_FACTORY_H_ +#define EXTENSIONS_BROWSER_VALUE_STORE_VALUE_STORE_FACTORY_H_ + +#include <set> + +#include "base/memory/ref_counted.h" +#include "base/memory/scoped_ptr.h" +#include "extensions/browser/api/storage/settings_namespace.h" +#include "extensions/common/extension.h" + +class ValueStore; + +namespace extensions { + +// Create new value stores for rules, state, or settings. For settings will +// also create stores for the specified namespace and model type. +// +// Note: This factory creates the lower level stores that directly read/write to +// disk. Sync/Managed stores are created directly, but delegate their +// calls to a |ValueStore| created by this interface. +class ValueStoreFactory : public base::RefCountedThreadSafe<ValueStoreFactory> { + public: + enum class ModelType { APP, EXTENSION }; + + // Create a |ValueStore| to contain rules data. + virtual scoped_ptr<ValueStore> CreateRulesStore() = 0; + + // Create a |ValueStore| to contain state data. + virtual scoped_ptr<ValueStore> CreateStateStore() = 0; + + // Create a |ValueStore| to contain settings data for a specific extension + // namespace and model type. + virtual scoped_ptr<ValueStore> CreateSettingsStore( + settings_namespace::Namespace settings_namespace, + ModelType model_type, + const ExtensionId& extension_id) = 0; + + // Delete all settings for specified given extension in the specified + // namespace/model_type. + virtual void DeleteSettings(settings_namespace::Namespace settings_namespace, + ModelType model_type, + const ExtensionId& extension_id) = 0; + + // Are there any settings stored in the specified namespace/model_type for + // the given extension? + virtual bool HasSettings(settings_namespace::Namespace settings_namespace, + ModelType model_type, + const ExtensionId& extension_id) = 0; + + // Return all extension ID's with settings stored in the given + // namespace/model_type. + virtual std::set<ExtensionId> GetKnownExtensionIDs( + settings_namespace::Namespace settings_namespace, + ModelType model_type) const = 0; + + protected: + friend class base::RefCountedThreadSafe<ValueStoreFactory>; + virtual ~ValueStoreFactory() {} +}; + +} // namespace extensions + +#endif // EXTENSIONS_BROWSER_VALUE_STORE_VALUE_STORE_FACTORY_H_ diff --git a/extensions/browser/value_store/value_store_factory_impl.cc b/extensions/browser/value_store/value_store_factory_impl.cc new file mode 100644 index 0000000..27324fd --- /dev/null +++ b/extensions/browser/value_store/value_store_factory_impl.cc @@ -0,0 +1,53 @@ +// Copyright 2016 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 "extensions/browser/value_store/value_store_factory_impl.h" + +#include "extensions/browser/value_store/legacy_value_store_factory.h" + +namespace extensions { + +using SettingsNamespace = settings_namespace::Namespace; + +ValueStoreFactoryImpl::ValueStoreFactoryImpl(const base::FilePath& profile_path) + : legacy_factory_(new LegacyValueStoreFactory(profile_path)) {} + +ValueStoreFactoryImpl::~ValueStoreFactoryImpl() = default; + +scoped_ptr<ValueStore> ValueStoreFactoryImpl::CreateRulesStore() { + return legacy_factory_->CreateRulesStore(); +} + +scoped_ptr<ValueStore> ValueStoreFactoryImpl::CreateStateStore() { + return legacy_factory_->CreateStateStore(); +} + +scoped_ptr<ValueStore> ValueStoreFactoryImpl::CreateSettingsStore( + SettingsNamespace settings_namespace, + ModelType model_type, + const ExtensionId& extension_id) { + return legacy_factory_->CreateSettingsStore(settings_namespace, model_type, + extension_id); +} + +void ValueStoreFactoryImpl::DeleteSettings(SettingsNamespace settings_namespace, + ModelType model_type, + const ExtensionId& extension_id) { + legacy_factory_->DeleteSettings(settings_namespace, model_type, extension_id); +} + +bool ValueStoreFactoryImpl::HasSettings(SettingsNamespace settings_namespace, + ModelType model_type, + const ExtensionId& extension_id) { + return legacy_factory_->HasSettings(settings_namespace, model_type, + extension_id); +} + +std::set<ExtensionId> ValueStoreFactoryImpl::GetKnownExtensionIDs( + SettingsNamespace settings_namespace, + ModelType model_type) const { + return legacy_factory_->GetKnownExtensionIDs(settings_namespace, model_type); +} + +} // namespace extensions diff --git a/extensions/browser/value_store/value_store_factory_impl.h b/extensions/browser/value_store/value_store_factory_impl.h new file mode 100644 index 0000000..d0a0677 --- /dev/null +++ b/extensions/browser/value_store/value_store_factory_impl.h @@ -0,0 +1,56 @@ +// Copyright 2016 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. + +#ifndef EXTENSIONS_BROWSER_VALUE_STORE_VALUE_STORE_FACTORY_IMPL_H_ +#define EXTENSIONS_BROWSER_VALUE_STORE_VALUE_STORE_FACTORY_IMPL_H_ + +#include <set> +#include <string> + +#include "base/files/file_path.h" +#include "base/memory/ref_counted.h" +#include "base/memory/scoped_ptr.h" +#include "extensions/browser/value_store/value_store.h" +#include "extensions/browser/value_store/value_store_factory.h" +#include "extensions/common/extension.h" + +namespace extensions { + +class LegacyValueStoreFactory; + +// Mint new |ValueStore| instances for use by the extensions system. These are +// used for extension rules, state, and settings. +class ValueStoreFactoryImpl : public ValueStoreFactory { + public: + explicit ValueStoreFactoryImpl(const base::FilePath& profile_path); + + // ValueStoreFactory + scoped_ptr<ValueStore> CreateRulesStore() override; + scoped_ptr<ValueStore> CreateStateStore() override; + scoped_ptr<ValueStore> CreateSettingsStore( + settings_namespace::Namespace settings_namespace, + ModelType model_type, + const ExtensionId& extension_id) override; + void DeleteSettings(settings_namespace::Namespace settings_namespace, + ModelType model_type, + const ExtensionId& extension_id) override; + bool HasSettings(settings_namespace::Namespace settings_namespace, + ModelType model_type, + const ExtensionId& extension_id) override; + std::set<ExtensionId> GetKnownExtensionIDs( + settings_namespace::Namespace settings_namespace, + ModelType model_type) const override; + + private: + // ValueStoreFactory is refcounted. + ~ValueStoreFactoryImpl() override; + + scoped_refptr<LegacyValueStoreFactory> legacy_factory_; + + DISALLOW_COPY_AND_ASSIGN(ValueStoreFactoryImpl); +}; + +} // namespace extensions + +#endif // EXTENSIONS_BROWSER_VALUE_STORE_VALUE_STORE_FACTORY_IMPL_H_ diff --git a/extensions/browser/value_store/value_store_frontend.cc b/extensions/browser/value_store/value_store_frontend.cc index 6f37bbc..b3912d6 100644 --- a/extensions/browser/value_store/value_store_frontend.cc +++ b/extensions/browser/value_store/value_store_frontend.cc @@ -13,31 +13,21 @@ #include "base/trace_event/trace_event.h" #include "content/public/browser/browser_thread.h" #include "extensions/browser/value_store/leveldb_value_store.h" +#include "extensions/browser/value_store/value_store_factory.h" using content::BrowserThread; +using extensions::ValueStoreFactory; class ValueStoreFrontend::Backend : public base::RefCountedThreadSafe<Backend> { public: - Backend() : storage_(NULL) {} - - void Init(const std::string& uma_client_name, const base::FilePath& db_path) { - DCHECK_CURRENTLY_ON(BrowserThread::FILE); - DCHECK(!storage_); - TRACE_EVENT0("ValueStoreFrontend::Backend", "Init"); - db_path_ = db_path; - storage_ = new LeveldbValueStore(uma_client_name, db_path); - } - - // This variant is useful for testing (using a mock ValueStore). - void InitWithStore(scoped_ptr<ValueStore> storage) { - DCHECK_CURRENTLY_ON(BrowserThread::FILE); - DCHECK(!storage_); - storage_ = storage.release(); - } + Backend(const scoped_refptr<ValueStoreFactory>& store_factory, + BackendType backend_type) + : store_factory_(store_factory), backend_type_(backend_type) {} void Get(const std::string& key, const ValueStoreFrontend::ReadCallback& callback) { DCHECK_CURRENTLY_ON(BrowserThread::FILE); + LazyInit(); ValueStore::ReadResult result = storage_->Get(key); // Extract the value from the ReadResult and pass ownership of it to the @@ -57,6 +47,7 @@ class ValueStoreFrontend::Backend : public base::RefCountedThreadSafe<Backend> { void Set(const std::string& key, scoped_ptr<base::Value> value) { DCHECK_CURRENTLY_ON(BrowserThread::FILE); + LazyInit(); // We don't need the old value, so skip generating changes. ValueStore::WriteResult result = storage_->Set( ValueStore::IGNORE_QUOTA | ValueStore::NO_GENERATE_CHANGES, @@ -68,6 +59,7 @@ class ValueStoreFrontend::Backend : public base::RefCountedThreadSafe<Backend> { void Remove(const std::string& key) { DCHECK_CURRENTLY_ON(BrowserThread::FILE); + LazyInit(); storage_->Remove(key); } @@ -75,10 +67,23 @@ class ValueStoreFrontend::Backend : public base::RefCountedThreadSafe<Backend> { friend class base::RefCountedThreadSafe<Backend>; virtual ~Backend() { - if (BrowserThread::CurrentlyOn(BrowserThread::FILE)) { - delete storage_; - } else { - BrowserThread::DeleteSoon(BrowserThread::FILE, FROM_HERE, storage_); + if (storage_ && !BrowserThread::CurrentlyOn(BrowserThread::FILE)) + BrowserThread::DeleteSoon(BrowserThread::FILE, FROM_HERE, + storage_.release()); + } + + void LazyInit() { + DCHECK_CURRENTLY_ON(BrowserThread::FILE); + if (storage_) + return; + TRACE_EVENT0("ValueStoreFrontend::Backend", "LazyInit"); + switch (backend_type_) { + case BackendType::RULES: + storage_ = store_factory_->CreateRulesStore(); + break; + case BackendType::STATE: + storage_ = store_factory_->CreateStateStore(); + break; } } @@ -88,43 +93,29 @@ class ValueStoreFrontend::Backend : public base::RefCountedThreadSafe<Backend> { callback.Run(std::move(value)); } + // The factory which will be used to lazily create the ValueStore when needed. + // Used exclusively on the FILE thread. + scoped_refptr<ValueStoreFactory> store_factory_; + BackendType backend_type_; + // The actual ValueStore that handles persisting the data to disk. Used // exclusively on the FILE thread. - ValueStore* storage_; + scoped_ptr<ValueStore> storage_; base::FilePath db_path_; DISALLOW_COPY_AND_ASSIGN(Backend); }; -ValueStoreFrontend::ValueStoreFrontend() - : backend_(new Backend()) { -} - -ValueStoreFrontend::ValueStoreFrontend(const std::string& uma_client_name, - const base::FilePath& db_path) - : backend_(new Backend()) { - Init(uma_client_name, db_path); -} - -ValueStoreFrontend::ValueStoreFrontend(scoped_ptr<ValueStore> value_store) - : backend_(new Backend()) { - BrowserThread::PostTask(BrowserThread::FILE, FROM_HERE, - base::Bind(&ValueStoreFrontend::Backend::InitWithStore, - backend_, base::Passed(&value_store))); -} +ValueStoreFrontend::ValueStoreFrontend( + const scoped_refptr<ValueStoreFactory>& store_factory, + BackendType backend_type) + : backend_(new Backend(store_factory, backend_type)) {} ValueStoreFrontend::~ValueStoreFrontend() { DCHECK(CalledOnValidThread()); } -void ValueStoreFrontend::Init(const std::string& uma_client_name, - const base::FilePath& db_path) { - BrowserThread::PostTask(BrowserThread::FILE, FROM_HERE, - base::Bind(&ValueStoreFrontend::Backend::Init, - backend_, uma_client_name, db_path)); -} - void ValueStoreFrontend::Get(const std::string& key, const ReadCallback& callback) { DCHECK(CalledOnValidThread()); diff --git a/extensions/browser/value_store/value_store_frontend.h b/extensions/browser/value_store/value_store_frontend.h index 972efad..55253fb 100644 --- a/extensions/browser/value_store/value_store_frontend.h +++ b/extensions/browser/value_store/value_store_frontend.h @@ -20,22 +20,25 @@ namespace base { class FilePath; } +namespace extensions { +class ValueStoreFactory; +} // namespace extensions + // A frontend for a LeveldbValueStore, for use on the UI thread. class ValueStoreFrontend : public base::SupportsWeakPtr<ValueStoreFrontend>, public base::NonThreadSafe { public: + // The kind of extensions data stored in a backend. + enum class BackendType { RULES, STATE }; + typedef base::Callback<void(scoped_ptr<base::Value>)> ReadCallback; - ValueStoreFrontend(); - ValueStoreFrontend(const std::string& uma_client_name, - const base::FilePath& db_path); - // This variant is useful for testing (using a mock ValueStore). - explicit ValueStoreFrontend(scoped_ptr<ValueStore> value_store); + ValueStoreFrontend( + const scoped_refptr<extensions::ValueStoreFactory>& store_factory, + BackendType backend_type); ~ValueStoreFrontend(); - void Init(const std::string& uma_client_name, const base::FilePath& db_path); - // Retrieves a value from the database asynchronously, passing a copy to // |callback| when ready. NULL is passed if no matching entry is found. void Get(const std::string& key, const ReadCallback& callback); diff --git a/extensions/browser/value_store/value_store_frontend_unittest.cc b/extensions/browser/value_store/value_store_frontend_unittest.cc index 0ee4c5c..21f95b6 100644 --- a/extensions/browser/value_store/value_store_frontend_unittest.cc +++ b/extensions/browser/value_store/value_store_frontend_unittest.cc @@ -12,13 +12,12 @@ #include "base/message_loop/message_loop.h" #include "base/path_service.h" #include "content/public/test/test_browser_thread.h" +#include "extensions/browser/value_store/test_value_store_factory.h" #include "extensions/common/extension_paths.h" #include "testing/gtest/include/gtest/gtest.h" using content::BrowserThread; -const char kDatabaseUMAClientName[] = "Test"; - class ValueStoreFrontendTest : public testing::Test { public: ValueStoreFrontendTest() @@ -35,6 +34,8 @@ class ValueStoreFrontendTest : public testing::Test { db_path_ = temp_dir_.path().AppendASCII("temp_db"); base::CopyDirectory(src_db, db_path_, true); + factory_ = new extensions::TestValueStoreFactory(db_path_); + ResetStorage(); } @@ -45,7 +46,8 @@ class ValueStoreFrontendTest : public testing::Test { // Reset the value store, reloading the DB from disk. void ResetStorage() { - storage_.reset(new ValueStoreFrontend(kDatabaseUMAClientName, db_path_)); + storage_.reset(new ValueStoreFrontend( + factory_, ValueStoreFrontend::BackendType::RULES)); } bool Get(const std::string& key, scoped_ptr<base::Value>* output) { @@ -62,6 +64,7 @@ class ValueStoreFrontendTest : public testing::Test { base::MessageLoop::current()->QuitWhenIdle(); } + scoped_refptr<extensions::TestValueStoreFactory> factory_; scoped_ptr<ValueStoreFrontend> storage_; base::ScopedTempDir temp_dir_; base::FilePath db_path_; diff --git a/extensions/extensions.gypi b/extensions/extensions.gypi index 7af2233..69963fa 100644 --- a/extensions/extensions.gypi +++ b/extensions/extensions.gypi @@ -432,14 +432,11 @@ 'browser/api/sockets_udp/sockets_udp_api.h', 'browser/api/sockets_udp/udp_socket_event_dispatcher.cc', 'browser/api/sockets_udp/udp_socket_event_dispatcher.h', - 'browser/api/storage/leveldb_settings_storage_factory.cc', - 'browser/api/storage/leveldb_settings_storage_factory.h', 'browser/api/storage/local_value_store_cache.cc', 'browser/api/storage/local_value_store_cache.h', 'browser/api/storage/settings_namespace.cc', 'browser/api/storage/settings_namespace.h', 'browser/api/storage/settings_observer.h', - 'browser/api/storage/settings_storage_factory.h', 'browser/api/storage/settings_storage_quota_enforcer.cc', 'browser/api/storage/settings_storage_quota_enforcer.h', 'browser/api/storage/storage_api.cc', @@ -790,14 +787,21 @@ 'browser/user_script_loader.h', 'browser/value_store/lazy_leveldb.cc', 'browser/value_store/lazy_leveldb.h', + 'browser/value_store/legacy_value_store_factory.cc', + 'browser/value_store/legacy_value_store_factory.h', 'browser/value_store/leveldb_value_store.cc', 'browser/value_store/leveldb_value_store.h', + 'browser/value_store/test_value_store_factory.cc', + 'browser/value_store/test_value_store_factory.h', 'browser/value_store/testing_value_store.cc', 'browser/value_store/testing_value_store.h', 'browser/value_store/value_store.cc', 'browser/value_store/value_store.h', 'browser/value_store/value_store_change.cc', 'browser/value_store/value_store_change.h', + 'browser/value_store/value_store_factory.h', + 'browser/value_store/value_store_factory_impl.cc', + 'browser/value_store/value_store_factory_impl.h', 'browser/value_store/value_store_frontend.cc', 'browser/value_store/value_store_frontend.h', 'browser/verified_contents.cc', diff --git a/extensions/extensions_tests.gypi b/extensions/extensions_tests.gypi index 5463453..845924d 100644 --- a/extensions/extensions_tests.gypi +++ b/extensions/extensions_tests.gypi @@ -94,11 +94,11 @@ 'browser/process_map_unittest.cc', 'browser/quota_service_unittest.cc', 'browser/runtime_data_unittest.cc', - 'browser/sandboxed_unpacker_unittest.cc', 'browser/extension_throttle_simulation_unittest.cc', 'browser/extension_throttle_test_support.cc', 'browser/extension_throttle_test_support.h', 'browser/extension_throttle_unittest.cc', + 'browser/sandboxed_unpacker_unittest.cc', 'browser/updater/update_service_unittest.cc', 'browser/value_store/leveldb_value_store_unittest.cc', 'browser/value_store/testing_value_store_unittest.cc', diff --git a/extensions/shell/browser/shell_extension_system.cc b/extensions/shell/browser/shell_extension_system.cc index 2ed9d7b..3fbbf14 100644 --- a/extensions/shell/browser/shell_extension_system.cc +++ b/extensions/shell/browser/shell_extension_system.cc @@ -22,6 +22,7 @@ #include "extensions/browser/quota_service.h" #include "extensions/browser/runtime_data.h" #include "extensions/browser/service_worker_manager.h" +#include "extensions/browser/value_store/value_store_factory_impl.h" #include "extensions/common/constants.h" #include "extensions/common/file_util.h" @@ -31,7 +32,9 @@ using content::BrowserThread; namespace extensions { ShellExtensionSystem::ShellExtensionSystem(BrowserContext* browser_context) - : browser_context_(browser_context), weak_factory_(this) {} + : browser_context_(browser_context), + store_factory_(new ValueStoreFactoryImpl(browser_context->GetPath())), + weak_factory_(this) {} ShellExtensionSystem::~ShellExtensionSystem() { } @@ -134,6 +137,10 @@ StateStore* ShellExtensionSystem::rules_store() { return nullptr; } +scoped_refptr<ValueStoreFactory> ShellExtensionSystem::store_factory() { + return store_factory_; +} + InfoMap* ShellExtensionSystem::info_map() { if (!info_map_.get()) info_map_ = new InfoMap; diff --git a/extensions/shell/browser/shell_extension_system.h b/extensions/shell/browser/shell_extension_system.h index b84c3fd..4448fb9 100644 --- a/extensions/shell/browser/shell_extension_system.h +++ b/extensions/shell/browser/shell_extension_system.h @@ -23,6 +23,8 @@ class BrowserContext; namespace extensions { +class ValueStoreFactory; + // A simplified version of ExtensionSystem for app_shell. Allows // app_shell to skip initialization of services it doesn't need. class ShellExtensionSystem : public ExtensionSystem { @@ -52,6 +54,7 @@ class ShellExtensionSystem : public ExtensionSystem { SharedUserScriptMaster* shared_user_script_master() override; StateStore* state_store() override; StateStore* rules_store() override; + scoped_refptr<ValueStoreFactory> store_factory() override; InfoMap* info_map() override; QuotaService* quota_service() override; AppSorting* app_sorting() override; @@ -81,6 +84,8 @@ class ShellExtensionSystem : public ExtensionSystem { scoped_ptr<QuotaService> quota_service_; scoped_ptr<AppSorting> app_sorting_; + scoped_refptr<ValueStoreFactory> store_factory_; + // Signaled when the extension system has completed its startup tasks. OneShotEvent ready_; |