blob: 77078cf03216875df98ceacc145d40640e608767 [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>
Daniel Zheng5eece042023-05-17 14:44:10 -070020#include <filesystem>
21#include <unistd.h>
Alex Deymod6f60072015-10-12 12:22:27 -070022
Kelvin Zhangc80d39b2023-10-25 09:30:16 -070023#include <android-base/file.h>
Andrew065d78d2020-04-07 15:43:07 -070024#include <base/files/file_enumerator.h>
Ben Chan06c76a42014-09-05 08:21:06 -070025#include <base/files/file_util.h>
Darin Petkov36275772010-10-01 11:40:57 -070026#include <base/logging.h>
Alex Vakulenko75039d72014-03-25 12:36:28 -070027#include <base/strings/string_number_conversions.h>
Andrew065d78d2020-04-07 15:43:07 -070028#include <base/strings/string_split.h>
Alex Vakulenko75039d72014-03-25 12:36:28 -070029#include <base/strings/string_util.h>
Darin Petkov36275772010-10-01 11:40:57 -070030
Alex Deymo39910dc2015-11-09 17:04:30 -080031#include "update_engine/common/utils.h"
Darin Petkov30030592010-07-27 13:53:20 -070032
33using std::string;
Jae Hoon Kimc1f36922020-05-11 18:20:18 -070034using std::vector;
Darin Petkov30030592010-07-27 13:53:20 -070035
36namespace chromeos_update_engine {
37
Jae Hoon Kimc1f36922020-05-11 18:20:18 -070038namespace {
39
40void DeleteEmptyDirectories(const base::FilePath& path) {
41 base::FileEnumerator path_enum(
42 path, false /* recursive */, base::FileEnumerator::DIRECTORIES);
43 for (base::FilePath dir_path = path_enum.Next(); !dir_path.empty();
44 dir_path = path_enum.Next()) {
45 DeleteEmptyDirectories(dir_path);
46 if (base::IsDirectoryEmpty(dir_path))
hscham043355b2020-11-17 16:50:10 +090047#if BASE_VER < 800000
Jae Hoon Kimc1f36922020-05-11 18:20:18 -070048 base::DeleteFile(dir_path, false);
hscham043355b2020-11-17 16:50:10 +090049#else
50 base::DeleteFile(dir_path);
51#endif
Jae Hoon Kimc1f36922020-05-11 18:20:18 -070052 }
53}
54
55} // namespace
56
Kelvin Zhang1c86a922021-05-13 10:30:48 -040057bool PrefsBase::GetString(const std::string_view key, string* value) const {
Alex Deymoa0284ac2016-07-22 12:51:41 -070058 return storage_->GetKey(key, value);
Darin Petkov30030592010-07-27 13:53:20 -070059}
60
Kelvin Zhang1c86a922021-05-13 10:30:48 -040061bool PrefsBase::SetString(std::string_view key, std::string_view value) {
Alex Deymoa0284ac2016-07-22 12:51:41 -070062 TEST_AND_RETURN_FALSE(storage_->SetKey(key, value));
Alex Deymod6f60072015-10-12 12:22:27 -070063 const auto observers_for_key = observers_.find(key);
64 if (observers_for_key != observers_.end()) {
65 std::vector<ObserverInterface*> copy_observers(observers_for_key->second);
66 for (ObserverInterface* observer : copy_observers)
67 observer->OnPrefSet(key);
68 }
Darin Petkov30030592010-07-27 13:53:20 -070069 return true;
70}
71
Kelvin Zhang1c86a922021-05-13 10:30:48 -040072bool PrefsBase::GetInt64(const std::string_view key, int64_t* value) const {
Darin Petkov30030592010-07-27 13:53:20 -070073 string str_value;
Jay Srinivasan08fce042012-06-07 16:31:01 -070074 if (!GetString(key, &str_value))
75 return false;
Ben Chan736fcb52014-05-21 18:28:22 -070076 base::TrimWhitespaceASCII(str_value, base::TRIM_ALL, &str_value);
Kelvin Zhangc80d39b2023-10-25 09:30:16 -070077 if (str_value.empty()) {
78 LOG(ERROR) << "When reading pref " << key
79 << ", got an empty value after trim";
80 return false;
81 }
82 if (!base::StringToInt64(str_value, value)) {
83 LOG(ERROR) << "When reading pref " << key << ", failed to convert value "
84 << str_value << " to integer";
85 return false;
86 }
Darin Petkov30030592010-07-27 13:53:20 -070087 return true;
88}
89
Kelvin Zhang1c86a922021-05-13 10:30:48 -040090bool PrefsBase::SetInt64(std::string_view key, const int64_t value) {
hscham00b6aa22020-02-20 12:32:06 +090091 return SetString(key, base::NumberToString(value));
Darin Petkov30030592010-07-27 13:53:20 -070092}
93
Kelvin Zhang1c86a922021-05-13 10:30:48 -040094bool PrefsBase::GetBoolean(std::string_view key, bool* value) const {
Alex Deymoefb7c4c2013-07-09 14:34:00 -070095 string str_value;
96 if (!GetString(key, &str_value))
97 return false;
Ben Chan736fcb52014-05-21 18:28:22 -070098 base::TrimWhitespaceASCII(str_value, base::TRIM_ALL, &str_value);
Alex Deymoefb7c4c2013-07-09 14:34:00 -070099 if (str_value == "false") {
100 *value = false;
101 return true;
102 }
103 if (str_value == "true") {
104 *value = true;
105 return true;
106 }
107 return false;
108}
109
Kelvin Zhang1c86a922021-05-13 10:30:48 -0400110bool PrefsBase::SetBoolean(std::string_view key, const bool value) {
Alex Deymoefb7c4c2013-07-09 14:34:00 -0700111 return SetString(key, value ? "true" : "false");
112}
113
Kelvin Zhang1c86a922021-05-13 10:30:48 -0400114bool PrefsBase::Exists(std::string_view key) const {
Alex Deymoa0284ac2016-07-22 12:51:41 -0700115 return storage_->KeyExists(key);
Jay Srinivasan480ddfa2012-06-01 19:15:26 -0700116}
117
Kelvin Zhang1c86a922021-05-13 10:30:48 -0400118bool PrefsBase::Delete(std::string_view key) {
Alex Deymoa0284ac2016-07-22 12:51:41 -0700119 TEST_AND_RETURN_FALSE(storage_->DeleteKey(key));
Alex Deymod6f60072015-10-12 12:22:27 -0700120 const auto observers_for_key = observers_.find(key);
121 if (observers_for_key != observers_.end()) {
122 std::vector<ObserverInterface*> copy_observers(observers_for_key->second);
123 for (ObserverInterface* observer : copy_observers)
124 observer->OnPrefDeleted(key);
125 }
126 return true;
127}
128
Kelvin Zhang1c86a922021-05-13 10:30:48 -0400129bool PrefsBase::Delete(std::string_view pref_key, const vector<string>& nss) {
Vyshu Khota4c5413d2020-11-04 16:17:25 -0800130 // Delete pref key for platform.
131 bool success = Delete(pref_key);
132 // Delete pref key in each namespace.
133 for (const auto& ns : nss) {
134 vector<string> namespace_keys;
135 success = GetSubKeys(ns, &namespace_keys) && success;
136 for (const auto& key : namespace_keys) {
137 auto last_key_seperator = key.find_last_of(kKeySeparator);
138 if (last_key_seperator != string::npos &&
139 pref_key == key.substr(last_key_seperator + 1)) {
140 success = Delete(key) && success;
141 }
142 }
143 }
144 return success;
145}
146
Kelvin Zhang1c86a922021-05-13 10:30:48 -0400147bool PrefsBase::GetSubKeys(std::string_view ns, vector<string>* keys) const {
Jae Hoon Kim29a80e02020-05-11 20:18:49 -0700148 return storage_->GetSubKeys(ns, keys);
149}
150
Kelvin Zhang1c86a922021-05-13 10:30:48 -0400151void PrefsBase::AddObserver(std::string_view key, ObserverInterface* observer) {
152 observers_[std::string{key}].push_back(observer);
Alex Deymod6f60072015-10-12 12:22:27 -0700153}
154
Kelvin Zhang1c86a922021-05-13 10:30:48 -0400155void PrefsBase::RemoveObserver(std::string_view key,
156 ObserverInterface* observer) {
157 std::vector<ObserverInterface*>& observers_for_key =
158 observers_[std::string{key}];
Alex Deymod6f60072015-10-12 12:22:27 -0700159 auto observer_it =
160 std::find(observers_for_key.begin(), observers_for_key.end(), observer);
161 if (observer_it != observers_for_key.end())
162 observers_for_key.erase(observer_it);
Jay Srinivasan480ddfa2012-06-01 19:15:26 -0700163}
164
Jae Hoon Kimc1f36922020-05-11 18:20:18 -0700165string PrefsInterface::CreateSubKey(const vector<string>& ns_and_key) {
166 return base::JoinString(ns_and_key, string(1, kKeySeparator));
Andrew065d78d2020-04-07 15:43:07 -0700167}
168
Alex Deymoa0284ac2016-07-22 12:51:41 -0700169// Prefs
170
171bool Prefs::Init(const base::FilePath& prefs_dir) {
172 return file_storage_.Init(prefs_dir);
173}
174
Daniel Zheng5eece042023-05-17 14:44:10 -0700175bool PrefsBase::StartTransaction() {
176 return storage_->CreateTemporaryPrefs();
177}
178
179bool PrefsBase::CancelTransaction() {
180 return storage_->DeleteTemporaryPrefs();
181}
182
183bool PrefsBase::SubmitTransaction() {
184 return storage_->SwapPrefs();
185}
186
187std::string Prefs::FileStorage::GetTemporaryDir() const {
188 return prefs_dir_.value() + "_tmp";
189}
190
191bool Prefs::FileStorage::CreateTemporaryPrefs() {
192 // Delete any existing prefs_tmp
193 DeleteTemporaryPrefs();
194 // Get the paths to the source and destination directories.
195 std::filesystem::path source_directory(prefs_dir_.value());
196 std::filesystem::path destination_directory(GetTemporaryDir());
197
198 if (!std::filesystem::exists(source_directory)) {
199 LOG(ERROR) << "prefs directory does not exist: " << source_directory;
200 return false;
201 }
202 // Copy the directory.
Daniel Zhenga77e6452024-08-13 13:52:55 -0700203 std::error_code e;
204 std::filesystem::copy(source_directory, destination_directory, e);
205 if (e) {
206 LOG(ERROR) << "failed to copy prefs to prefs_tmp: " << e.message();
207 return false;
208 }
Daniel Zheng5eece042023-05-17 14:44:10 -0700209
210 return true;
211}
212
213bool Prefs::FileStorage::DeleteTemporaryPrefs() {
214 std::filesystem::path destination_directory(GetTemporaryDir());
215
216 if (std::filesystem::exists(destination_directory)) {
Daniel Zhenga77e6452024-08-13 13:52:55 -0700217 std::error_code e;
218 std::filesystem::remove_all(destination_directory, e);
219 if (e) {
220 LOG(ERROR) << "failed to remove directory: " << e.message();
221 return false;
222 }
Daniel Zheng5eece042023-05-17 14:44:10 -0700223 }
224 return true;
225}
226
227bool Prefs::FileStorage::SwapPrefs() {
Kelvin Zhangc80d39b2023-10-25 09:30:16 -0700228 if (!utils::DeleteDirectory(prefs_dir_.value().c_str())) {
229 LOG(ERROR) << "Failed to remove prefs dir " << prefs_dir_;
230 return false;
Daniel Zheng5eece042023-05-17 14:44:10 -0700231 }
232 if (rename(GetTemporaryDir().c_str(), prefs_dir_.value().c_str()) != 0) {
233 LOG(ERROR) << "Error replacing prefs with prefs_tmp" << strerror(errno);
234 return false;
235 }
Kelvin Zhangc80d39b2023-10-25 09:30:16 -0700236 if (!utils::FsyncDirectory(
237 android::base::Dirname(prefs_dir_.value()).c_str())) {
238 PLOG(ERROR) << "Failed to fsync prefs parent dir after swapping prefs";
239 }
Daniel Zheng5eece042023-05-17 14:44:10 -0700240 return true;
241}
242
Alex Deymoa0284ac2016-07-22 12:51:41 -0700243bool Prefs::FileStorage::Init(const base::FilePath& prefs_dir) {
244 prefs_dir_ = prefs_dir;
Kelvin Zhangc80d39b2023-10-25 09:30:16 -0700245 if (!std::filesystem::exists(prefs_dir_.value())) {
246 LOG(INFO) << "Prefs dir does not exist, possibly due to an interrupted "
247 "transaction.";
248 if (std::filesystem::exists(GetTemporaryDir())) {
249 SwapPrefs();
250 }
251 }
Daniel Zheng5eece042023-05-17 14:44:10 -0700252
Daniel Zheng37f212d2024-08-13 12:38:13 -0700253 if (std::filesystem::exists(GetTemporaryDir())) {
254 LOG(INFO)
255 << "Deleting temporary prefs, checkpoint transaction was interrupted";
256 if (!utils::DeleteDirectory(GetTemporaryDir().c_str())) {
257 LOG(ERROR) << "Failed to delete temporary prefs";
258 return false;
259 }
260 }
261
Andrew065d78d2020-04-07 15:43:07 -0700262 // Delete empty directories. Ignore errors when deleting empty directories.
Jae Hoon Kimc1f36922020-05-11 18:20:18 -0700263 DeleteEmptyDirectories(prefs_dir_);
Alex Deymoa0284ac2016-07-22 12:51:41 -0700264 return true;
265}
266
Kelvin Zhang1c86a922021-05-13 10:30:48 -0400267bool Prefs::FileStorage::GetKey(std::string_view key, string* value) const {
Alex Deymoa0284ac2016-07-22 12:51:41 -0700268 base::FilePath filename;
269 TEST_AND_RETURN_FALSE(GetFileNameForKey(key, &filename));
270 if (!base::ReadFileToString(filename, value)) {
Alex Deymoa0284ac2016-07-22 12:51:41 -0700271 return false;
272 }
273 return true;
274}
275
Kelvin Zhang1c86a922021-05-13 10:30:48 -0400276bool Prefs::FileStorage::GetSubKeys(std::string_view ns,
Jae Hoon Kim29a80e02020-05-11 20:18:49 -0700277 vector<string>* keys) const {
278 base::FilePath filename;
279 TEST_AND_RETURN_FALSE(GetFileNameForKey(ns, &filename));
280 base::FileEnumerator namespace_enum(
281 prefs_dir_, true, base::FileEnumerator::FILES);
282 for (base::FilePath f = namespace_enum.Next(); !f.empty();
283 f = namespace_enum.Next()) {
284 auto filename_str = filename.value();
285 if (f.value().compare(0, filename_str.length(), filename_str) == 0) {
286 // Only return the key portion excluding the |prefs_dir_| with slash.
287 keys->push_back(f.value().substr(
288 prefs_dir_.AsEndingWithSeparator().value().length()));
289 }
290 }
291 return true;
292}
293
Kelvin Zhang1c86a922021-05-13 10:30:48 -0400294bool Prefs::FileStorage::SetKey(std::string_view key, std::string_view value) {
Alex Deymoa0284ac2016-07-22 12:51:41 -0700295 base::FilePath filename;
296 TEST_AND_RETURN_FALSE(GetFileNameForKey(key, &filename));
297 if (!base::DirectoryExists(filename.DirName())) {
298 // Only attempt to create the directory if it doesn't exist to avoid calls
299 // to parent directories where we might not have permission to write to.
300 TEST_AND_RETURN_FALSE(base::CreateDirectory(filename.DirName()));
301 }
Kelvin Zhang49170aa2022-11-28 10:55:16 -0800302 TEST_AND_RETURN_FALSE(
303 utils::WriteStringToFileAtomic(filename.value(), value));
Alex Deymoa0284ac2016-07-22 12:51:41 -0700304 return true;
305}
306
Kelvin Zhang1c86a922021-05-13 10:30:48 -0400307bool Prefs::FileStorage::KeyExists(std::string_view key) const {
Alex Deymoa0284ac2016-07-22 12:51:41 -0700308 base::FilePath filename;
309 TEST_AND_RETURN_FALSE(GetFileNameForKey(key, &filename));
310 return base::PathExists(filename);
311}
312
Kelvin Zhang1c86a922021-05-13 10:30:48 -0400313bool Prefs::FileStorage::DeleteKey(std::string_view key) {
Alex Deymoa0284ac2016-07-22 12:51:41 -0700314 base::FilePath filename;
315 TEST_AND_RETURN_FALSE(GetFileNameForKey(key, &filename));
hscham043355b2020-11-17 16:50:10 +0900316#if BASE_VER < 800000
Jae Hoon Kim29a80e02020-05-11 20:18:49 -0700317 TEST_AND_RETURN_FALSE(base::DeleteFile(filename, false));
hscham043355b2020-11-17 16:50:10 +0900318#else
319 TEST_AND_RETURN_FALSE(base::DeleteFile(filename));
320#endif
Alex Deymoa0284ac2016-07-22 12:51:41 -0700321 return true;
322}
323
Kelvin Zhang1c86a922021-05-13 10:30:48 -0400324bool Prefs::FileStorage::GetFileNameForKey(std::string_view key,
Alex Deymoa0284ac2016-07-22 12:51:41 -0700325 base::FilePath* filename) const {
Jae Hoon Kim29a80e02020-05-11 20:18:49 -0700326 // Allows only non-empty keys containing [A-Za-z0-9_-/].
Darin Petkov30030592010-07-27 13:53:20 -0700327 TEST_AND_RETURN_FALSE(!key.empty());
Jae Hoon Kim29a80e02020-05-11 20:18:49 -0700328 for (char c : key)
Alex Vakulenko0103c362016-01-20 07:56:15 -0800329 TEST_AND_RETURN_FALSE(base::IsAsciiAlpha(c) || base::IsAsciiDigit(c) ||
Andrew065d78d2020-04-07 15:43:07 -0700330 c == '_' || c == '-' || c == kKeySeparator);
Daniel Zheng5eece042023-05-17 14:44:10 -0700331 if (std::filesystem::exists(GetTemporaryDir())) {
332 *filename =
333 base::FilePath(GetTemporaryDir())
334 .Append(base::FilePath::StringPieceType(key.data(), key.size()));
335 } else {
336 *filename = prefs_dir_.Append(
337 base::FilePath::StringPieceType(key.data(), key.size()));
338 }
Darin Petkov30030592010-07-27 13:53:20 -0700339 return true;
340}
341
Alex Deymoa0284ac2016-07-22 12:51:41 -0700342// MemoryPrefs
343
Kelvin Zhang1c86a922021-05-13 10:30:48 -0400344bool MemoryPrefs::MemoryStorage::GetKey(std::string_view key,
Alex Deymoa0284ac2016-07-22 12:51:41 -0700345 string* value) const {
346 auto it = values_.find(key);
347 if (it == values_.end())
348 return false;
349 *value = it->second;
350 return true;
351}
352
Kelvin Zhang1c86a922021-05-13 10:30:48 -0400353bool MemoryPrefs::MemoryStorage::GetSubKeys(std::string_view ns,
Jae Hoon Kim29a80e02020-05-11 20:18:49 -0700354 vector<string>* keys) const {
Kelvin Zhang1c86a922021-05-13 10:30:48 -0400355 auto lower_comp = [](const auto& pr, const auto& ns) {
356 return std::string_view{pr.first.data(), ns.length()} < ns;
Jae Hoon Kim29a80e02020-05-11 20:18:49 -0700357 };
Kelvin Zhang1c86a922021-05-13 10:30:48 -0400358 auto upper_comp = [](const auto& ns, const auto& pr) {
359 return ns < std::string_view{pr.first.data(), ns.length()};
Jae Hoon Kim29a80e02020-05-11 20:18:49 -0700360 };
361 auto lower_it =
362 std::lower_bound(begin(values_), end(values_), ns, lower_comp);
363 auto upper_it = std::upper_bound(lower_it, end(values_), ns, upper_comp);
364 while (lower_it != upper_it)
365 keys->push_back((lower_it++)->first);
366 return true;
367}
368
Kelvin Zhang1c86a922021-05-13 10:30:48 -0400369bool MemoryPrefs::MemoryStorage::SetKey(std::string_view key,
Kelvin Zhangcf4600e2020-10-27 15:50:33 -0400370 std::string_view value) {
Kelvin Zhang1c86a922021-05-13 10:30:48 -0400371 values_[std::string{key}] = value;
Alex Deymoa0284ac2016-07-22 12:51:41 -0700372 return true;
373}
374
Kelvin Zhang1c86a922021-05-13 10:30:48 -0400375bool MemoryPrefs::MemoryStorage::KeyExists(std::string_view key) const {
Alex Deymoa0284ac2016-07-22 12:51:41 -0700376 return values_.find(key) != values_.end();
377}
378
Kelvin Zhang1c86a922021-05-13 10:30:48 -0400379bool MemoryPrefs::MemoryStorage::DeleteKey(std::string_view key) {
Alex Deymoa0284ac2016-07-22 12:51:41 -0700380 auto it = values_.find(key);
Andrew914f5542020-04-21 10:56:33 -0700381 if (it != values_.end())
382 values_.erase(it);
Alex Deymoa0284ac2016-07-22 12:51:41 -0700383 return true;
384}
385
Darin Petkov30030592010-07-27 13:53:20 -0700386} // namespace chromeos_update_engine