| // | 
 | // Copyright (C) 2012 The Android Open Source Project | 
 | // | 
 | // Licensed under the Apache License, Version 2.0 (the "License"); | 
 | // you may not use this file except in compliance with the License. | 
 | // You may obtain a copy of the License at | 
 | // | 
 | //      http://www.apache.org/licenses/LICENSE-2.0 | 
 | // | 
 | // Unless required by applicable law or agreed to in writing, software | 
 | // distributed under the License is distributed on an "AS IS" BASIS, | 
 | // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. | 
 | // See the License for the specific language governing permissions and | 
 | // limitations under the License. | 
 | // | 
 |  | 
 | #include "update_engine/common/prefs.h" | 
 |  | 
 | #include <algorithm> | 
 |  | 
 | #include <base/files/file_util.h> | 
 | #include <base/logging.h> | 
 | #include <base/strings/string_number_conversions.h> | 
 | #include <base/strings/string_util.h> | 
 |  | 
 | #include "update_engine/common/utils.h" | 
 |  | 
 | using std::string; | 
 |  | 
 | namespace chromeos_update_engine { | 
 |  | 
 | bool PrefsBase::GetString(const string& key, string* value) const { | 
 |   return storage_->GetKey(key, value); | 
 | } | 
 |  | 
 | bool PrefsBase::SetString(const string& key, const string& value) { | 
 |   TEST_AND_RETURN_FALSE(storage_->SetKey(key, value)); | 
 |   const auto observers_for_key = observers_.find(key); | 
 |   if (observers_for_key != observers_.end()) { | 
 |     std::vector<ObserverInterface*> copy_observers(observers_for_key->second); | 
 |     for (ObserverInterface* observer : copy_observers) | 
 |       observer->OnPrefSet(key); | 
 |   } | 
 |   return true; | 
 | } | 
 |  | 
 | bool PrefsBase::GetInt64(const string& key, int64_t* value) const { | 
 |   string str_value; | 
 |   if (!GetString(key, &str_value)) | 
 |     return false; | 
 |   base::TrimWhitespaceASCII(str_value, base::TRIM_ALL, &str_value); | 
 |   TEST_AND_RETURN_FALSE(base::StringToInt64(str_value, value)); | 
 |   return true; | 
 | } | 
 |  | 
 | bool PrefsBase::SetInt64(const string& key, const int64_t value) { | 
 |   return SetString(key, base::Int64ToString(value)); | 
 | } | 
 |  | 
 | bool PrefsBase::GetBoolean(const string& key, bool* value) const { | 
 |   string str_value; | 
 |   if (!GetString(key, &str_value)) | 
 |     return false; | 
 |   base::TrimWhitespaceASCII(str_value, base::TRIM_ALL, &str_value); | 
 |   if (str_value == "false") { | 
 |     *value = false; | 
 |     return true; | 
 |   } | 
 |   if (str_value == "true") { | 
 |     *value = true; | 
 |     return true; | 
 |   } | 
 |   return false; | 
 | } | 
 |  | 
 | bool PrefsBase::SetBoolean(const string& key, const bool value) { | 
 |   return SetString(key, value ? "true" : "false"); | 
 | } | 
 |  | 
 | bool PrefsBase::Exists(const string& key) const { | 
 |   return storage_->KeyExists(key); | 
 | } | 
 |  | 
 | bool PrefsBase::Delete(const string& key) { | 
 |   TEST_AND_RETURN_FALSE(storage_->DeleteKey(key)); | 
 |   const auto observers_for_key = observers_.find(key); | 
 |   if (observers_for_key != observers_.end()) { | 
 |     std::vector<ObserverInterface*> copy_observers(observers_for_key->second); | 
 |     for (ObserverInterface* observer : copy_observers) | 
 |       observer->OnPrefDeleted(key); | 
 |   } | 
 |   return true; | 
 | } | 
 |  | 
 | void PrefsBase::AddObserver(const string& key, ObserverInterface* observer) { | 
 |   observers_[key].push_back(observer); | 
 | } | 
 |  | 
 | void PrefsBase::RemoveObserver(const string& key, ObserverInterface* observer) { | 
 |   std::vector<ObserverInterface*>& observers_for_key = observers_[key]; | 
 |   auto observer_it = | 
 |       std::find(observers_for_key.begin(), observers_for_key.end(), observer); | 
 |   if (observer_it != observers_for_key.end()) | 
 |     observers_for_key.erase(observer_it); | 
 | } | 
 |  | 
 | // Prefs | 
 |  | 
 | bool Prefs::Init(const base::FilePath& prefs_dir) { | 
 |   return file_storage_.Init(prefs_dir); | 
 | } | 
 |  | 
 | bool Prefs::FileStorage::Init(const base::FilePath& prefs_dir) { | 
 |   prefs_dir_ = prefs_dir; | 
 |   return true; | 
 | } | 
 |  | 
 | bool Prefs::FileStorage::GetKey(const string& key, string* value) const { | 
 |   base::FilePath filename; | 
 |   TEST_AND_RETURN_FALSE(GetFileNameForKey(key, &filename)); | 
 |   if (!base::ReadFileToString(filename, value)) { | 
 |     LOG(INFO) << key << " not present in " << prefs_dir_.value(); | 
 |     return false; | 
 |   } | 
 |   return true; | 
 | } | 
 |  | 
 | bool Prefs::FileStorage::SetKey(const string& key, const string& value) { | 
 |   base::FilePath filename; | 
 |   TEST_AND_RETURN_FALSE(GetFileNameForKey(key, &filename)); | 
 |   if (!base::DirectoryExists(filename.DirName())) { | 
 |     // Only attempt to create the directory if it doesn't exist to avoid calls | 
 |     // to parent directories where we might not have permission to write to. | 
 |     TEST_AND_RETURN_FALSE(base::CreateDirectory(filename.DirName())); | 
 |   } | 
 |   TEST_AND_RETURN_FALSE(base::WriteFile(filename, value.data(), value.size()) == | 
 |                         static_cast<int>(value.size())); | 
 |   return true; | 
 | } | 
 |  | 
 | bool Prefs::FileStorage::KeyExists(const string& key) const { | 
 |   base::FilePath filename; | 
 |   TEST_AND_RETURN_FALSE(GetFileNameForKey(key, &filename)); | 
 |   return base::PathExists(filename); | 
 | } | 
 |  | 
 | bool Prefs::FileStorage::DeleteKey(const string& key) { | 
 |   base::FilePath filename; | 
 |   TEST_AND_RETURN_FALSE(GetFileNameForKey(key, &filename)); | 
 |   TEST_AND_RETURN_FALSE(base::DeleteFile(filename, false)); | 
 |   return true; | 
 | } | 
 |  | 
 | bool Prefs::FileStorage::GetFileNameForKey(const string& key, | 
 |                                            base::FilePath* filename) const { | 
 |   // Allows only non-empty keys containing [A-Za-z0-9_-]. | 
 |   TEST_AND_RETURN_FALSE(!key.empty()); | 
 |   for (size_t i = 0; i < key.size(); ++i) { | 
 |     char c = key.at(i); | 
 |     TEST_AND_RETURN_FALSE(base::IsAsciiAlpha(c) || base::IsAsciiDigit(c) || | 
 |                           c == '_' || c == '-'); | 
 |   } | 
 |   *filename = prefs_dir_.Append(key); | 
 |   return true; | 
 | } | 
 |  | 
 | // MemoryPrefs | 
 |  | 
 | bool MemoryPrefs::MemoryStorage::GetKey(const string& key, | 
 |                                         string* value) const { | 
 |   auto it = values_.find(key); | 
 |   if (it == values_.end()) | 
 |     return false; | 
 |   *value = it->second; | 
 |   return true; | 
 | } | 
 |  | 
 | bool MemoryPrefs::MemoryStorage::SetKey(const string& key, | 
 |                                         const string& value) { | 
 |   values_[key] = value; | 
 |   return true; | 
 | } | 
 |  | 
 | bool MemoryPrefs::MemoryStorage::KeyExists(const string& key) const { | 
 |   return values_.find(key) != values_.end(); | 
 | } | 
 |  | 
 | bool MemoryPrefs::MemoryStorage::DeleteKey(const string& key) { | 
 |   auto it = values_.find(key); | 
 |   if (it == values_.end()) | 
 |     return false; | 
 |   values_.erase(it); | 
 |   return true; | 
 | } | 
 |  | 
 | }  // namespace chromeos_update_engine |