• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
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 
21 #include <base/files/file_enumerator.h>
22 #include <base/files/file_util.h>
23 #include <base/logging.h>
24 #include <base/strings/string_number_conversions.h>
25 #include <base/strings/string_split.h>
26 #include <base/strings/string_util.h>
27 
28 #include "update_engine/common/utils.h"
29 
30 using std::string;
31 using std::vector;
32 
33 namespace chromeos_update_engine {
34 
35 namespace {
36 
DeleteEmptyDirectories(const base::FilePath & path)37 void DeleteEmptyDirectories(const base::FilePath& path) {
38   base::FileEnumerator path_enum(
39       path, false /* recursive */, base::FileEnumerator::DIRECTORIES);
40   for (base::FilePath dir_path = path_enum.Next(); !dir_path.empty();
41        dir_path = path_enum.Next()) {
42     DeleteEmptyDirectories(dir_path);
43     if (base::IsDirectoryEmpty(dir_path))
44 #if BASE_VER < 800000
45       base::DeleteFile(dir_path, false);
46 #else
47       base::DeleteFile(dir_path);
48 #endif
49   }
50 }
51 
52 }  // namespace
53 
GetString(const std::string_view key,string * value) const54 bool PrefsBase::GetString(const std::string_view key, string* value) const {
55   return storage_->GetKey(key, value);
56 }
57 
SetString(std::string_view key,std::string_view value)58 bool PrefsBase::SetString(std::string_view key, std::string_view value) {
59   TEST_AND_RETURN_FALSE(storage_->SetKey(key, value));
60   const auto observers_for_key = observers_.find(key);
61   if (observers_for_key != observers_.end()) {
62     std::vector<ObserverInterface*> copy_observers(observers_for_key->second);
63     for (ObserverInterface* observer : copy_observers)
64       observer->OnPrefSet(key);
65   }
66   return true;
67 }
68 
GetInt64(const std::string_view key,int64_t * value) const69 bool PrefsBase::GetInt64(const std::string_view key, int64_t* value) const {
70   string str_value;
71   if (!GetString(key, &str_value))
72     return false;
73   base::TrimWhitespaceASCII(str_value, base::TRIM_ALL, &str_value);
74   TEST_AND_RETURN_FALSE(base::StringToInt64(str_value, value));
75   return true;
76 }
77 
SetInt64(std::string_view key,const int64_t value)78 bool PrefsBase::SetInt64(std::string_view key, const int64_t value) {
79   return SetString(key, base::NumberToString(value));
80 }
81 
GetBoolean(std::string_view key,bool * value) const82 bool PrefsBase::GetBoolean(std::string_view key, bool* value) const {
83   string str_value;
84   if (!GetString(key, &str_value))
85     return false;
86   base::TrimWhitespaceASCII(str_value, base::TRIM_ALL, &str_value);
87   if (str_value == "false") {
88     *value = false;
89     return true;
90   }
91   if (str_value == "true") {
92     *value = true;
93     return true;
94   }
95   return false;
96 }
97 
SetBoolean(std::string_view key,const bool value)98 bool PrefsBase::SetBoolean(std::string_view key, const bool value) {
99   return SetString(key, value ? "true" : "false");
100 }
101 
Exists(std::string_view key) const102 bool PrefsBase::Exists(std::string_view key) const {
103   return storage_->KeyExists(key);
104 }
105 
Delete(std::string_view key)106 bool PrefsBase::Delete(std::string_view key) {
107   TEST_AND_RETURN_FALSE(storage_->DeleteKey(key));
108   const auto observers_for_key = observers_.find(key);
109   if (observers_for_key != observers_.end()) {
110     std::vector<ObserverInterface*> copy_observers(observers_for_key->second);
111     for (ObserverInterface* observer : copy_observers)
112       observer->OnPrefDeleted(key);
113   }
114   return true;
115 }
116 
Delete(std::string_view pref_key,const vector<string> & nss)117 bool PrefsBase::Delete(std::string_view pref_key, const vector<string>& nss) {
118   // Delete pref key for platform.
119   bool success = Delete(pref_key);
120   // Delete pref key in each namespace.
121   for (const auto& ns : nss) {
122     vector<string> namespace_keys;
123     success = GetSubKeys(ns, &namespace_keys) && success;
124     for (const auto& key : namespace_keys) {
125       auto last_key_seperator = key.find_last_of(kKeySeparator);
126       if (last_key_seperator != string::npos &&
127           pref_key == key.substr(last_key_seperator + 1)) {
128         success = Delete(key) && success;
129       }
130     }
131   }
132   return success;
133 }
134 
GetSubKeys(std::string_view ns,vector<string> * keys) const135 bool PrefsBase::GetSubKeys(std::string_view ns, vector<string>* keys) const {
136   return storage_->GetSubKeys(ns, keys);
137 }
138 
AddObserver(std::string_view key,ObserverInterface * observer)139 void PrefsBase::AddObserver(std::string_view key, ObserverInterface* observer) {
140   observers_[std::string{key}].push_back(observer);
141 }
142 
RemoveObserver(std::string_view key,ObserverInterface * observer)143 void PrefsBase::RemoveObserver(std::string_view key,
144                                ObserverInterface* observer) {
145   std::vector<ObserverInterface*>& observers_for_key =
146       observers_[std::string{key}];
147   auto observer_it =
148       std::find(observers_for_key.begin(), observers_for_key.end(), observer);
149   if (observer_it != observers_for_key.end())
150     observers_for_key.erase(observer_it);
151 }
152 
CreateSubKey(const vector<string> & ns_and_key)153 string PrefsInterface::CreateSubKey(const vector<string>& ns_and_key) {
154   return base::JoinString(ns_and_key, string(1, kKeySeparator));
155 }
156 
157 // Prefs
158 
Init(const base::FilePath & prefs_dir)159 bool Prefs::Init(const base::FilePath& prefs_dir) {
160   return file_storage_.Init(prefs_dir);
161 }
162 
Init(const base::FilePath & prefs_dir)163 bool Prefs::FileStorage::Init(const base::FilePath& prefs_dir) {
164   prefs_dir_ = prefs_dir;
165   // Delete empty directories. Ignore errors when deleting empty directories.
166   DeleteEmptyDirectories(prefs_dir_);
167   return true;
168 }
169 
GetKey(std::string_view key,string * value) const170 bool Prefs::FileStorage::GetKey(std::string_view key, string* value) const {
171   base::FilePath filename;
172   TEST_AND_RETURN_FALSE(GetFileNameForKey(key, &filename));
173   if (!base::ReadFileToString(filename, value)) {
174     return false;
175   }
176   return true;
177 }
178 
GetSubKeys(std::string_view ns,vector<string> * keys) const179 bool Prefs::FileStorage::GetSubKeys(std::string_view ns,
180                                     vector<string>* keys) const {
181   base::FilePath filename;
182   TEST_AND_RETURN_FALSE(GetFileNameForKey(ns, &filename));
183   base::FileEnumerator namespace_enum(
184       prefs_dir_, true, base::FileEnumerator::FILES);
185   for (base::FilePath f = namespace_enum.Next(); !f.empty();
186        f = namespace_enum.Next()) {
187     auto filename_str = filename.value();
188     if (f.value().compare(0, filename_str.length(), filename_str) == 0) {
189       // Only return the key portion excluding the |prefs_dir_| with slash.
190       keys->push_back(f.value().substr(
191           prefs_dir_.AsEndingWithSeparator().value().length()));
192     }
193   }
194   return true;
195 }
196 
SetKey(std::string_view key,std::string_view value)197 bool Prefs::FileStorage::SetKey(std::string_view key, std::string_view value) {
198   base::FilePath filename;
199   TEST_AND_RETURN_FALSE(GetFileNameForKey(key, &filename));
200   if (!base::DirectoryExists(filename.DirName())) {
201     // Only attempt to create the directory if it doesn't exist to avoid calls
202     // to parent directories where we might not have permission to write to.
203     TEST_AND_RETURN_FALSE(base::CreateDirectory(filename.DirName()));
204   }
205   TEST_AND_RETURN_FALSE(
206       utils::WriteStringToFileAtomic(filename.value(), value));
207   return true;
208 }
209 
KeyExists(std::string_view key) const210 bool Prefs::FileStorage::KeyExists(std::string_view key) const {
211   base::FilePath filename;
212   TEST_AND_RETURN_FALSE(GetFileNameForKey(key, &filename));
213   return base::PathExists(filename);
214 }
215 
DeleteKey(std::string_view key)216 bool Prefs::FileStorage::DeleteKey(std::string_view key) {
217   base::FilePath filename;
218   TEST_AND_RETURN_FALSE(GetFileNameForKey(key, &filename));
219 #if BASE_VER < 800000
220   TEST_AND_RETURN_FALSE(base::DeleteFile(filename, false));
221 #else
222   TEST_AND_RETURN_FALSE(base::DeleteFile(filename));
223 #endif
224   return true;
225 }
226 
GetFileNameForKey(std::string_view key,base::FilePath * filename) const227 bool Prefs::FileStorage::GetFileNameForKey(std::string_view key,
228                                            base::FilePath* filename) const {
229   // Allows only non-empty keys containing [A-Za-z0-9_-/].
230   TEST_AND_RETURN_FALSE(!key.empty());
231   for (char c : key)
232     TEST_AND_RETURN_FALSE(base::IsAsciiAlpha(c) || base::IsAsciiDigit(c) ||
233                           c == '_' || c == '-' || c == kKeySeparator);
234   *filename = prefs_dir_.Append(
235       base::FilePath::StringPieceType(key.data(), key.size()));
236   return true;
237 }
238 
239 // MemoryPrefs
240 
GetKey(std::string_view key,string * value) const241 bool MemoryPrefs::MemoryStorage::GetKey(std::string_view key,
242                                         string* value) const {
243   auto it = values_.find(key);
244   if (it == values_.end())
245     return false;
246   *value = it->second;
247   return true;
248 }
249 
GetSubKeys(std::string_view ns,vector<string> * keys) const250 bool MemoryPrefs::MemoryStorage::GetSubKeys(std::string_view ns,
251                                             vector<string>* keys) const {
252   auto lower_comp = [](const auto& pr, const auto& ns) {
253     return std::string_view{pr.first.data(), ns.length()} < ns;
254   };
255   auto upper_comp = [](const auto& ns, const auto& pr) {
256     return ns < std::string_view{pr.first.data(), ns.length()};
257   };
258   auto lower_it =
259       std::lower_bound(begin(values_), end(values_), ns, lower_comp);
260   auto upper_it = std::upper_bound(lower_it, end(values_), ns, upper_comp);
261   while (lower_it != upper_it)
262     keys->push_back((lower_it++)->first);
263   return true;
264 }
265 
SetKey(std::string_view key,std::string_view value)266 bool MemoryPrefs::MemoryStorage::SetKey(std::string_view key,
267                                         std::string_view value) {
268   values_[std::string{key}] = value;
269   return true;
270 }
271 
KeyExists(std::string_view key) const272 bool MemoryPrefs::MemoryStorage::KeyExists(std::string_view key) const {
273   return values_.find(key) != values_.end();
274 }
275 
DeleteKey(std::string_view key)276 bool MemoryPrefs::MemoryStorage::DeleteKey(std::string_view key) {
277   auto it = values_.find(key);
278   if (it != values_.end())
279     values_.erase(it);
280   return true;
281 }
282 
283 }  // namespace chromeos_update_engine
284