xref: /aosp_15_r20/system/update_engine/common/prefs.cc (revision 5a9231315b4521097b8dc3750bc806fcafe0c72f)
1*5a923131SAndroid Build Coastguard Worker //
2*5a923131SAndroid Build Coastguard Worker // Copyright (C) 2012 The Android Open Source Project
3*5a923131SAndroid Build Coastguard Worker //
4*5a923131SAndroid Build Coastguard Worker // Licensed under the Apache License, Version 2.0 (the "License");
5*5a923131SAndroid Build Coastguard Worker // you may not use this file except in compliance with the License.
6*5a923131SAndroid Build Coastguard Worker // You may obtain a copy of the License at
7*5a923131SAndroid Build Coastguard Worker //
8*5a923131SAndroid Build Coastguard Worker //      http://www.apache.org/licenses/LICENSE-2.0
9*5a923131SAndroid Build Coastguard Worker //
10*5a923131SAndroid Build Coastguard Worker // Unless required by applicable law or agreed to in writing, software
11*5a923131SAndroid Build Coastguard Worker // distributed under the License is distributed on an "AS IS" BASIS,
12*5a923131SAndroid Build Coastguard Worker // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13*5a923131SAndroid Build Coastguard Worker // See the License for the specific language governing permissions and
14*5a923131SAndroid Build Coastguard Worker // limitations under the License.
15*5a923131SAndroid Build Coastguard Worker //
16*5a923131SAndroid Build Coastguard Worker 
17*5a923131SAndroid Build Coastguard Worker #include "update_engine/common/prefs.h"
18*5a923131SAndroid Build Coastguard Worker 
19*5a923131SAndroid Build Coastguard Worker #include <algorithm>
20*5a923131SAndroid Build Coastguard Worker #include <filesystem>
21*5a923131SAndroid Build Coastguard Worker #include <unistd.h>
22*5a923131SAndroid Build Coastguard Worker 
23*5a923131SAndroid Build Coastguard Worker #include <android-base/file.h>
24*5a923131SAndroid Build Coastguard Worker #include <android-base/parseint.h>
25*5a923131SAndroid Build Coastguard Worker #include <base/files/file_enumerator.h>
26*5a923131SAndroid Build Coastguard Worker #include <base/files/file_util.h>
27*5a923131SAndroid Build Coastguard Worker #include <base/logging.h>
28*5a923131SAndroid Build Coastguard Worker #include <base/strings/string_split.h>
29*5a923131SAndroid Build Coastguard Worker 
30*5a923131SAndroid Build Coastguard Worker #include "android-base/strings.h"
31*5a923131SAndroid Build Coastguard Worker #include "update_engine/common/utils.h"
32*5a923131SAndroid Build Coastguard Worker 
33*5a923131SAndroid Build Coastguard Worker using std::string;
34*5a923131SAndroid Build Coastguard Worker using std::vector;
35*5a923131SAndroid Build Coastguard Worker 
36*5a923131SAndroid Build Coastguard Worker namespace chromeos_update_engine {
37*5a923131SAndroid Build Coastguard Worker 
38*5a923131SAndroid Build Coastguard Worker namespace {
39*5a923131SAndroid Build Coastguard Worker 
DeleteEmptyDirectories(const base::FilePath & path)40*5a923131SAndroid Build Coastguard Worker void DeleteEmptyDirectories(const base::FilePath& path) {
41*5a923131SAndroid Build Coastguard Worker   base::FileEnumerator path_enum(
42*5a923131SAndroid Build Coastguard Worker       path, false /* recursive */, base::FileEnumerator::DIRECTORIES);
43*5a923131SAndroid Build Coastguard Worker   for (base::FilePath dir_path = path_enum.Next(); !dir_path.empty();
44*5a923131SAndroid Build Coastguard Worker        dir_path = path_enum.Next()) {
45*5a923131SAndroid Build Coastguard Worker     DeleteEmptyDirectories(dir_path);
46*5a923131SAndroid Build Coastguard Worker     if (base::IsDirectoryEmpty(dir_path))
47*5a923131SAndroid Build Coastguard Worker #if BASE_VER < 800000
48*5a923131SAndroid Build Coastguard Worker       base::DeleteFile(dir_path, false);
49*5a923131SAndroid Build Coastguard Worker #else
50*5a923131SAndroid Build Coastguard Worker       base::DeleteFile(dir_path);
51*5a923131SAndroid Build Coastguard Worker #endif
52*5a923131SAndroid Build Coastguard Worker   }
53*5a923131SAndroid Build Coastguard Worker }
54*5a923131SAndroid Build Coastguard Worker 
55*5a923131SAndroid Build Coastguard Worker }  // namespace
56*5a923131SAndroid Build Coastguard Worker 
GetString(const std::string_view key,string * value) const57*5a923131SAndroid Build Coastguard Worker bool PrefsBase::GetString(const std::string_view key, string* value) const {
58*5a923131SAndroid Build Coastguard Worker   return storage_->GetKey(key, value);
59*5a923131SAndroid Build Coastguard Worker }
60*5a923131SAndroid Build Coastguard Worker 
SetString(std::string_view key,std::string_view value)61*5a923131SAndroid Build Coastguard Worker bool PrefsBase::SetString(std::string_view key, std::string_view value) {
62*5a923131SAndroid Build Coastguard Worker   TEST_AND_RETURN_FALSE(storage_->SetKey(key, value));
63*5a923131SAndroid Build Coastguard Worker   const auto observers_for_key = observers_.find(key);
64*5a923131SAndroid Build Coastguard Worker   if (observers_for_key != observers_.end()) {
65*5a923131SAndroid Build Coastguard Worker     std::vector<ObserverInterface*> copy_observers(observers_for_key->second);
66*5a923131SAndroid Build Coastguard Worker     for (ObserverInterface* observer : copy_observers)
67*5a923131SAndroid Build Coastguard Worker       observer->OnPrefSet(key);
68*5a923131SAndroid Build Coastguard Worker   }
69*5a923131SAndroid Build Coastguard Worker   return true;
70*5a923131SAndroid Build Coastguard Worker }
71*5a923131SAndroid Build Coastguard Worker 
GetInt64(const std::string_view key,int64_t * value) const72*5a923131SAndroid Build Coastguard Worker bool PrefsBase::GetInt64(const std::string_view key, int64_t* value) const {
73*5a923131SAndroid Build Coastguard Worker   string str_value;
74*5a923131SAndroid Build Coastguard Worker   if (!GetString(key, &str_value))
75*5a923131SAndroid Build Coastguard Worker     return false;
76*5a923131SAndroid Build Coastguard Worker   str_value = android::base::Trim(str_value);
77*5a923131SAndroid Build Coastguard Worker   if (str_value.empty()) {
78*5a923131SAndroid Build Coastguard Worker     LOG(ERROR) << "When reading pref " << key
79*5a923131SAndroid Build Coastguard Worker                << ", got an empty value after trim";
80*5a923131SAndroid Build Coastguard Worker     return false;
81*5a923131SAndroid Build Coastguard Worker   }
82*5a923131SAndroid Build Coastguard Worker   if (!android::base::ParseInt<int64_t>(str_value, value)) {
83*5a923131SAndroid Build Coastguard Worker     LOG(ERROR) << "When reading pref " << key << ", failed to convert value "
84*5a923131SAndroid Build Coastguard Worker                << str_value << " to integer";
85*5a923131SAndroid Build Coastguard Worker     return false;
86*5a923131SAndroid Build Coastguard Worker   }
87*5a923131SAndroid Build Coastguard Worker   return true;
88*5a923131SAndroid Build Coastguard Worker }
89*5a923131SAndroid Build Coastguard Worker 
SetInt64(std::string_view key,const int64_t value)90*5a923131SAndroid Build Coastguard Worker bool PrefsBase::SetInt64(std::string_view key, const int64_t value) {
91*5a923131SAndroid Build Coastguard Worker   return SetString(key, std::format("{}", value));
92*5a923131SAndroid Build Coastguard Worker }
93*5a923131SAndroid Build Coastguard Worker 
GetBoolean(std::string_view key,bool * value) const94*5a923131SAndroid Build Coastguard Worker bool PrefsBase::GetBoolean(std::string_view key, bool* value) const {
95*5a923131SAndroid Build Coastguard Worker   string str_value;
96*5a923131SAndroid Build Coastguard Worker   if (!GetString(key, &str_value))
97*5a923131SAndroid Build Coastguard Worker     return false;
98*5a923131SAndroid Build Coastguard Worker   str_value = android::base::Trim(str_value);
99*5a923131SAndroid Build Coastguard Worker   if (str_value == "false") {
100*5a923131SAndroid Build Coastguard Worker     *value = false;
101*5a923131SAndroid Build Coastguard Worker     return true;
102*5a923131SAndroid Build Coastguard Worker   }
103*5a923131SAndroid Build Coastguard Worker   if (str_value == "true") {
104*5a923131SAndroid Build Coastguard Worker     *value = true;
105*5a923131SAndroid Build Coastguard Worker     return true;
106*5a923131SAndroid Build Coastguard Worker   }
107*5a923131SAndroid Build Coastguard Worker   return false;
108*5a923131SAndroid Build Coastguard Worker }
109*5a923131SAndroid Build Coastguard Worker 
SetBoolean(std::string_view key,const bool value)110*5a923131SAndroid Build Coastguard Worker bool PrefsBase::SetBoolean(std::string_view key, const bool value) {
111*5a923131SAndroid Build Coastguard Worker   return SetString(key, value ? "true" : "false");
112*5a923131SAndroid Build Coastguard Worker }
113*5a923131SAndroid Build Coastguard Worker 
Exists(std::string_view key) const114*5a923131SAndroid Build Coastguard Worker bool PrefsBase::Exists(std::string_view key) const {
115*5a923131SAndroid Build Coastguard Worker   return storage_->KeyExists(key);
116*5a923131SAndroid Build Coastguard Worker }
117*5a923131SAndroid Build Coastguard Worker 
Delete(std::string_view key)118*5a923131SAndroid Build Coastguard Worker bool PrefsBase::Delete(std::string_view key) {
119*5a923131SAndroid Build Coastguard Worker   TEST_AND_RETURN_FALSE(storage_->DeleteKey(key));
120*5a923131SAndroid Build Coastguard Worker   const auto observers_for_key = observers_.find(key);
121*5a923131SAndroid Build Coastguard Worker   if (observers_for_key != observers_.end()) {
122*5a923131SAndroid Build Coastguard Worker     std::vector<ObserverInterface*> copy_observers(observers_for_key->second);
123*5a923131SAndroid Build Coastguard Worker     for (ObserverInterface* observer : copy_observers)
124*5a923131SAndroid Build Coastguard Worker       observer->OnPrefDeleted(key);
125*5a923131SAndroid Build Coastguard Worker   }
126*5a923131SAndroid Build Coastguard Worker   return true;
127*5a923131SAndroid Build Coastguard Worker }
128*5a923131SAndroid Build Coastguard Worker 
Delete(std::string_view pref_key,const vector<string> & nss)129*5a923131SAndroid Build Coastguard Worker bool PrefsBase::Delete(std::string_view pref_key, const vector<string>& nss) {
130*5a923131SAndroid Build Coastguard Worker   // Delete pref key for platform.
131*5a923131SAndroid Build Coastguard Worker   bool success = Delete(pref_key);
132*5a923131SAndroid Build Coastguard Worker   // Delete pref key in each namespace.
133*5a923131SAndroid Build Coastguard Worker   for (const auto& ns : nss) {
134*5a923131SAndroid Build Coastguard Worker     vector<string> namespace_keys;
135*5a923131SAndroid Build Coastguard Worker     success = GetSubKeys(ns, &namespace_keys) && success;
136*5a923131SAndroid Build Coastguard Worker     for (const auto& key : namespace_keys) {
137*5a923131SAndroid Build Coastguard Worker       auto last_key_seperator = key.find_last_of(kKeySeparator);
138*5a923131SAndroid Build Coastguard Worker       if (last_key_seperator != string::npos &&
139*5a923131SAndroid Build Coastguard Worker           pref_key == key.substr(last_key_seperator + 1)) {
140*5a923131SAndroid Build Coastguard Worker         success = Delete(key) && success;
141*5a923131SAndroid Build Coastguard Worker       }
142*5a923131SAndroid Build Coastguard Worker     }
143*5a923131SAndroid Build Coastguard Worker   }
144*5a923131SAndroid Build Coastguard Worker   return success;
145*5a923131SAndroid Build Coastguard Worker }
146*5a923131SAndroid Build Coastguard Worker 
GetSubKeys(std::string_view ns,vector<string> * keys) const147*5a923131SAndroid Build Coastguard Worker bool PrefsBase::GetSubKeys(std::string_view ns, vector<string>* keys) const {
148*5a923131SAndroid Build Coastguard Worker   return storage_->GetSubKeys(ns, keys);
149*5a923131SAndroid Build Coastguard Worker }
150*5a923131SAndroid Build Coastguard Worker 
AddObserver(std::string_view key,ObserverInterface * observer)151*5a923131SAndroid Build Coastguard Worker void PrefsBase::AddObserver(std::string_view key, ObserverInterface* observer) {
152*5a923131SAndroid Build Coastguard Worker   observers_[std::string{key}].push_back(observer);
153*5a923131SAndroid Build Coastguard Worker }
154*5a923131SAndroid Build Coastguard Worker 
RemoveObserver(std::string_view key,ObserverInterface * observer)155*5a923131SAndroid Build Coastguard Worker void PrefsBase::RemoveObserver(std::string_view key,
156*5a923131SAndroid Build Coastguard Worker                                ObserverInterface* observer) {
157*5a923131SAndroid Build Coastguard Worker   std::vector<ObserverInterface*>& observers_for_key =
158*5a923131SAndroid Build Coastguard Worker       observers_[std::string{key}];
159*5a923131SAndroid Build Coastguard Worker   auto observer_it =
160*5a923131SAndroid Build Coastguard Worker       std::find(observers_for_key.begin(), observers_for_key.end(), observer);
161*5a923131SAndroid Build Coastguard Worker   if (observer_it != observers_for_key.end())
162*5a923131SAndroid Build Coastguard Worker     observers_for_key.erase(observer_it);
163*5a923131SAndroid Build Coastguard Worker }
164*5a923131SAndroid Build Coastguard Worker 
CreateSubKey(const vector<string> & ns_and_key)165*5a923131SAndroid Build Coastguard Worker string PrefsInterface::CreateSubKey(const vector<string>& ns_and_key) {
166*5a923131SAndroid Build Coastguard Worker   return android::base::Join(ns_and_key, string(1, kKeySeparator));
167*5a923131SAndroid Build Coastguard Worker }
168*5a923131SAndroid Build Coastguard Worker 
169*5a923131SAndroid Build Coastguard Worker // Prefs
170*5a923131SAndroid Build Coastguard Worker 
Init(const base::FilePath & prefs_dir)171*5a923131SAndroid Build Coastguard Worker bool Prefs::Init(const base::FilePath& prefs_dir) {
172*5a923131SAndroid Build Coastguard Worker   return file_storage_.Init(prefs_dir);
173*5a923131SAndroid Build Coastguard Worker }
174*5a923131SAndroid Build Coastguard Worker 
StartTransaction()175*5a923131SAndroid Build Coastguard Worker bool PrefsBase::StartTransaction() {
176*5a923131SAndroid Build Coastguard Worker   return storage_->CreateTemporaryPrefs();
177*5a923131SAndroid Build Coastguard Worker }
178*5a923131SAndroid Build Coastguard Worker 
CancelTransaction()179*5a923131SAndroid Build Coastguard Worker bool PrefsBase::CancelTransaction() {
180*5a923131SAndroid Build Coastguard Worker   return storage_->DeleteTemporaryPrefs();
181*5a923131SAndroid Build Coastguard Worker }
182*5a923131SAndroid Build Coastguard Worker 
SubmitTransaction()183*5a923131SAndroid Build Coastguard Worker bool PrefsBase::SubmitTransaction() {
184*5a923131SAndroid Build Coastguard Worker   return storage_->SwapPrefs();
185*5a923131SAndroid Build Coastguard Worker }
186*5a923131SAndroid Build Coastguard Worker 
GetTemporaryDir() const187*5a923131SAndroid Build Coastguard Worker std::string Prefs::FileStorage::GetTemporaryDir() const {
188*5a923131SAndroid Build Coastguard Worker   return prefs_dir_.value() + "_tmp";
189*5a923131SAndroid Build Coastguard Worker }
190*5a923131SAndroid Build Coastguard Worker 
CreateTemporaryPrefs()191*5a923131SAndroid Build Coastguard Worker bool Prefs::FileStorage::CreateTemporaryPrefs() {
192*5a923131SAndroid Build Coastguard Worker   // Delete any existing prefs_tmp
193*5a923131SAndroid Build Coastguard Worker   DeleteTemporaryPrefs();
194*5a923131SAndroid Build Coastguard Worker   // Get the paths to the source and destination directories.
195*5a923131SAndroid Build Coastguard Worker   std::filesystem::path source_directory(prefs_dir_.value());
196*5a923131SAndroid Build Coastguard Worker   std::filesystem::path destination_directory(GetTemporaryDir());
197*5a923131SAndroid Build Coastguard Worker 
198*5a923131SAndroid Build Coastguard Worker   if (!std::filesystem::exists(source_directory)) {
199*5a923131SAndroid Build Coastguard Worker     LOG(ERROR) << "prefs directory does not exist: " << source_directory;
200*5a923131SAndroid Build Coastguard Worker     return false;
201*5a923131SAndroid Build Coastguard Worker   }
202*5a923131SAndroid Build Coastguard Worker   // Copy the directory.
203*5a923131SAndroid Build Coastguard Worker   std::error_code e;
204*5a923131SAndroid Build Coastguard Worker   std::filesystem::copy(source_directory, destination_directory, e);
205*5a923131SAndroid Build Coastguard Worker   if (e) {
206*5a923131SAndroid Build Coastguard Worker     LOG(ERROR) << "failed to copy prefs to prefs_tmp: " << e.message();
207*5a923131SAndroid Build Coastguard Worker     return false;
208*5a923131SAndroid Build Coastguard Worker   }
209*5a923131SAndroid Build Coastguard Worker 
210*5a923131SAndroid Build Coastguard Worker   return true;
211*5a923131SAndroid Build Coastguard Worker }
212*5a923131SAndroid Build Coastguard Worker 
DeleteTemporaryPrefs()213*5a923131SAndroid Build Coastguard Worker bool Prefs::FileStorage::DeleteTemporaryPrefs() {
214*5a923131SAndroid Build Coastguard Worker   std::filesystem::path destination_directory(GetTemporaryDir());
215*5a923131SAndroid Build Coastguard Worker 
216*5a923131SAndroid Build Coastguard Worker   if (std::filesystem::exists(destination_directory)) {
217*5a923131SAndroid Build Coastguard Worker     std::error_code e;
218*5a923131SAndroid Build Coastguard Worker     std::filesystem::remove_all(destination_directory, e);
219*5a923131SAndroid Build Coastguard Worker     if (e) {
220*5a923131SAndroid Build Coastguard Worker       LOG(ERROR) << "failed to remove directory: " << e.message();
221*5a923131SAndroid Build Coastguard Worker       return false;
222*5a923131SAndroid Build Coastguard Worker     }
223*5a923131SAndroid Build Coastguard Worker   }
224*5a923131SAndroid Build Coastguard Worker   return true;
225*5a923131SAndroid Build Coastguard Worker }
226*5a923131SAndroid Build Coastguard Worker 
SwapPrefs()227*5a923131SAndroid Build Coastguard Worker bool Prefs::FileStorage::SwapPrefs() {
228*5a923131SAndroid Build Coastguard Worker   if (!utils::DeleteDirectory(prefs_dir_.value().c_str())) {
229*5a923131SAndroid Build Coastguard Worker     LOG(ERROR) << "Failed to remove prefs dir " << prefs_dir_;
230*5a923131SAndroid Build Coastguard Worker     return false;
231*5a923131SAndroid Build Coastguard Worker   }
232*5a923131SAndroid Build Coastguard Worker   if (rename(GetTemporaryDir().c_str(), prefs_dir_.value().c_str()) != 0) {
233*5a923131SAndroid Build Coastguard Worker     LOG(ERROR) << "Error replacing prefs with prefs_tmp" << strerror(errno);
234*5a923131SAndroid Build Coastguard Worker     return false;
235*5a923131SAndroid Build Coastguard Worker   }
236*5a923131SAndroid Build Coastguard Worker   if (!utils::FsyncDirectory(
237*5a923131SAndroid Build Coastguard Worker           android::base::Dirname(prefs_dir_.value()).c_str())) {
238*5a923131SAndroid Build Coastguard Worker     PLOG(ERROR) << "Failed to fsync prefs parent dir after swapping prefs";
239*5a923131SAndroid Build Coastguard Worker   }
240*5a923131SAndroid Build Coastguard Worker   return true;
241*5a923131SAndroid Build Coastguard Worker }
242*5a923131SAndroid Build Coastguard Worker 
Init(const base::FilePath & prefs_dir)243*5a923131SAndroid Build Coastguard Worker bool Prefs::FileStorage::Init(const base::FilePath& prefs_dir) {
244*5a923131SAndroid Build Coastguard Worker   prefs_dir_ = prefs_dir;
245*5a923131SAndroid Build Coastguard Worker   if (!std::filesystem::exists(prefs_dir_.value())) {
246*5a923131SAndroid Build Coastguard Worker     LOG(INFO) << "Prefs dir does not exist, possibly due to an interrupted "
247*5a923131SAndroid Build Coastguard Worker                  "transaction.";
248*5a923131SAndroid Build Coastguard Worker     if (std::filesystem::exists(GetTemporaryDir())) {
249*5a923131SAndroid Build Coastguard Worker       SwapPrefs();
250*5a923131SAndroid Build Coastguard Worker     }
251*5a923131SAndroid Build Coastguard Worker   }
252*5a923131SAndroid Build Coastguard Worker 
253*5a923131SAndroid Build Coastguard Worker   if (std::filesystem::exists(GetTemporaryDir())) {
254*5a923131SAndroid Build Coastguard Worker     LOG(INFO)
255*5a923131SAndroid Build Coastguard Worker         << "Deleting temporary prefs, checkpoint transaction was interrupted";
256*5a923131SAndroid Build Coastguard Worker     if (!utils::DeleteDirectory(GetTemporaryDir().c_str())) {
257*5a923131SAndroid Build Coastguard Worker       LOG(ERROR) << "Failed to delete temporary prefs";
258*5a923131SAndroid Build Coastguard Worker       return false;
259*5a923131SAndroid Build Coastguard Worker     }
260*5a923131SAndroid Build Coastguard Worker   }
261*5a923131SAndroid Build Coastguard Worker 
262*5a923131SAndroid Build Coastguard Worker   // Delete empty directories. Ignore errors when deleting empty directories.
263*5a923131SAndroid Build Coastguard Worker   DeleteEmptyDirectories(prefs_dir_);
264*5a923131SAndroid Build Coastguard Worker   return true;
265*5a923131SAndroid Build Coastguard Worker }
266*5a923131SAndroid Build Coastguard Worker 
GetKey(std::string_view key,string * value) const267*5a923131SAndroid Build Coastguard Worker bool Prefs::FileStorage::GetKey(std::string_view key, string* value) const {
268*5a923131SAndroid Build Coastguard Worker   base::FilePath filename;
269*5a923131SAndroid Build Coastguard Worker   TEST_AND_RETURN_FALSE(GetFileNameForKey(key, &filename));
270*5a923131SAndroid Build Coastguard Worker   if (!base::ReadFileToString(filename, value)) {
271*5a923131SAndroid Build Coastguard Worker     return false;
272*5a923131SAndroid Build Coastguard Worker   }
273*5a923131SAndroid Build Coastguard Worker   return true;
274*5a923131SAndroid Build Coastguard Worker }
275*5a923131SAndroid Build Coastguard Worker 
GetSubKeys(std::string_view ns,vector<string> * keys) const276*5a923131SAndroid Build Coastguard Worker bool Prefs::FileStorage::GetSubKeys(std::string_view ns,
277*5a923131SAndroid Build Coastguard Worker                                     vector<string>* keys) const {
278*5a923131SAndroid Build Coastguard Worker   base::FilePath filename;
279*5a923131SAndroid Build Coastguard Worker   TEST_AND_RETURN_FALSE(GetFileNameForKey(ns, &filename));
280*5a923131SAndroid Build Coastguard Worker   base::FileEnumerator namespace_enum(
281*5a923131SAndroid Build Coastguard Worker       prefs_dir_, true, base::FileEnumerator::FILES);
282*5a923131SAndroid Build Coastguard Worker   for (base::FilePath f = namespace_enum.Next(); !f.empty();
283*5a923131SAndroid Build Coastguard Worker        f = namespace_enum.Next()) {
284*5a923131SAndroid Build Coastguard Worker     auto filename_str = filename.value();
285*5a923131SAndroid Build Coastguard Worker     if (f.value().compare(0, filename_str.length(), filename_str) == 0) {
286*5a923131SAndroid Build Coastguard Worker       // Only return the key portion excluding the |prefs_dir_| with slash.
287*5a923131SAndroid Build Coastguard Worker       keys->push_back(f.value().substr(
288*5a923131SAndroid Build Coastguard Worker           prefs_dir_.AsEndingWithSeparator().value().length()));
289*5a923131SAndroid Build Coastguard Worker     }
290*5a923131SAndroid Build Coastguard Worker   }
291*5a923131SAndroid Build Coastguard Worker   return true;
292*5a923131SAndroid Build Coastguard Worker }
293*5a923131SAndroid Build Coastguard Worker 
SetKey(std::string_view key,std::string_view value)294*5a923131SAndroid Build Coastguard Worker bool Prefs::FileStorage::SetKey(std::string_view key, std::string_view value) {
295*5a923131SAndroid Build Coastguard Worker   base::FilePath filename;
296*5a923131SAndroid Build Coastguard Worker   TEST_AND_RETURN_FALSE(GetFileNameForKey(key, &filename));
297*5a923131SAndroid Build Coastguard Worker   if (!base::DirectoryExists(filename.DirName())) {
298*5a923131SAndroid Build Coastguard Worker     // Only attempt to create the directory if it doesn't exist to avoid calls
299*5a923131SAndroid Build Coastguard Worker     // to parent directories where we might not have permission to write to.
300*5a923131SAndroid Build Coastguard Worker     TEST_AND_RETURN_FALSE(base::CreateDirectory(filename.DirName()));
301*5a923131SAndroid Build Coastguard Worker   }
302*5a923131SAndroid Build Coastguard Worker   TEST_AND_RETURN_FALSE(
303*5a923131SAndroid Build Coastguard Worker       utils::WriteStringToFileAtomic(filename.value(), value));
304*5a923131SAndroid Build Coastguard Worker   return true;
305*5a923131SAndroid Build Coastguard Worker }
306*5a923131SAndroid Build Coastguard Worker 
KeyExists(std::string_view key) const307*5a923131SAndroid Build Coastguard Worker bool Prefs::FileStorage::KeyExists(std::string_view key) const {
308*5a923131SAndroid Build Coastguard Worker   base::FilePath filename;
309*5a923131SAndroid Build Coastguard Worker   TEST_AND_RETURN_FALSE(GetFileNameForKey(key, &filename));
310*5a923131SAndroid Build Coastguard Worker   return base::PathExists(filename);
311*5a923131SAndroid Build Coastguard Worker }
312*5a923131SAndroid Build Coastguard Worker 
DeleteKey(std::string_view key)313*5a923131SAndroid Build Coastguard Worker bool Prefs::FileStorage::DeleteKey(std::string_view key) {
314*5a923131SAndroid Build Coastguard Worker   base::FilePath filename;
315*5a923131SAndroid Build Coastguard Worker   TEST_AND_RETURN_FALSE(GetFileNameForKey(key, &filename));
316*5a923131SAndroid Build Coastguard Worker #if BASE_VER < 800000
317*5a923131SAndroid Build Coastguard Worker   TEST_AND_RETURN_FALSE(base::DeleteFile(filename, false));
318*5a923131SAndroid Build Coastguard Worker #else
319*5a923131SAndroid Build Coastguard Worker   TEST_AND_RETURN_FALSE(base::DeleteFile(filename));
320*5a923131SAndroid Build Coastguard Worker #endif
321*5a923131SAndroid Build Coastguard Worker   return true;
322*5a923131SAndroid Build Coastguard Worker }
323*5a923131SAndroid Build Coastguard Worker 
GetFileNameForKey(std::string_view key,base::FilePath * filename) const324*5a923131SAndroid Build Coastguard Worker bool Prefs::FileStorage::GetFileNameForKey(std::string_view key,
325*5a923131SAndroid Build Coastguard Worker                                            base::FilePath* filename) const {
326*5a923131SAndroid Build Coastguard Worker   // Allows only non-empty keys containing [A-Za-z0-9_-/].
327*5a923131SAndroid Build Coastguard Worker   TEST_AND_RETURN_FALSE(!key.empty());
328*5a923131SAndroid Build Coastguard Worker   for (char c : key)
329*5a923131SAndroid Build Coastguard Worker     TEST_AND_RETURN_FALSE(isalpha(c) || isdigit(c) ||
330*5a923131SAndroid Build Coastguard Worker                           c == '_' || c == '-' || c == kKeySeparator);
331*5a923131SAndroid Build Coastguard Worker   if (std::filesystem::exists(GetTemporaryDir())) {
332*5a923131SAndroid Build Coastguard Worker     *filename =
333*5a923131SAndroid Build Coastguard Worker         base::FilePath(GetTemporaryDir())
334*5a923131SAndroid Build Coastguard Worker             .Append(base::FilePath::StringPieceType(key.data(), key.size()));
335*5a923131SAndroid Build Coastguard Worker   } else {
336*5a923131SAndroid Build Coastguard Worker     *filename = prefs_dir_.Append(
337*5a923131SAndroid Build Coastguard Worker         base::FilePath::StringPieceType(key.data(), key.size()));
338*5a923131SAndroid Build Coastguard Worker   }
339*5a923131SAndroid Build Coastguard Worker   return true;
340*5a923131SAndroid Build Coastguard Worker }
341*5a923131SAndroid Build Coastguard Worker 
342*5a923131SAndroid Build Coastguard Worker // MemoryPrefs
343*5a923131SAndroid Build Coastguard Worker 
GetKey(std::string_view key,string * value) const344*5a923131SAndroid Build Coastguard Worker bool MemoryPrefs::MemoryStorage::GetKey(std::string_view key,
345*5a923131SAndroid Build Coastguard Worker                                         string* value) const {
346*5a923131SAndroid Build Coastguard Worker   auto it = values_.find(key);
347*5a923131SAndroid Build Coastguard Worker   if (it == values_.end())
348*5a923131SAndroid Build Coastguard Worker     return false;
349*5a923131SAndroid Build Coastguard Worker   *value = it->second;
350*5a923131SAndroid Build Coastguard Worker   return true;
351*5a923131SAndroid Build Coastguard Worker }
352*5a923131SAndroid Build Coastguard Worker 
GetSubKeys(std::string_view ns,vector<string> * keys) const353*5a923131SAndroid Build Coastguard Worker bool MemoryPrefs::MemoryStorage::GetSubKeys(std::string_view ns,
354*5a923131SAndroid Build Coastguard Worker                                             vector<string>* keys) const {
355*5a923131SAndroid Build Coastguard Worker   auto lower_comp = [](const auto& pr, const auto& ns) {
356*5a923131SAndroid Build Coastguard Worker     return std::string_view{pr.first.data(), ns.length()} < ns;
357*5a923131SAndroid Build Coastguard Worker   };
358*5a923131SAndroid Build Coastguard Worker   auto upper_comp = [](const auto& ns, const auto& pr) {
359*5a923131SAndroid Build Coastguard Worker     return ns < std::string_view{pr.first.data(), ns.length()};
360*5a923131SAndroid Build Coastguard Worker   };
361*5a923131SAndroid Build Coastguard Worker   auto lower_it =
362*5a923131SAndroid Build Coastguard Worker       std::lower_bound(begin(values_), end(values_), ns, lower_comp);
363*5a923131SAndroid Build Coastguard Worker   auto upper_it = std::upper_bound(lower_it, end(values_), ns, upper_comp);
364*5a923131SAndroid Build Coastguard Worker   while (lower_it != upper_it)
365*5a923131SAndroid Build Coastguard Worker     keys->push_back((lower_it++)->first);
366*5a923131SAndroid Build Coastguard Worker   return true;
367*5a923131SAndroid Build Coastguard Worker }
368*5a923131SAndroid Build Coastguard Worker 
SetKey(std::string_view key,std::string_view value)369*5a923131SAndroid Build Coastguard Worker bool MemoryPrefs::MemoryStorage::SetKey(std::string_view key,
370*5a923131SAndroid Build Coastguard Worker                                         std::string_view value) {
371*5a923131SAndroid Build Coastguard Worker   values_[std::string{key}] = value;
372*5a923131SAndroid Build Coastguard Worker   return true;
373*5a923131SAndroid Build Coastguard Worker }
374*5a923131SAndroid Build Coastguard Worker 
KeyExists(std::string_view key) const375*5a923131SAndroid Build Coastguard Worker bool MemoryPrefs::MemoryStorage::KeyExists(std::string_view key) const {
376*5a923131SAndroid Build Coastguard Worker   return values_.find(key) != values_.end();
377*5a923131SAndroid Build Coastguard Worker }
378*5a923131SAndroid Build Coastguard Worker 
DeleteKey(std::string_view key)379*5a923131SAndroid Build Coastguard Worker bool MemoryPrefs::MemoryStorage::DeleteKey(std::string_view key) {
380*5a923131SAndroid Build Coastguard Worker   auto it = values_.find(key);
381*5a923131SAndroid Build Coastguard Worker   if (it != values_.end())
382*5a923131SAndroid Build Coastguard Worker     values_.erase(it);
383*5a923131SAndroid Build Coastguard Worker   return true;
384*5a923131SAndroid Build Coastguard Worker }
385*5a923131SAndroid Build Coastguard Worker 
386*5a923131SAndroid Build Coastguard Worker }  // namespace chromeos_update_engine
387