blob: 6a330378e493140e9a6666292263f74a87926707 [file] [log] [blame]
Alex Deymoaea4c1c2015-08-19 20:24:43 -07001//
2// Copyright (C) 2012 The Android Open Source Project
3//
4// Licensed under the Apache License, Version 2.0 (the "License");
5// you may not use this file except in compliance with the License.
6// You may obtain a copy of the License at
7//
8// http://www.apache.org/licenses/LICENSE-2.0
9//
10// Unless required by applicable law or agreed to in writing, software
11// distributed under the License is distributed on an "AS IS" BASIS,
12// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13// See the License for the specific language governing permissions and
14// limitations under the License.
15//
Darin Petkov30030592010-07-27 13:53:20 -070016
Alex Deymo39910dc2015-11-09 17:04:30 -080017#include "update_engine/common/prefs.h"
Darin Petkov30030592010-07-27 13:53:20 -070018
Alex Deymod6f60072015-10-12 12:22:27 -070019#include <algorithm>
20
Andrew065d78d2020-04-07 15:43:07 -070021#include <base/files/file_enumerator.h>
Ben Chan06c76a42014-09-05 08:21:06 -070022#include <base/files/file_util.h>
Darin Petkov36275772010-10-01 11:40:57 -070023#include <base/logging.h>
Alex Vakulenko75039d72014-03-25 12:36:28 -070024#include <base/strings/string_number_conversions.h>
Andrew065d78d2020-04-07 15:43:07 -070025#include <base/strings/string_split.h>
Alex Vakulenko75039d72014-03-25 12:36:28 -070026#include <base/strings/string_util.h>
Darin Petkov36275772010-10-01 11:40:57 -070027
Alex Deymo39910dc2015-11-09 17:04:30 -080028#include "update_engine/common/utils.h"
Darin Petkov30030592010-07-27 13:53:20 -070029
30using std::string;
31
32namespace chromeos_update_engine {
33
Andrew065d78d2020-04-07 15:43:07 -070034const char kKeySeparator = '/';
35
Alex Deymoa0284ac2016-07-22 12:51:41 -070036bool PrefsBase::GetString(const string& key, string* value) const {
37 return storage_->GetKey(key, value);
Darin Petkov30030592010-07-27 13:53:20 -070038}
39
Alex Deymoa0284ac2016-07-22 12:51:41 -070040bool PrefsBase::SetString(const string& key, const string& value) {
41 TEST_AND_RETURN_FALSE(storage_->SetKey(key, value));
Alex Deymod6f60072015-10-12 12:22:27 -070042 const auto observers_for_key = observers_.find(key);
43 if (observers_for_key != observers_.end()) {
44 std::vector<ObserverInterface*> copy_observers(observers_for_key->second);
45 for (ObserverInterface* observer : copy_observers)
46 observer->OnPrefSet(key);
47 }
Darin Petkov30030592010-07-27 13:53:20 -070048 return true;
49}
50
Alex Deymoa0284ac2016-07-22 12:51:41 -070051bool PrefsBase::GetInt64(const string& key, int64_t* value) const {
Darin Petkov30030592010-07-27 13:53:20 -070052 string str_value;
Jay Srinivasan08fce042012-06-07 16:31:01 -070053 if (!GetString(key, &str_value))
54 return false;
Ben Chan736fcb52014-05-21 18:28:22 -070055 base::TrimWhitespaceASCII(str_value, base::TRIM_ALL, &str_value);
Chris Masone790e62e2010-08-12 10:41:18 -070056 TEST_AND_RETURN_FALSE(base::StringToInt64(str_value, value));
Darin Petkov30030592010-07-27 13:53:20 -070057 return true;
58}
59
Alex Deymoa0284ac2016-07-22 12:51:41 -070060bool PrefsBase::SetInt64(const string& key, const int64_t value) {
hscham00b6aa22020-02-20 12:32:06 +090061 return SetString(key, base::NumberToString(value));
Darin Petkov30030592010-07-27 13:53:20 -070062}
63
Alex Deymoa0284ac2016-07-22 12:51:41 -070064bool PrefsBase::GetBoolean(const string& key, bool* value) const {
Alex Deymoefb7c4c2013-07-09 14:34:00 -070065 string str_value;
66 if (!GetString(key, &str_value))
67 return false;
Ben Chan736fcb52014-05-21 18:28:22 -070068 base::TrimWhitespaceASCII(str_value, base::TRIM_ALL, &str_value);
Alex Deymoefb7c4c2013-07-09 14:34:00 -070069 if (str_value == "false") {
70 *value = false;
71 return true;
72 }
73 if (str_value == "true") {
74 *value = true;
75 return true;
76 }
77 return false;
78}
79
Alex Deymoa0284ac2016-07-22 12:51:41 -070080bool PrefsBase::SetBoolean(const string& key, const bool value) {
Alex Deymoefb7c4c2013-07-09 14:34:00 -070081 return SetString(key, value ? "true" : "false");
82}
83
Alex Deymoa0284ac2016-07-22 12:51:41 -070084bool PrefsBase::Exists(const string& key) const {
85 return storage_->KeyExists(key);
Jay Srinivasan480ddfa2012-06-01 19:15:26 -070086}
87
Alex Deymoa0284ac2016-07-22 12:51:41 -070088bool PrefsBase::Delete(const string& key) {
89 TEST_AND_RETURN_FALSE(storage_->DeleteKey(key));
Alex Deymod6f60072015-10-12 12:22:27 -070090 const auto observers_for_key = observers_.find(key);
91 if (observers_for_key != observers_.end()) {
92 std::vector<ObserverInterface*> copy_observers(observers_for_key->second);
93 for (ObserverInterface* observer : copy_observers)
94 observer->OnPrefDeleted(key);
95 }
96 return true;
97}
98
Alex Deymoa0284ac2016-07-22 12:51:41 -070099void PrefsBase::AddObserver(const string& key, ObserverInterface* observer) {
Alex Deymod6f60072015-10-12 12:22:27 -0700100 observers_[key].push_back(observer);
101}
102
Alex Deymoa0284ac2016-07-22 12:51:41 -0700103void PrefsBase::RemoveObserver(const string& key, ObserverInterface* observer) {
Alex Deymod6f60072015-10-12 12:22:27 -0700104 std::vector<ObserverInterface*>& observers_for_key = observers_[key];
105 auto observer_it =
106 std::find(observers_for_key.begin(), observers_for_key.end(), observer);
107 if (observer_it != observers_for_key.end())
108 observers_for_key.erase(observer_it);
Jay Srinivasan480ddfa2012-06-01 19:15:26 -0700109}
110
Andrew065d78d2020-04-07 15:43:07 -0700111string PrefsInterface::CreateSubKey(const string& name_space,
112 const string& sub_pref,
113 const string& key) {
114 return base::JoinString({name_space, sub_pref, key},
115 string(1, kKeySeparator));
116}
117
Alex Deymoa0284ac2016-07-22 12:51:41 -0700118// Prefs
119
120bool Prefs::Init(const base::FilePath& prefs_dir) {
121 return file_storage_.Init(prefs_dir);
122}
123
124bool Prefs::FileStorage::Init(const base::FilePath& prefs_dir) {
125 prefs_dir_ = prefs_dir;
Andrew065d78d2020-04-07 15:43:07 -0700126 // Delete empty directories. Ignore errors when deleting empty directories.
127 base::FileEnumerator namespace_enum(
128 prefs_dir_, false /* recursive */, base::FileEnumerator::DIRECTORIES);
129 for (base::FilePath namespace_path = namespace_enum.Next();
130 !namespace_path.empty();
131 namespace_path = namespace_enum.Next()) {
132 base::FileEnumerator sub_pref_enum(namespace_path,
133 false /* recursive */,
134 base::FileEnumerator::DIRECTORIES);
135 for (base::FilePath sub_pref_path = sub_pref_enum.Next();
136 !sub_pref_path.empty();
137 sub_pref_path = sub_pref_enum.Next()) {
138 if (base::IsDirectoryEmpty(sub_pref_path))
139 base::DeleteFile(sub_pref_path, false);
140 }
141 if (base::IsDirectoryEmpty(namespace_path))
142 base::DeleteFile(namespace_path, false);
143 }
Alex Deymoa0284ac2016-07-22 12:51:41 -0700144 return true;
145}
146
147bool Prefs::FileStorage::GetKey(const string& key, string* value) const {
148 base::FilePath filename;
149 TEST_AND_RETURN_FALSE(GetFileNameForKey(key, &filename));
150 if (!base::ReadFileToString(filename, value)) {
Alex Deymoa0284ac2016-07-22 12:51:41 -0700151 return false;
152 }
153 return true;
154}
155
156bool Prefs::FileStorage::SetKey(const string& key, const string& value) {
157 base::FilePath filename;
158 TEST_AND_RETURN_FALSE(GetFileNameForKey(key, &filename));
159 if (!base::DirectoryExists(filename.DirName())) {
160 // Only attempt to create the directory if it doesn't exist to avoid calls
161 // to parent directories where we might not have permission to write to.
162 TEST_AND_RETURN_FALSE(base::CreateDirectory(filename.DirName()));
163 }
164 TEST_AND_RETURN_FALSE(base::WriteFile(filename, value.data(), value.size()) ==
165 static_cast<int>(value.size()));
166 return true;
167}
168
169bool Prefs::FileStorage::KeyExists(const string& key) const {
170 base::FilePath filename;
171 TEST_AND_RETURN_FALSE(GetFileNameForKey(key, &filename));
172 return base::PathExists(filename);
173}
174
175bool Prefs::FileStorage::DeleteKey(const string& key) {
176 base::FilePath filename;
177 TEST_AND_RETURN_FALSE(GetFileNameForKey(key, &filename));
Andrew065d78d2020-04-07 15:43:07 -0700178 TEST_AND_RETURN_FALSE(base::DeleteFile(filename, true));
Alex Deymoa0284ac2016-07-22 12:51:41 -0700179 return true;
180}
181
182bool Prefs::FileStorage::GetFileNameForKey(const string& key,
183 base::FilePath* filename) const {
Darin Petkov30030592010-07-27 13:53:20 -0700184 // Allows only non-empty keys containing [A-Za-z0-9_-].
185 TEST_AND_RETURN_FALSE(!key.empty());
186 for (size_t i = 0; i < key.size(); ++i) {
187 char c = key.at(i);
Alex Vakulenko0103c362016-01-20 07:56:15 -0800188 TEST_AND_RETURN_FALSE(base::IsAsciiAlpha(c) || base::IsAsciiDigit(c) ||
Andrew065d78d2020-04-07 15:43:07 -0700189 c == '_' || c == '-' || c == kKeySeparator);
Darin Petkov30030592010-07-27 13:53:20 -0700190 }
191 *filename = prefs_dir_.Append(key);
192 return true;
193}
194
Alex Deymoa0284ac2016-07-22 12:51:41 -0700195// MemoryPrefs
196
197bool MemoryPrefs::MemoryStorage::GetKey(const string& key,
198 string* value) const {
199 auto it = values_.find(key);
200 if (it == values_.end())
201 return false;
202 *value = it->second;
203 return true;
204}
205
206bool MemoryPrefs::MemoryStorage::SetKey(const string& key,
207 const string& value) {
208 values_[key] = value;
209 return true;
210}
211
212bool MemoryPrefs::MemoryStorage::KeyExists(const string& key) const {
213 return values_.find(key) != values_.end();
214}
215
216bool MemoryPrefs::MemoryStorage::DeleteKey(const string& key) {
217 auto it = values_.find(key);
Andrew914f5542020-04-21 10:56:33 -0700218 if (it != values_.end())
219 values_.erase(it);
Alex Deymoa0284ac2016-07-22 12:51:41 -0700220 return true;
221}
222
Darin Petkov30030592010-07-27 13:53:20 -0700223} // namespace chromeos_update_engine