1 // Copyright (c) 2012 The Chromium Authors. All rights reserved.
2 // Use of this source code is governed by a BSD-style license that can be
3 // found in the LICENSE file.
4
5 #include "chrome/browser/chromeos/settings/cros_settings.h"
6
7 #include "base/bind.h"
8 #include "base/command_line.h"
9 #include "base/logging.h"
10 #include "base/stl_util.h"
11 #include "base/strings/string_util.h"
12 #include "base/values.h"
13 #include "chrome/browser/chromeos/settings/device_settings_provider.h"
14 #include "chrome/browser/chromeos/settings/device_settings_service.h"
15 #include "chrome/browser/chromeos/settings/stub_cros_settings_provider.h"
16 #include "chrome/browser/chromeos/settings/system_settings_provider.h"
17 #include "chromeos/chromeos_switches.h"
18 #include "google_apis/gaia/gaia_auth_util.h"
19
20 namespace chromeos {
21
22 static CrosSettings* g_cros_settings = NULL;
23
24 // static
Initialize()25 void CrosSettings::Initialize() {
26 CHECK(!g_cros_settings);
27 g_cros_settings = new CrosSettings(DeviceSettingsService::Get());
28 }
29
30 // static
IsInitialized()31 bool CrosSettings::IsInitialized() {
32 return g_cros_settings;
33 }
34
35 // static
Shutdown()36 void CrosSettings::Shutdown() {
37 DCHECK(g_cros_settings);
38 delete g_cros_settings;
39 g_cros_settings = NULL;
40 }
41
42 // static
Get()43 CrosSettings* CrosSettings::Get() {
44 CHECK(g_cros_settings);
45 return g_cros_settings;
46 }
47
CrosSettings(DeviceSettingsService * device_settings_service)48 CrosSettings::CrosSettings(DeviceSettingsService* device_settings_service) {
49 CrosSettingsProvider::NotifyObserversCallback notify_cb(
50 base::Bind(&CrosSettings::FireObservers,
51 // This is safe since |this| is never deleted.
52 base::Unretained(this)));
53 if (CommandLine::ForCurrentProcess()->HasSwitch(
54 switches::kStubCrosSettings)) {
55 AddSettingsProvider(new StubCrosSettingsProvider(notify_cb));
56 } else {
57 AddSettingsProvider(
58 new DeviceSettingsProvider(notify_cb, device_settings_service));
59 }
60 // System settings are not mocked currently.
61 AddSettingsProvider(new SystemSettingsProvider(notify_cb));
62 }
63
~CrosSettings()64 CrosSettings::~CrosSettings() {
65 STLDeleteElements(&providers_);
66 STLDeleteValues(&settings_observers_);
67 }
68
IsCrosSettings(const std::string & path)69 bool CrosSettings::IsCrosSettings(const std::string& path) {
70 return StartsWithASCII(path, kCrosSettingsPrefix, true);
71 }
72
Set(const std::string & path,const base::Value & in_value)73 void CrosSettings::Set(const std::string& path, const base::Value& in_value) {
74 DCHECK(CalledOnValidThread());
75 CrosSettingsProvider* provider;
76 provider = GetProvider(path);
77 if (provider)
78 provider->Set(path, in_value);
79 }
80
GetPref(const std::string & path) const81 const base::Value* CrosSettings::GetPref(const std::string& path) const {
82 DCHECK(CalledOnValidThread());
83 CrosSettingsProvider* provider = GetProvider(path);
84 if (provider)
85 return provider->Get(path);
86 NOTREACHED() << path << " preference was not found in the signed settings.";
87 return NULL;
88 }
89
PrepareTrustedValues(const base::Closure & callback) const90 CrosSettingsProvider::TrustedStatus CrosSettings::PrepareTrustedValues(
91 const base::Closure& callback) const {
92 DCHECK(CalledOnValidThread());
93 for (size_t i = 0; i < providers_.size(); ++i) {
94 CrosSettingsProvider::TrustedStatus status =
95 providers_[i]->PrepareTrustedValues(callback);
96 if (status != CrosSettingsProvider::TRUSTED)
97 return status;
98 }
99 return CrosSettingsProvider::TRUSTED;
100 }
101
SetBoolean(const std::string & path,bool in_value)102 void CrosSettings::SetBoolean(const std::string& path, bool in_value) {
103 DCHECK(CalledOnValidThread());
104 base::FundamentalValue value(in_value);
105 Set(path, value);
106 }
107
SetInteger(const std::string & path,int in_value)108 void CrosSettings::SetInteger(const std::string& path, int in_value) {
109 DCHECK(CalledOnValidThread());
110 base::FundamentalValue value(in_value);
111 Set(path, value);
112 }
113
SetDouble(const std::string & path,double in_value)114 void CrosSettings::SetDouble(const std::string& path, double in_value) {
115 DCHECK(CalledOnValidThread());
116 base::FundamentalValue value(in_value);
117 Set(path, value);
118 }
119
SetString(const std::string & path,const std::string & in_value)120 void CrosSettings::SetString(const std::string& path,
121 const std::string& in_value) {
122 DCHECK(CalledOnValidThread());
123 base::StringValue value(in_value);
124 Set(path, value);
125 }
126
AppendToList(const std::string & path,const base::Value * value)127 void CrosSettings::AppendToList(const std::string& path,
128 const base::Value* value) {
129 DCHECK(CalledOnValidThread());
130 const base::Value* old_value = GetPref(path);
131 scoped_ptr<base::Value> new_value(
132 old_value ? old_value->DeepCopy() : new base::ListValue());
133 static_cast<base::ListValue*>(new_value.get())->Append(value->DeepCopy());
134 Set(path, *new_value);
135 }
136
RemoveFromList(const std::string & path,const base::Value * value)137 void CrosSettings::RemoveFromList(const std::string& path,
138 const base::Value* value) {
139 DCHECK(CalledOnValidThread());
140 const base::Value* old_value = GetPref(path);
141 scoped_ptr<base::Value> new_value(
142 old_value ? old_value->DeepCopy() : new base::ListValue());
143 static_cast<base::ListValue*>(new_value.get())->Remove(*value, NULL);
144 Set(path, *new_value);
145 }
146
GetBoolean(const std::string & path,bool * bool_value) const147 bool CrosSettings::GetBoolean(const std::string& path,
148 bool* bool_value) const {
149 DCHECK(CalledOnValidThread());
150 const base::Value* value = GetPref(path);
151 if (value)
152 return value->GetAsBoolean(bool_value);
153 return false;
154 }
155
GetInteger(const std::string & path,int * out_value) const156 bool CrosSettings::GetInteger(const std::string& path,
157 int* out_value) const {
158 DCHECK(CalledOnValidThread());
159 const base::Value* value = GetPref(path);
160 if (value)
161 return value->GetAsInteger(out_value);
162 return false;
163 }
164
GetDouble(const std::string & path,double * out_value) const165 bool CrosSettings::GetDouble(const std::string& path,
166 double* out_value) const {
167 DCHECK(CalledOnValidThread());
168 const base::Value* value = GetPref(path);
169 if (value)
170 return value->GetAsDouble(out_value);
171 return false;
172 }
173
GetString(const std::string & path,std::string * out_value) const174 bool CrosSettings::GetString(const std::string& path,
175 std::string* out_value) const {
176 DCHECK(CalledOnValidThread());
177 const base::Value* value = GetPref(path);
178 if (value)
179 return value->GetAsString(out_value);
180 return false;
181 }
182
GetList(const std::string & path,const base::ListValue ** out_value) const183 bool CrosSettings::GetList(const std::string& path,
184 const base::ListValue** out_value) const {
185 DCHECK(CalledOnValidThread());
186 const base::Value* value = GetPref(path);
187 if (value)
188 return value->GetAsList(out_value);
189 return false;
190 }
191
GetDictionary(const std::string & path,const base::DictionaryValue ** out_value) const192 bool CrosSettings::GetDictionary(
193 const std::string& path,
194 const base::DictionaryValue** out_value) const {
195 DCHECK(CalledOnValidThread());
196 const base::Value* value = GetPref(path);
197 if (value)
198 return value->GetAsDictionary(out_value);
199 return false;
200 }
201
FindEmailInList(const std::string & path,const std::string & email,bool * wildcard_match) const202 bool CrosSettings::FindEmailInList(const std::string& path,
203 const std::string& email,
204 bool* wildcard_match) const {
205 DCHECK(CalledOnValidThread());
206 std::string canonicalized_email(
207 gaia::CanonicalizeEmail(gaia::SanitizeEmail(email)));
208 std::string wildcard_email;
209 std::string::size_type at_pos = canonicalized_email.find('@');
210 if (at_pos != std::string::npos) {
211 wildcard_email =
212 std::string("*").append(canonicalized_email.substr(at_pos));
213 }
214
215 if (wildcard_match)
216 *wildcard_match = false;
217
218 const base::ListValue* list;
219 if (!GetList(path, &list))
220 return false;
221
222 bool found_wildcard_match = false;
223 for (base::ListValue::const_iterator entry(list->begin());
224 entry != list->end();
225 ++entry) {
226 std::string entry_string;
227 if (!(*entry)->GetAsString(&entry_string)) {
228 NOTREACHED();
229 continue;
230 }
231 std::string canonicalized_entry(
232 gaia::CanonicalizeEmail(gaia::SanitizeEmail(entry_string)));
233
234 if (canonicalized_entry != wildcard_email &&
235 canonicalized_entry == canonicalized_email) {
236 return true;
237 }
238
239 // If there is a wildcard match, don't exit early. There might be an exact
240 // match further down the list that should take precedence if present.
241 if (canonicalized_entry == wildcard_email)
242 found_wildcard_match = true;
243 }
244
245 if (wildcard_match)
246 *wildcard_match = found_wildcard_match;
247
248 return found_wildcard_match;
249 }
250
AddSettingsProvider(CrosSettingsProvider * provider)251 bool CrosSettings::AddSettingsProvider(CrosSettingsProvider* provider) {
252 DCHECK(CalledOnValidThread());
253 providers_.push_back(provider);
254
255 // Allow the provider to notify this object when settings have changed.
256 // Providers instantiated inside this class will have the same callback
257 // passed to their constructor, but doing it here allows for providers
258 // to be instantiated outside this class.
259 CrosSettingsProvider::NotifyObserversCallback notify_cb(
260 base::Bind(&CrosSettings::FireObservers, base::Unretained(this)));
261 provider->SetNotifyObserversCallback(notify_cb);
262 return true;
263 }
264
RemoveSettingsProvider(CrosSettingsProvider * provider)265 bool CrosSettings::RemoveSettingsProvider(CrosSettingsProvider* provider) {
266 DCHECK(CalledOnValidThread());
267 std::vector<CrosSettingsProvider*>::iterator it =
268 std::find(providers_.begin(), providers_.end(), provider);
269 if (it != providers_.end()) {
270 providers_.erase(it);
271 return true;
272 }
273 return false;
274 }
275
276 scoped_ptr<CrosSettings::ObserverSubscription>
AddSettingsObserver(const std::string & path,const base::Closure & callback)277 CrosSettings::AddSettingsObserver(const std::string& path,
278 const base::Closure& callback) {
279 DCHECK(!path.empty());
280 DCHECK(!callback.is_null());
281 DCHECK(CalledOnValidThread());
282
283 if (!GetProvider(path)) {
284 NOTREACHED() << "Trying to add an observer for an unregistered setting: "
285 << path;
286 return scoped_ptr<CrosSettings::ObserverSubscription>();
287 }
288
289 // Get the callback registry associated with the path.
290 base::CallbackList<void(void)>* registry = NULL;
291 SettingsObserverMap::iterator observer_iterator =
292 settings_observers_.find(path);
293 if (observer_iterator == settings_observers_.end()) {
294 registry = new base::CallbackList<void(void)>;
295 settings_observers_[path] = registry;
296 } else {
297 registry = observer_iterator->second;
298 }
299
300 return registry->Add(callback);
301 }
302
GetProvider(const std::string & path) const303 CrosSettingsProvider* CrosSettings::GetProvider(
304 const std::string& path) const {
305 for (size_t i = 0; i < providers_.size(); ++i) {
306 if (providers_[i]->HandlesSetting(path))
307 return providers_[i];
308 }
309 return NULL;
310 }
311
FireObservers(const std::string & path)312 void CrosSettings::FireObservers(const std::string& path) {
313 DCHECK(CalledOnValidThread());
314 SettingsObserverMap::iterator observer_iterator =
315 settings_observers_.find(path);
316 if (observer_iterator == settings_observers_.end())
317 return;
318
319 observer_iterator->second->Notify();
320 }
321
ScopedTestCrosSettings()322 ScopedTestCrosSettings::ScopedTestCrosSettings() {
323 CrosSettings::Initialize();
324 }
325
~ScopedTestCrosSettings()326 ScopedTestCrosSettings::~ScopedTestCrosSettings() {
327 CrosSettings::Shutdown();
328 }
329
330 } // namespace chromeos
331