|  | // | 
|  | // 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_enumerator.h> | 
|  | #include <base/files/file_util.h> | 
|  | #include <base/logging.h> | 
|  | #include <base/strings/string_number_conversions.h> | 
|  | #include <base/strings/string_split.h> | 
|  | #include <base/strings/string_util.h> | 
|  |  | 
|  | #include "update_engine/common/utils.h" | 
|  |  | 
|  | using std::string; | 
|  | using std::vector; | 
|  |  | 
|  | namespace chromeos_update_engine { | 
|  |  | 
|  | namespace { | 
|  |  | 
|  | void DeleteEmptyDirectories(const base::FilePath& path) { | 
|  | base::FileEnumerator path_enum( | 
|  | path, false /* recursive */, base::FileEnumerator::DIRECTORIES); | 
|  | for (base::FilePath dir_path = path_enum.Next(); !dir_path.empty(); | 
|  | dir_path = path_enum.Next()) { | 
|  | DeleteEmptyDirectories(dir_path); | 
|  | if (base::IsDirectoryEmpty(dir_path)) | 
|  | #if BASE_VER < 800000 | 
|  | base::DeleteFile(dir_path, false); | 
|  | #else | 
|  | base::DeleteFile(dir_path); | 
|  | #endif | 
|  | } | 
|  | } | 
|  |  | 
|  | }  // namespace | 
|  |  | 
|  | bool PrefsBase::GetString(const std::string_view key, string* value) const { | 
|  | return storage_->GetKey(key, value); | 
|  | } | 
|  |  | 
|  | bool PrefsBase::SetString(std::string_view key, std::string_view 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 std::string_view 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(std::string_view key, const int64_t value) { | 
|  | return SetString(key, base::NumberToString(value)); | 
|  | } | 
|  |  | 
|  | bool PrefsBase::GetBoolean(std::string_view 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(std::string_view key, const bool value) { | 
|  | return SetString(key, value ? "true" : "false"); | 
|  | } | 
|  |  | 
|  | bool PrefsBase::Exists(std::string_view key) const { | 
|  | return storage_->KeyExists(key); | 
|  | } | 
|  |  | 
|  | bool PrefsBase::Delete(std::string_view 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; | 
|  | } | 
|  |  | 
|  | bool PrefsBase::Delete(std::string_view pref_key, const vector<string>& nss) { | 
|  | // Delete pref key for platform. | 
|  | bool success = Delete(pref_key); | 
|  | // Delete pref key in each namespace. | 
|  | for (const auto& ns : nss) { | 
|  | vector<string> namespace_keys; | 
|  | success = GetSubKeys(ns, &namespace_keys) && success; | 
|  | for (const auto& key : namespace_keys) { | 
|  | auto last_key_seperator = key.find_last_of(kKeySeparator); | 
|  | if (last_key_seperator != string::npos && | 
|  | pref_key == key.substr(last_key_seperator + 1)) { | 
|  | success = Delete(key) && success; | 
|  | } | 
|  | } | 
|  | } | 
|  | return success; | 
|  | } | 
|  |  | 
|  | bool PrefsBase::GetSubKeys(std::string_view ns, vector<string>* keys) const { | 
|  | return storage_->GetSubKeys(ns, keys); | 
|  | } | 
|  |  | 
|  | void PrefsBase::AddObserver(std::string_view key, ObserverInterface* observer) { | 
|  | observers_[std::string{key}].push_back(observer); | 
|  | } | 
|  |  | 
|  | void PrefsBase::RemoveObserver(std::string_view key, | 
|  | ObserverInterface* observer) { | 
|  | std::vector<ObserverInterface*>& observers_for_key = | 
|  | observers_[std::string{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); | 
|  | } | 
|  |  | 
|  | string PrefsInterface::CreateSubKey(const vector<string>& ns_and_key) { | 
|  | return base::JoinString(ns_and_key, string(1, kKeySeparator)); | 
|  | } | 
|  |  | 
|  | // 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; | 
|  | // Delete empty directories. Ignore errors when deleting empty directories. | 
|  | DeleteEmptyDirectories(prefs_dir_); | 
|  | return true; | 
|  | } | 
|  |  | 
|  | bool Prefs::FileStorage::GetKey(std::string_view key, string* value) const { | 
|  | base::FilePath filename; | 
|  | TEST_AND_RETURN_FALSE(GetFileNameForKey(key, &filename)); | 
|  | if (!base::ReadFileToString(filename, value)) { | 
|  | return false; | 
|  | } | 
|  | return true; | 
|  | } | 
|  |  | 
|  | bool Prefs::FileStorage::GetSubKeys(std::string_view ns, | 
|  | vector<string>* keys) const { | 
|  | base::FilePath filename; | 
|  | TEST_AND_RETURN_FALSE(GetFileNameForKey(ns, &filename)); | 
|  | base::FileEnumerator namespace_enum( | 
|  | prefs_dir_, true, base::FileEnumerator::FILES); | 
|  | for (base::FilePath f = namespace_enum.Next(); !f.empty(); | 
|  | f = namespace_enum.Next()) { | 
|  | auto filename_str = filename.value(); | 
|  | if (f.value().compare(0, filename_str.length(), filename_str) == 0) { | 
|  | // Only return the key portion excluding the |prefs_dir_| with slash. | 
|  | keys->push_back(f.value().substr( | 
|  | prefs_dir_.AsEndingWithSeparator().value().length())); | 
|  | } | 
|  | } | 
|  | return true; | 
|  | } | 
|  |  | 
|  | bool Prefs::FileStorage::SetKey(std::string_view key, std::string_view 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(std::string_view key) const { | 
|  | base::FilePath filename; | 
|  | TEST_AND_RETURN_FALSE(GetFileNameForKey(key, &filename)); | 
|  | return base::PathExists(filename); | 
|  | } | 
|  |  | 
|  | bool Prefs::FileStorage::DeleteKey(std::string_view key) { | 
|  | base::FilePath filename; | 
|  | TEST_AND_RETURN_FALSE(GetFileNameForKey(key, &filename)); | 
|  | #if BASE_VER < 800000 | 
|  | TEST_AND_RETURN_FALSE(base::DeleteFile(filename, false)); | 
|  | #else | 
|  | TEST_AND_RETURN_FALSE(base::DeleteFile(filename)); | 
|  | #endif | 
|  | return true; | 
|  | } | 
|  |  | 
|  | bool Prefs::FileStorage::GetFileNameForKey(std::string_view key, | 
|  | base::FilePath* filename) const { | 
|  | // Allows only non-empty keys containing [A-Za-z0-9_-/]. | 
|  | TEST_AND_RETURN_FALSE(!key.empty()); | 
|  | for (char c : key) | 
|  | TEST_AND_RETURN_FALSE(base::IsAsciiAlpha(c) || base::IsAsciiDigit(c) || | 
|  | c == '_' || c == '-' || c == kKeySeparator); | 
|  | *filename = prefs_dir_.Append( | 
|  | base::FilePath::StringPieceType(key.data(), key.size())); | 
|  | return true; | 
|  | } | 
|  |  | 
|  | // MemoryPrefs | 
|  |  | 
|  | bool MemoryPrefs::MemoryStorage::GetKey(std::string_view key, | 
|  | string* value) const { | 
|  | auto it = values_.find(key); | 
|  | if (it == values_.end()) | 
|  | return false; | 
|  | *value = it->second; | 
|  | return true; | 
|  | } | 
|  |  | 
|  | bool MemoryPrefs::MemoryStorage::GetSubKeys(std::string_view ns, | 
|  | vector<string>* keys) const { | 
|  | auto lower_comp = [](const auto& pr, const auto& ns) { | 
|  | return std::string_view{pr.first.data(), ns.length()} < ns; | 
|  | }; | 
|  | auto upper_comp = [](const auto& ns, const auto& pr) { | 
|  | return ns < std::string_view{pr.first.data(), ns.length()}; | 
|  | }; | 
|  | auto lower_it = | 
|  | std::lower_bound(begin(values_), end(values_), ns, lower_comp); | 
|  | auto upper_it = std::upper_bound(lower_it, end(values_), ns, upper_comp); | 
|  | while (lower_it != upper_it) | 
|  | keys->push_back((lower_it++)->first); | 
|  | return true; | 
|  | } | 
|  |  | 
|  | bool MemoryPrefs::MemoryStorage::SetKey(std::string_view key, | 
|  | std::string_view value) { | 
|  | values_[std::string{key}] = value; | 
|  | return true; | 
|  | } | 
|  |  | 
|  | bool MemoryPrefs::MemoryStorage::KeyExists(std::string_view key) const { | 
|  | return values_.find(key) != values_.end(); | 
|  | } | 
|  |  | 
|  | bool MemoryPrefs::MemoryStorage::DeleteKey(std::string_view key) { | 
|  | auto it = values_.find(key); | 
|  | if (it != values_.end()) | 
|  | values_.erase(it); | 
|  | return true; | 
|  | } | 
|  |  | 
|  | }  // namespace chromeos_update_engine |