• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
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