| // |
| // 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 |