Home | History | Annotate | Download | only in common
      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_util.h>
     22 #include <base/logging.h>
     23 #include <base/strings/string_number_conversions.h>
     24 #include <base/strings/string_util.h>
     25 
     26 #include "update_engine/common/utils.h"
     27 
     28 using std::string;
     29 
     30 namespace chromeos_update_engine {
     31 
     32 bool PrefsBase::GetString(const string& key, string* value) const {
     33   return storage_->GetKey(key, value);
     34 }
     35 
     36 bool PrefsBase::SetString(const string& key, const string& value) {
     37   TEST_AND_RETURN_FALSE(storage_->SetKey(key, value));
     38   const auto observers_for_key = observers_.find(key);
     39   if (observers_for_key != observers_.end()) {
     40     std::vector<ObserverInterface*> copy_observers(observers_for_key->second);
     41     for (ObserverInterface* observer : copy_observers)
     42       observer->OnPrefSet(key);
     43   }
     44   return true;
     45 }
     46 
     47 bool PrefsBase::GetInt64(const string& key, int64_t* value) const {
     48   string str_value;
     49   if (!GetString(key, &str_value))
     50     return false;
     51   base::TrimWhitespaceASCII(str_value, base::TRIM_ALL, &str_value);
     52   TEST_AND_RETURN_FALSE(base::StringToInt64(str_value, value));
     53   return true;
     54 }
     55 
     56 bool PrefsBase::SetInt64(const string& key, const int64_t value) {
     57   return SetString(key, base::Int64ToString(value));
     58 }
     59 
     60 bool PrefsBase::GetBoolean(const string& key, bool* value) const {
     61   string str_value;
     62   if (!GetString(key, &str_value))
     63     return false;
     64   base::TrimWhitespaceASCII(str_value, base::TRIM_ALL, &str_value);
     65   if (str_value == "false") {
     66     *value = false;
     67     return true;
     68   }
     69   if (str_value == "true") {
     70     *value = true;
     71     return true;
     72   }
     73   return false;
     74 }
     75 
     76 bool PrefsBase::SetBoolean(const string& key, const bool value) {
     77   return SetString(key, value ? "true" : "false");
     78 }
     79 
     80 bool PrefsBase::Exists(const string& key) const {
     81   return storage_->KeyExists(key);
     82 }
     83 
     84 bool PrefsBase::Delete(const string& key) {
     85   TEST_AND_RETURN_FALSE(storage_->DeleteKey(key));
     86   const auto observers_for_key = observers_.find(key);
     87   if (observers_for_key != observers_.end()) {
     88     std::vector<ObserverInterface*> copy_observers(observers_for_key->second);
     89     for (ObserverInterface* observer : copy_observers)
     90       observer->OnPrefDeleted(key);
     91   }
     92   return true;
     93 }
     94 
     95 void PrefsBase::AddObserver(const string& key, ObserverInterface* observer) {
     96   observers_[key].push_back(observer);
     97 }
     98 
     99 void PrefsBase::RemoveObserver(const string& key, ObserverInterface* observer) {
    100   std::vector<ObserverInterface*>& observers_for_key = observers_[key];
    101   auto observer_it =
    102       std::find(observers_for_key.begin(), observers_for_key.end(), observer);
    103   if (observer_it != observers_for_key.end())
    104     observers_for_key.erase(observer_it);
    105 }
    106 
    107 // Prefs
    108 
    109 bool Prefs::Init(const base::FilePath& prefs_dir) {
    110   return file_storage_.Init(prefs_dir);
    111 }
    112 
    113 bool Prefs::FileStorage::Init(const base::FilePath& prefs_dir) {
    114   prefs_dir_ = prefs_dir;
    115   return true;
    116 }
    117 
    118 bool Prefs::FileStorage::GetKey(const string& key, string* value) const {
    119   base::FilePath filename;
    120   TEST_AND_RETURN_FALSE(GetFileNameForKey(key, &filename));
    121   if (!base::ReadFileToString(filename, value)) {
    122     LOG(INFO) << key << " not present in " << prefs_dir_.value();
    123     return false;
    124   }
    125   return true;
    126 }
    127 
    128 bool Prefs::FileStorage::SetKey(const string& key, const string& value) {
    129   base::FilePath filename;
    130   TEST_AND_RETURN_FALSE(GetFileNameForKey(key, &filename));
    131   if (!base::DirectoryExists(filename.DirName())) {
    132     // Only attempt to create the directory if it doesn't exist to avoid calls
    133     // to parent directories where we might not have permission to write to.
    134     TEST_AND_RETURN_FALSE(base::CreateDirectory(filename.DirName()));
    135   }
    136   TEST_AND_RETURN_FALSE(base::WriteFile(filename, value.data(), value.size()) ==
    137                         static_cast<int>(value.size()));
    138   return true;
    139 }
    140 
    141 bool Prefs::FileStorage::KeyExists(const string& key) const {
    142   base::FilePath filename;
    143   TEST_AND_RETURN_FALSE(GetFileNameForKey(key, &filename));
    144   return base::PathExists(filename);
    145 }
    146 
    147 bool Prefs::FileStorage::DeleteKey(const string& key) {
    148   base::FilePath filename;
    149   TEST_AND_RETURN_FALSE(GetFileNameForKey(key, &filename));
    150   TEST_AND_RETURN_FALSE(base::DeleteFile(filename, false));
    151   return true;
    152 }
    153 
    154 bool Prefs::FileStorage::GetFileNameForKey(const string& key,
    155                                            base::FilePath* filename) const {
    156   // Allows only non-empty keys containing [A-Za-z0-9_-].
    157   TEST_AND_RETURN_FALSE(!key.empty());
    158   for (size_t i = 0; i < key.size(); ++i) {
    159     char c = key.at(i);
    160     TEST_AND_RETURN_FALSE(base::IsAsciiAlpha(c) || base::IsAsciiDigit(c) ||
    161                           c == '_' || c == '-');
    162   }
    163   *filename = prefs_dir_.Append(key);
    164   return true;
    165 }
    166 
    167 // MemoryPrefs
    168 
    169 bool MemoryPrefs::MemoryStorage::GetKey(const string& key,
    170                                         string* value) const {
    171   auto it = values_.find(key);
    172   if (it == values_.end())
    173     return false;
    174   *value = it->second;
    175   return true;
    176 }
    177 
    178 bool MemoryPrefs::MemoryStorage::SetKey(const string& key,
    179                                         const string& value) {
    180   values_[key] = value;
    181   return true;
    182 }
    183 
    184 bool MemoryPrefs::MemoryStorage::KeyExists(const string& key) const {
    185   return values_.find(key) != values_.end();
    186 }
    187 
    188 bool MemoryPrefs::MemoryStorage::DeleteKey(const string& key) {
    189   auto it = values_.find(key);
    190   if (it == values_.end())
    191     return false;
    192   values_.erase(it);
    193   return true;
    194 }
    195 
    196 }  // namespace chromeos_update_engine
    197