1 // Copyright (c) 2012 The Chromium Authors. All rights reserved.
2 // Use of this source code is governed by a BSD-style license that can be
3 // found in the LICENSE file.
5 #include "chrome/browser/extensions/test_extension_prefs.h"
8 #include "base/bind_helpers.h"
9 #include "base/file_util.h"
10 #include "base/memory/scoped_ptr.h"
11 #include "base/message_loop.h"
12 #include "base/message_loop/message_loop_proxy.h"
13 #include "base/prefs/json_pref_store.h"
14 #include "base/prefs/pref_value_store.h"
15 #include "base/run_loop.h"
16 #include "base/sequenced_task_runner.h"
17 #include "base/synchronization/waitable_event.h"
18 #include "base/values.h"
19 #include "chrome/browser/extensions/extension_pref_store.h"
20 #include "chrome/browser/extensions/extension_pref_value_map.h"
21 #include "chrome/browser/extensions/extension_prefs.h"
22 #include "chrome/browser/prefs/pref_service_mock_builder.h"
23 #include "chrome/browser/prefs/pref_service_syncable.h"
24 #include "chrome/common/extensions/extension.h"
25 #include "chrome/common/extensions/extension_manifest_constants.h"
26 #include "components/user_prefs/pref_registry_syncable.h"
27 #include "content/public/browser/browser_thread.h"
28 #include "sync/api/string_ordinal.h"
29 #include "testing/gtest/include/gtest/gtest.h"
31 using content::BrowserThread
;
33 namespace extensions
{
37 // A TimeProvider which returns an incrementally later time each time
38 // GetCurrentTime is called.
39 class IncrementalTimeProvider
: public ExtensionPrefs::TimeProvider
{
41 IncrementalTimeProvider() : current_time_(base::Time::Now()) {
44 virtual ~IncrementalTimeProvider() {
47 virtual base::Time
GetCurrentTime() const OVERRIDE
{
48 current_time_
+= base::TimeDelta::FromSeconds(10);
53 DISALLOW_COPY_AND_ASSIGN(IncrementalTimeProvider
);
55 mutable base::Time current_time_
;
60 TestExtensionPrefs::TestExtensionPrefs(base::SequencedTaskRunner
* task_runner
)
61 : task_runner_(task_runner
), extensions_disabled_(false) {
62 EXPECT_TRUE(temp_dir_
.CreateUniqueTempDir());
63 preferences_file_
= temp_dir_
.path().AppendASCII("Preferences");
64 extensions_dir_
= temp_dir_
.path().AppendASCII("Extensions");
65 EXPECT_TRUE(file_util::CreateDirectory(extensions_dir_
));
68 RecreateExtensionPrefs();
71 TestExtensionPrefs::~TestExtensionPrefs() {
74 PrefService
* TestExtensionPrefs::pref_service() {
75 return pref_service_
.get();
78 const scoped_refptr
<user_prefs::PrefRegistrySyncable
>&
79 TestExtensionPrefs::pref_registry() {
80 return pref_registry_
;
83 void TestExtensionPrefs::ResetPrefRegistry() {
84 pref_registry_
= new user_prefs::PrefRegistrySyncable
;
85 ExtensionPrefs::RegisterUserPrefs(pref_registry_
.get());
88 void TestExtensionPrefs::RecreateExtensionPrefs() {
89 // We persist and reload the PrefService's PrefStores because this process
90 // deletes all empty dictionaries. The ExtensionPrefs implementation
91 // needs to be able to handle this situation.
93 // Commit a pending write (which posts a task to task_runner_) and wait for
95 pref_service_
->CommitPendingWrite();
96 base::RunLoop run_loop
;
98 task_runner_
->PostTaskAndReply(
100 base::Bind(&base::DoNothing
),
101 run_loop
.QuitClosure()));
105 extension_pref_value_map_
.reset(new ExtensionPrefValueMap
);
106 PrefServiceMockBuilder builder
;
107 builder
.WithUserFilePrefs(preferences_file_
, task_runner_
.get());
108 builder
.WithExtensionPrefs(
109 new ExtensionPrefStore(extension_pref_value_map_
.get(), false));
110 pref_service_
.reset(builder
.CreateSyncable(pref_registry_
.get()));
112 prefs_
.reset(ExtensionPrefs::Create(
115 extension_pref_value_map_
.get(),
116 extensions_disabled_
,
117 // Guarantee that no two extensions get the same installation time
118 // stamp and we can reliably assert the installation order in the tests.
119 scoped_ptr
<ExtensionPrefs::TimeProvider
>(
120 new IncrementalTimeProvider())));
123 scoped_refptr
<Extension
> TestExtensionPrefs::AddExtension(std::string name
) {
124 DictionaryValue dictionary
;
125 dictionary
.SetString(extension_manifest_keys::kName
, name
);
126 dictionary
.SetString(extension_manifest_keys::kVersion
, "0.1");
127 return AddExtensionWithManifest(dictionary
, Manifest::INTERNAL
);
130 scoped_refptr
<Extension
> TestExtensionPrefs::AddApp(std::string name
) {
131 DictionaryValue dictionary
;
132 dictionary
.SetString(extension_manifest_keys::kName
, name
);
133 dictionary
.SetString(extension_manifest_keys::kVersion
, "0.1");
134 dictionary
.SetString(extension_manifest_keys::kApp
, "true");
135 dictionary
.SetString(extension_manifest_keys::kLaunchWebURL
,
136 "http://example.com");
137 return AddExtensionWithManifest(dictionary
, Manifest::INTERNAL
);
141 scoped_refptr
<Extension
> TestExtensionPrefs::AddExtensionWithManifest(
142 const DictionaryValue
& manifest
, Manifest::Location location
) {
143 return AddExtensionWithManifestAndFlags(manifest
, location
,
144 Extension::NO_FLAGS
);
147 scoped_refptr
<Extension
> TestExtensionPrefs::AddExtensionWithManifestAndFlags(
148 const DictionaryValue
& manifest
,
149 Manifest::Location location
,
152 EXPECT_TRUE(manifest
.GetString(extension_manifest_keys::kName
, &name
));
153 base::FilePath path
= extensions_dir_
.AppendASCII(name
);
155 scoped_refptr
<Extension
> extension
= Extension::Create(
156 path
, location
, manifest
, extra_flags
, &errors
);
157 EXPECT_TRUE(extension
.get()) << errors
;
158 if (!extension
.get())
161 EXPECT_TRUE(Extension::IdIsValid(extension
->id()));
162 prefs_
->OnExtensionInstalled(extension
.get(),
164 syncer::StringOrdinal::CreateInitialOrdinal());
168 std::string
TestExtensionPrefs::AddExtensionAndReturnId(std::string name
) {
169 scoped_refptr
<Extension
> extension(AddExtension(name
));
170 return extension
->id();
173 PrefService
* TestExtensionPrefs::CreateIncognitoPrefService() const {
174 return pref_service_
->CreateIncognitoPrefService(
175 new ExtensionPrefStore(extension_pref_value_map_
.get(), true));
178 void TestExtensionPrefs::set_extensions_disabled(bool extensions_disabled
) {
179 extensions_disabled_
= extensions_disabled
;
182 } // namespace extensions