// Copyright (c) 2012 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 "chrome/browser/chromeos/settings/cros_settings.h" #include "base/bind.h" #include "base/command_line.h" #include "base/logging.h" #include "base/stl_util.h" #include "base/strings/string_util.h" #include "base/values.h" #include "chrome/browser/chromeos/settings/device_settings_provider.h" #include "chrome/browser/chromeos/settings/device_settings_service.h" #include "chrome/browser/chromeos/settings/stub_cros_settings_provider.h" #include "chrome/browser/chromeos/settings/system_settings_provider.h" #include "chromeos/chromeos_switches.h" #include "google_apis/gaia/gaia_auth_util.h" namespace chromeos { static CrosSettings* g_cros_settings = NULL; // static void CrosSettings::Initialize() { CHECK(!g_cros_settings); g_cros_settings = new CrosSettings(DeviceSettingsService::Get()); } // static bool CrosSettings::IsInitialized() { return g_cros_settings; } // static void CrosSettings::Shutdown() { DCHECK(g_cros_settings); delete g_cros_settings; g_cros_settings = NULL; } // static CrosSettings* CrosSettings::Get() { CHECK(g_cros_settings); return g_cros_settings; } CrosSettings::CrosSettings(DeviceSettingsService* device_settings_service) { CrosSettingsProvider::NotifyObserversCallback notify_cb( base::Bind(&CrosSettings::FireObservers, // This is safe since |this| is never deleted. base::Unretained(this))); if (CommandLine::ForCurrentProcess()->HasSwitch( switches::kStubCrosSettings)) { AddSettingsProvider(new StubCrosSettingsProvider(notify_cb)); } else { AddSettingsProvider( new DeviceSettingsProvider(notify_cb, device_settings_service)); } // System settings are not mocked currently. AddSettingsProvider(new SystemSettingsProvider(notify_cb)); } CrosSettings::~CrosSettings() { STLDeleteElements(&providers_); STLDeleteValues(&settings_observers_); } bool CrosSettings::IsCrosSettings(const std::string& path) { return StartsWithASCII(path, kCrosSettingsPrefix, true); } void CrosSettings::Set(const std::string& path, const base::Value& in_value) { DCHECK(CalledOnValidThread()); CrosSettingsProvider* provider; provider = GetProvider(path); if (provider) provider->Set(path, in_value); } const base::Value* CrosSettings::GetPref(const std::string& path) const { DCHECK(CalledOnValidThread()); CrosSettingsProvider* provider = GetProvider(path); if (provider) return provider->Get(path); NOTREACHED() << path << " preference was not found in the signed settings."; return NULL; } CrosSettingsProvider::TrustedStatus CrosSettings::PrepareTrustedValues( const base::Closure& callback) const { DCHECK(CalledOnValidThread()); for (size_t i = 0; i < providers_.size(); ++i) { CrosSettingsProvider::TrustedStatus status = providers_[i]->PrepareTrustedValues(callback); if (status != CrosSettingsProvider::TRUSTED) return status; } return CrosSettingsProvider::TRUSTED; } void CrosSettings::SetBoolean(const std::string& path, bool in_value) { DCHECK(CalledOnValidThread()); base::FundamentalValue value(in_value); Set(path, value); } void CrosSettings::SetInteger(const std::string& path, int in_value) { DCHECK(CalledOnValidThread()); base::FundamentalValue value(in_value); Set(path, value); } void CrosSettings::SetDouble(const std::string& path, double in_value) { DCHECK(CalledOnValidThread()); base::FundamentalValue value(in_value); Set(path, value); } void CrosSettings::SetString(const std::string& path, const std::string& in_value) { DCHECK(CalledOnValidThread()); base::StringValue value(in_value); Set(path, value); } void CrosSettings::AppendToList(const std::string& path, const base::Value* value) { DCHECK(CalledOnValidThread()); const base::Value* old_value = GetPref(path); scoped_ptr new_value( old_value ? old_value->DeepCopy() : new base::ListValue()); static_cast(new_value.get())->Append(value->DeepCopy()); Set(path, *new_value); } void CrosSettings::RemoveFromList(const std::string& path, const base::Value* value) { DCHECK(CalledOnValidThread()); const base::Value* old_value = GetPref(path); scoped_ptr new_value( old_value ? old_value->DeepCopy() : new base::ListValue()); static_cast(new_value.get())->Remove(*value, NULL); Set(path, *new_value); } bool CrosSettings::GetBoolean(const std::string& path, bool* bool_value) const { DCHECK(CalledOnValidThread()); const base::Value* value = GetPref(path); if (value) return value->GetAsBoolean(bool_value); return false; } bool CrosSettings::GetInteger(const std::string& path, int* out_value) const { DCHECK(CalledOnValidThread()); const base::Value* value = GetPref(path); if (value) return value->GetAsInteger(out_value); return false; } bool CrosSettings::GetDouble(const std::string& path, double* out_value) const { DCHECK(CalledOnValidThread()); const base::Value* value = GetPref(path); if (value) return value->GetAsDouble(out_value); return false; } bool CrosSettings::GetString(const std::string& path, std::string* out_value) const { DCHECK(CalledOnValidThread()); const base::Value* value = GetPref(path); if (value) return value->GetAsString(out_value); return false; } bool CrosSettings::GetList(const std::string& path, const base::ListValue** out_value) const { DCHECK(CalledOnValidThread()); const base::Value* value = GetPref(path); if (value) return value->GetAsList(out_value); return false; } bool CrosSettings::GetDictionary( const std::string& path, const base::DictionaryValue** out_value) const { DCHECK(CalledOnValidThread()); const base::Value* value = GetPref(path); if (value) return value->GetAsDictionary(out_value); return false; } bool CrosSettings::FindEmailInList(const std::string& path, const std::string& email, bool* wildcard_match) const { DCHECK(CalledOnValidThread()); std::string canonicalized_email( gaia::CanonicalizeEmail(gaia::SanitizeEmail(email))); std::string wildcard_email; std::string::size_type at_pos = canonicalized_email.find('@'); if (at_pos != std::string::npos) { wildcard_email = std::string("*").append(canonicalized_email.substr(at_pos)); } if (wildcard_match) *wildcard_match = false; const base::ListValue* list; if (!GetList(path, &list)) return false; bool found_wildcard_match = false; for (base::ListValue::const_iterator entry(list->begin()); entry != list->end(); ++entry) { std::string entry_string; if (!(*entry)->GetAsString(&entry_string)) { NOTREACHED(); continue; } std::string canonicalized_entry( gaia::CanonicalizeEmail(gaia::SanitizeEmail(entry_string))); if (canonicalized_entry != wildcard_email && canonicalized_entry == canonicalized_email) { return true; } // If there is a wildcard match, don't exit early. There might be an exact // match further down the list that should take precedence if present. if (canonicalized_entry == wildcard_email) found_wildcard_match = true; } if (wildcard_match) *wildcard_match = found_wildcard_match; return found_wildcard_match; } bool CrosSettings::AddSettingsProvider(CrosSettingsProvider* provider) { DCHECK(CalledOnValidThread()); providers_.push_back(provider); // Allow the provider to notify this object when settings have changed. // Providers instantiated inside this class will have the same callback // passed to their constructor, but doing it here allows for providers // to be instantiated outside this class. CrosSettingsProvider::NotifyObserversCallback notify_cb( base::Bind(&CrosSettings::FireObservers, base::Unretained(this))); provider->SetNotifyObserversCallback(notify_cb); return true; } bool CrosSettings::RemoveSettingsProvider(CrosSettingsProvider* provider) { DCHECK(CalledOnValidThread()); std::vector::iterator it = std::find(providers_.begin(), providers_.end(), provider); if (it != providers_.end()) { providers_.erase(it); return true; } return false; } scoped_ptr CrosSettings::AddSettingsObserver(const std::string& path, const base::Closure& callback) { DCHECK(!path.empty()); DCHECK(!callback.is_null()); DCHECK(CalledOnValidThread()); if (!GetProvider(path)) { NOTREACHED() << "Trying to add an observer for an unregistered setting: " << path; return scoped_ptr(); } // Get the callback registry associated with the path. base::CallbackList* registry = NULL; SettingsObserverMap::iterator observer_iterator = settings_observers_.find(path); if (observer_iterator == settings_observers_.end()) { registry = new base::CallbackList; settings_observers_[path] = registry; } else { registry = observer_iterator->second; } return registry->Add(callback); } CrosSettingsProvider* CrosSettings::GetProvider( const std::string& path) const { for (size_t i = 0; i < providers_.size(); ++i) { if (providers_[i]->HandlesSetting(path)) return providers_[i]; } return NULL; } void CrosSettings::FireObservers(const std::string& path) { DCHECK(CalledOnValidThread()); SettingsObserverMap::iterator observer_iterator = settings_observers_.find(path); if (observer_iterator == settings_observers_.end()) return; observer_iterator->second->Notify(); } ScopedTestCrosSettings::ScopedTestCrosSettings() { CrosSettings::Initialize(); } ScopedTestCrosSettings::~ScopedTestCrosSettings() { CrosSettings::Shutdown(); } } // namespace chromeos