| // | 
 | // Copyright (C) 2014 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/fake_prefs.h" | 
 |  | 
 | #include <algorithm> | 
 | #include <utility> | 
 |  | 
 | #include <gtest/gtest.h> | 
 |  | 
 | using std::string; | 
 | using std::vector; | 
 |  | 
 | using chromeos_update_engine::FakePrefs; | 
 |  | 
 | namespace { | 
 |  | 
 | void CheckNotNull(std::string_view key, void* ptr) { | 
 |   EXPECT_NE(nullptr, ptr) << "Called Get*() for key \"" << key | 
 |                           << "\" with a null parameter."; | 
 | } | 
 |  | 
 | }  // namespace | 
 |  | 
 | namespace chromeos_update_engine { | 
 |  | 
 | FakePrefs::~FakePrefs() { | 
 |   EXPECT_TRUE(observers_.empty()); | 
 | } | 
 |  | 
 | // Compile-time type-dependent constants definitions. | 
 | template <> | 
 | FakePrefs::PrefType const FakePrefs::PrefConsts<string>::type = | 
 |     FakePrefs::PrefType::kString; | 
 | template <> | 
 | string FakePrefs::PrefValue::*const  // NOLINT(runtime/string), not static str. | 
 |     FakePrefs::PrefConsts<string>::member = &FakePrefs::PrefValue::as_str; | 
 |  | 
 | template <> | 
 | FakePrefs::PrefType const FakePrefs::PrefConsts<int64_t>::type = | 
 |     FakePrefs::PrefType::kInt64; | 
 | template <> | 
 | int64_t FakePrefs::PrefValue::*const FakePrefs::PrefConsts<int64_t>::member = | 
 |     &FakePrefs::PrefValue::as_int64; | 
 |  | 
 | template <> | 
 | FakePrefs::PrefType const FakePrefs::PrefConsts<bool>::type = | 
 |     FakePrefs::PrefType::kBool; | 
 | template <> | 
 | bool FakePrefs::PrefValue::*const FakePrefs::PrefConsts<bool>::member = | 
 |     &FakePrefs::PrefValue::as_bool; | 
 |  | 
 | bool FakePrefs::GetString(std::string_view key, string* value) const { | 
 |   return GetValue(key, value); | 
 | } | 
 |  | 
 | bool FakePrefs::SetString(std::string_view key, std::string_view value) { | 
 |   SetValue(key, std::string(value)); | 
 |   return true; | 
 | } | 
 |  | 
 | bool FakePrefs::GetInt64(std::string_view key, int64_t* value) const { | 
 |   return GetValue(key, value); | 
 | } | 
 |  | 
 | bool FakePrefs::SetInt64(std::string_view key, const int64_t value) { | 
 |   SetValue(key, value); | 
 |   return true; | 
 | } | 
 |  | 
 | bool FakePrefs::GetBoolean(std::string_view key, bool* value) const { | 
 |   return GetValue(key, value); | 
 | } | 
 |  | 
 | bool FakePrefs::SetBoolean(std::string_view key, const bool value) { | 
 |   SetValue(key, value); | 
 |   return true; | 
 | } | 
 |  | 
 | bool FakePrefs::Exists(std::string_view key) const { | 
 |   return values_.find(key) != values_.end(); | 
 | } | 
 |  | 
 | bool FakePrefs::Delete(std::string_view key) { | 
 |   if (values_.find(key) == values_.end()) | 
 |     return false; | 
 |   values_.erase(std::string{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 FakePrefs::Delete(std::string_view key, const vector<string>& nss) { | 
 |   bool success = Delete(key); | 
 |   for (const auto& ns : nss) { | 
 |     vector<string> ns_keys; | 
 |     success = GetSubKeys(ns, &ns_keys) && success; | 
 |     for (const auto& sub_key : ns_keys) { | 
 |       auto last_key_seperator = sub_key.find_last_of(kKeySeparator); | 
 |       if (last_key_seperator != string::npos && | 
 |           key == sub_key.substr(last_key_seperator + 1)) { | 
 |         success = Delete(sub_key) && success; | 
 |       } | 
 |     } | 
 |   } | 
 |   return success; | 
 | } | 
 |  | 
 | bool FakePrefs::GetSubKeys(std::string_view ns, vector<string>* keys) const { | 
 |   for (const auto& pr : values_) | 
 |     if (pr.first.compare(0, ns.length(), ns) == 0) | 
 |       keys->push_back(pr.first); | 
 |   return true; | 
 | } | 
 |  | 
 | string FakePrefs::GetTypeName(PrefType type) { | 
 |   switch (type) { | 
 |     case PrefType::kString: | 
 |       return "string"; | 
 |     case PrefType::kInt64: | 
 |       return "int64_t"; | 
 |     case PrefType::kBool: | 
 |       return "bool"; | 
 |   } | 
 |   return "Unknown"; | 
 | } | 
 |  | 
 | void FakePrefs::CheckKeyType(std::string_view key, PrefType type) const { | 
 |   auto it = values_.find(key); | 
 |   EXPECT_TRUE(it == values_.end() || it->second.type == type) | 
 |       << "Key \"" << key << "\" if defined as " << GetTypeName(it->second.type) | 
 |       << " but is accessed as a " << GetTypeName(type); | 
 | } | 
 |  | 
 | template <typename T> | 
 | void FakePrefs::SetValue(std::string_view key, T value) { | 
 |   std::string str_key{key}; | 
 |   CheckKeyType(key, PrefConsts<T>::type); | 
 |   values_[str_key].type = PrefConsts<T>::type; | 
 |   values_[str_key].value.*(PrefConsts<T>::member) = std::move(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); | 
 |   } | 
 | } | 
 |  | 
 | template <typename T> | 
 | bool FakePrefs::GetValue(std::string_view key, T* value) const { | 
 |   CheckKeyType(key, PrefConsts<T>::type); | 
 |   auto it = values_.find(key); | 
 |   if (it == values_.end()) | 
 |     return false; | 
 |   CheckNotNull(key, value); | 
 |   *value = it->second.value.*(PrefConsts<T>::member); | 
 |   return true; | 
 | } | 
 |  | 
 | void FakePrefs::AddObserver(std::string_view key, ObserverInterface* observer) { | 
 |   observers_[string{key}].push_back(observer); | 
 | } | 
 |  | 
 | void FakePrefs::RemoveObserver(std::string_view key, | 
 |                                ObserverInterface* observer) { | 
 |   string str_key{key}; | 
 |   std::vector<ObserverInterface*>& observers_for_key = observers_[str_key]; | 
 |   auto observer_it = | 
 |       std::find(observers_for_key.begin(), observers_for_key.end(), observer); | 
 |   EXPECT_NE(observer_it, observers_for_key.end()) | 
 |       << "Trying to remove an observer instance not watching the key " << key; | 
 |   if (observer_it != observers_for_key.end()) | 
 |     observers_for_key.erase(observer_it); | 
 |   if (observers_for_key.empty()) | 
 |     observers_.erase(str_key); | 
 | } | 
 |  | 
 | }  // namespace chromeos_update_engine |