xref: /aosp_15_r20/system/update_engine/common/prefs.cc (revision 5a9231315b4521097b8dc3750bc806fcafe0c72f)
1 //
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 //
16 
17 #include "update_engine/common/prefs.h"
18 
19 #include <algorithm>
20 #include <filesystem>
21 #include <unistd.h>
22 
23 #include <android-base/file.h>
24 #include <android-base/parseint.h>
25 #include <base/files/file_enumerator.h>
26 #include <base/files/file_util.h>
27 #include <base/logging.h>
28 #include <base/strings/string_split.h>
29 
30 #include "android-base/strings.h"
31 #include "update_engine/common/utils.h"
32 
33 using std::string;
34 using std::vector;
35 
36 namespace chromeos_update_engine {
37 
38 namespace {
39 
DeleteEmptyDirectories(const base::FilePath & path)40 void 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))
47 #if BASE_VER < 800000
48       base::DeleteFile(dir_path, false);
49 #else
50       base::DeleteFile(dir_path);
51 #endif
52   }
53 }
54 
55 }  // namespace
56 
GetString(const std::string_view key,string * value) const57 bool PrefsBase::GetString(const std::string_view key, string* value) const {
58   return storage_->GetKey(key, value);
59 }
60 
SetString(std::string_view key,std::string_view value)61 bool PrefsBase::SetString(std::string_view key, std::string_view value) {
62   TEST_AND_RETURN_FALSE(storage_->SetKey(key, value));
63   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   }
69   return true;
70 }
71 
GetInt64(const std::string_view key,int64_t * value) const72 bool PrefsBase::GetInt64(const std::string_view key, int64_t* value) const {
73   string str_value;
74   if (!GetString(key, &str_value))
75     return false;
76   str_value = android::base::Trim(str_value);
77   if (str_value.empty()) {
78     LOG(ERROR) << "When reading pref " << key
79                << ", got an empty value after trim";
80     return false;
81   }
82   if (!android::base::ParseInt<int64_t>(str_value, value)) {
83     LOG(ERROR) << "When reading pref " << key << ", failed to convert value "
84                << str_value << " to integer";
85     return false;
86   }
87   return true;
88 }
89 
SetInt64(std::string_view key,const int64_t value)90 bool PrefsBase::SetInt64(std::string_view key, const int64_t value) {
91   return SetString(key, std::format("{}", value));
92 }
93 
GetBoolean(std::string_view key,bool * value) const94 bool PrefsBase::GetBoolean(std::string_view key, bool* value) const {
95   string str_value;
96   if (!GetString(key, &str_value))
97     return false;
98   str_value = android::base::Trim(str_value);
99   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 
SetBoolean(std::string_view key,const bool value)110 bool PrefsBase::SetBoolean(std::string_view key, const bool value) {
111   return SetString(key, value ? "true" : "false");
112 }
113 
Exists(std::string_view key) const114 bool PrefsBase::Exists(std::string_view key) const {
115   return storage_->KeyExists(key);
116 }
117 
Delete(std::string_view key)118 bool PrefsBase::Delete(std::string_view key) {
119   TEST_AND_RETURN_FALSE(storage_->DeleteKey(key));
120   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 
Delete(std::string_view pref_key,const vector<string> & nss)129 bool PrefsBase::Delete(std::string_view pref_key, const vector<string>& nss) {
130   // 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 
GetSubKeys(std::string_view ns,vector<string> * keys) const147 bool PrefsBase::GetSubKeys(std::string_view ns, vector<string>* keys) const {
148   return storage_->GetSubKeys(ns, keys);
149 }
150 
AddObserver(std::string_view key,ObserverInterface * observer)151 void PrefsBase::AddObserver(std::string_view key, ObserverInterface* observer) {
152   observers_[std::string{key}].push_back(observer);
153 }
154 
RemoveObserver(std::string_view key,ObserverInterface * observer)155 void PrefsBase::RemoveObserver(std::string_view key,
156                                ObserverInterface* observer) {
157   std::vector<ObserverInterface*>& observers_for_key =
158       observers_[std::string{key}];
159   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);
163 }
164 
CreateSubKey(const vector<string> & ns_and_key)165 string PrefsInterface::CreateSubKey(const vector<string>& ns_and_key) {
166   return android::base::Join(ns_and_key, string(1, kKeySeparator));
167 }
168 
169 // Prefs
170 
Init(const base::FilePath & prefs_dir)171 bool Prefs::Init(const base::FilePath& prefs_dir) {
172   return file_storage_.Init(prefs_dir);
173 }
174 
StartTransaction()175 bool PrefsBase::StartTransaction() {
176   return storage_->CreateTemporaryPrefs();
177 }
178 
CancelTransaction()179 bool PrefsBase::CancelTransaction() {
180   return storage_->DeleteTemporaryPrefs();
181 }
182 
SubmitTransaction()183 bool PrefsBase::SubmitTransaction() {
184   return storage_->SwapPrefs();
185 }
186 
GetTemporaryDir() const187 std::string Prefs::FileStorage::GetTemporaryDir() const {
188   return prefs_dir_.value() + "_tmp";
189 }
190 
CreateTemporaryPrefs()191 bool 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.
203   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   }
209 
210   return true;
211 }
212 
DeleteTemporaryPrefs()213 bool Prefs::FileStorage::DeleteTemporaryPrefs() {
214   std::filesystem::path destination_directory(GetTemporaryDir());
215 
216   if (std::filesystem::exists(destination_directory)) {
217     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     }
223   }
224   return true;
225 }
226 
SwapPrefs()227 bool Prefs::FileStorage::SwapPrefs() {
228   if (!utils::DeleteDirectory(prefs_dir_.value().c_str())) {
229     LOG(ERROR) << "Failed to remove prefs dir " << prefs_dir_;
230     return false;
231   }
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   }
236   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   }
240   return true;
241 }
242 
Init(const base::FilePath & prefs_dir)243 bool Prefs::FileStorage::Init(const base::FilePath& prefs_dir) {
244   prefs_dir_ = prefs_dir;
245   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   }
252 
253   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 
262   // Delete empty directories. Ignore errors when deleting empty directories.
263   DeleteEmptyDirectories(prefs_dir_);
264   return true;
265 }
266 
GetKey(std::string_view key,string * value) const267 bool Prefs::FileStorage::GetKey(std::string_view key, string* value) const {
268   base::FilePath filename;
269   TEST_AND_RETURN_FALSE(GetFileNameForKey(key, &filename));
270   if (!base::ReadFileToString(filename, value)) {
271     return false;
272   }
273   return true;
274 }
275 
GetSubKeys(std::string_view ns,vector<string> * keys) const276 bool Prefs::FileStorage::GetSubKeys(std::string_view ns,
277                                     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 
SetKey(std::string_view key,std::string_view value)294 bool Prefs::FileStorage::SetKey(std::string_view key, std::string_view value) {
295   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   }
302   TEST_AND_RETURN_FALSE(
303       utils::WriteStringToFileAtomic(filename.value(), value));
304   return true;
305 }
306 
KeyExists(std::string_view key) const307 bool Prefs::FileStorage::KeyExists(std::string_view key) const {
308   base::FilePath filename;
309   TEST_AND_RETURN_FALSE(GetFileNameForKey(key, &filename));
310   return base::PathExists(filename);
311 }
312 
DeleteKey(std::string_view key)313 bool Prefs::FileStorage::DeleteKey(std::string_view key) {
314   base::FilePath filename;
315   TEST_AND_RETURN_FALSE(GetFileNameForKey(key, &filename));
316 #if BASE_VER < 800000
317   TEST_AND_RETURN_FALSE(base::DeleteFile(filename, false));
318 #else
319   TEST_AND_RETURN_FALSE(base::DeleteFile(filename));
320 #endif
321   return true;
322 }
323 
GetFileNameForKey(std::string_view key,base::FilePath * filename) const324 bool Prefs::FileStorage::GetFileNameForKey(std::string_view key,
325                                            base::FilePath* filename) const {
326   // Allows only non-empty keys containing [A-Za-z0-9_-/].
327   TEST_AND_RETURN_FALSE(!key.empty());
328   for (char c : key)
329     TEST_AND_RETURN_FALSE(isalpha(c) || isdigit(c) ||
330                           c == '_' || c == '-' || c == kKeySeparator);
331   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   }
339   return true;
340 }
341 
342 // MemoryPrefs
343 
GetKey(std::string_view key,string * value) const344 bool MemoryPrefs::MemoryStorage::GetKey(std::string_view key,
345                                         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 
GetSubKeys(std::string_view ns,vector<string> * keys) const353 bool MemoryPrefs::MemoryStorage::GetSubKeys(std::string_view ns,
354                                             vector<string>* keys) const {
355   auto lower_comp = [](const auto& pr, const auto& ns) {
356     return std::string_view{pr.first.data(), ns.length()} < ns;
357   };
358   auto upper_comp = [](const auto& ns, const auto& pr) {
359     return ns < std::string_view{pr.first.data(), ns.length()};
360   };
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 
SetKey(std::string_view key,std::string_view value)369 bool MemoryPrefs::MemoryStorage::SetKey(std::string_view key,
370                                         std::string_view value) {
371   values_[std::string{key}] = value;
372   return true;
373 }
374 
KeyExists(std::string_view key) const375 bool MemoryPrefs::MemoryStorage::KeyExists(std::string_view key) const {
376   return values_.find(key) != values_.end();
377 }
378 
DeleteKey(std::string_view key)379 bool MemoryPrefs::MemoryStorage::DeleteKey(std::string_view key) {
380   auto it = values_.find(key);
381   if (it != values_.end())
382     values_.erase(it);
383   return true;
384 }
385 
386 }  // namespace chromeos_update_engine
387