// Copyright 2015 The Android Open Source Project // // Licensed under the Apache License, Version 2.0 (the "License"); // you may not use this file except in compliance with the License. // You may obtain a copy of the License at // // http://www.apache.org/licenses/LICENSE-2.0 // // Unless required by applicable law or agreed to in writing, software // distributed under the License is distributed on an "AS IS" BASIS, // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. // See the License for the specific language governing permissions and // limitations under the License. #include "buffet/buffet_config.h" #include <map> #include <set> #include <base/files/file_util.h> #include <base/files/important_file_writer.h> #include <base/logging.h> #include <base/message_loop/message_loop.h> #include <base/strings/string_number_conversions.h> #include <brillo/errors/error.h> #include <brillo/errors/error_codes.h> #include <brillo/osrelease_reader.h> #include <brillo/strings/string_utils.h> #include <weave/enum_to_string.h> namespace buffet { namespace { const char kErrorDomain[] = "buffet"; const char kFileReadError[] = "file_read_error"; const char kProductVersionKey[] = "product_version"; class DefaultFileIO : public BuffetConfig::FileIO { public: bool ReadFile(const base::FilePath& path, std::string* content) override { return base::ReadFileToString(path, content); } bool WriteFile(const base::FilePath& path, const std::string& content) override { return base::ImportantFileWriter::WriteFileAtomically(path, content); } }; } // namespace namespace config_keys { const char kClientId[] = "client_id"; const char kClientSecret[] = "client_secret"; const char kApiKey[] = "api_key"; const char kOAuthURL[] = "oauth_url"; const char kServiceURL[] = "service_url"; const char kName[] = "name"; const char kDescription[] = "description"; const char kLocation[] = "location"; const char kLocalAnonymousAccessRole[] = "local_anonymous_access_role"; const char kLocalDiscoveryEnabled[] = "local_discovery_enabled"; const char kLocalPairingEnabled[] = "local_pairing_enabled"; const char kOemName[] = "oem_name"; const char kModelName[] = "model_name"; const char kModelId[] = "model_id"; const char kWifiAutoSetupEnabled[] = "wifi_auto_setup_enabled"; const char kEmbeddedCode[] = "embedded_code"; const char kPairingModes[] = "pairing_modes"; } // namespace config_keys BuffetConfig::BuffetConfig(const Options& options) : options_(options), default_encryptor_(Encryptor::CreateDefaultEncryptor()), encryptor_(default_encryptor_.get()), default_file_io_(new DefaultFileIO), file_io_(default_file_io_.get()) {} bool BuffetConfig::LoadDefaults(weave::Settings* settings) { // Keep this hardcoded default for sometime. This previously was set by // libweave. It should be set by overlay's buffet.conf. // Keys owners: avakulenko, gene, vitalybuka. settings->client_id = "338428340000-vkb4p6h40c7kja1k3l70kke8t615cjit.apps.googleusercontent." "com"; settings->client_secret = "LS_iPYo_WIOE0m2VnLdduhnx"; settings->api_key = "AIzaSyACK3oZtmIylUKXiTMqkZqfuRiCgQmQSAQ"; settings->name = "Developer device"; settings->oem_name = "Chromium"; settings->model_name = "Brillo"; settings->model_id = "AAAAA"; if (!base::PathExists(options_.defaults)) return true; // Nothing to load. brillo::KeyValueStore store; if (!store.Load(options_.defaults)) return false; bool result = LoadDefaults(store, settings); settings->test_privet_ssid = options_.test_privet_ssid; if (!options_.client_id.empty()) settings->client_id = options_.client_id; if (!options_.client_secret.empty()) settings->client_secret = options_.client_secret; if (!options_.api_key.empty()) settings->api_key = options_.api_key; if (!options_.oauth_url.empty()) settings->oauth_url = options_.oauth_url; if (!options_.service_url.empty()) settings->service_url = options_.service_url; return result; } bool BuffetConfig::LoadDefaults(const brillo::KeyValueStore& store, weave::Settings* settings) { store.GetString(config_keys::kClientId, &settings->client_id); store.GetString(config_keys::kClientSecret, &settings->client_secret); store.GetString(config_keys::kApiKey, &settings->api_key); store.GetString(config_keys::kOAuthURL, &settings->oauth_url); store.GetString(config_keys::kServiceURL, &settings->service_url); store.GetString(config_keys::kOemName, &settings->oem_name); store.GetString(config_keys::kModelName, &settings->model_name); store.GetString(config_keys::kModelId, &settings->model_id); brillo::OsReleaseReader reader; reader.Load(); if (!reader.GetString(kProductVersionKey, &settings->firmware_version)) { LOG(ERROR) << "Could not read '" << kProductVersionKey << "' from OS"; } store.GetBoolean(config_keys::kWifiAutoSetupEnabled, &settings->wifi_auto_setup_enabled); store.GetString(config_keys::kEmbeddedCode, &settings->embedded_code); std::string modes_str; if (store.GetString(config_keys::kPairingModes, &modes_str)) { std::set<weave::PairingType> pairing_modes; for (const std::string& mode : brillo::string_utils::Split(modes_str, ",", true, true)) { weave::PairingType pairing_mode; if (!StringToEnum(mode, &pairing_mode)) return false; pairing_modes.insert(pairing_mode); } settings->pairing_modes = std::move(pairing_modes); } store.GetString(config_keys::kName, &settings->name); store.GetString(config_keys::kDescription, &settings->description); store.GetString(config_keys::kLocation, &settings->location); std::string role_str; if (store.GetString(config_keys::kLocalAnonymousAccessRole, &role_str)) { if (!StringToEnum(role_str, &settings->local_anonymous_access_role)) return false; } store.GetBoolean(config_keys::kLocalDiscoveryEnabled, &settings->local_discovery_enabled); store.GetBoolean(config_keys::kLocalPairingEnabled, &settings->local_pairing_enabled); return true; } std::string BuffetConfig::LoadSettings(const std::string& name) { std::string settings_blob; base::FilePath path = CreatePath(name); if (!file_io_->ReadFile(path, &settings_blob)) { LOG(WARNING) << "Failed to read \'" + path.value() + "\', proceeding with empty settings."; return std::string(); } std::string json_string; if (!encryptor_->DecryptWithAuthentication(settings_blob, &json_string)) { LOG(WARNING) << "Failed to decrypt settings, proceeding with empty settings."; SaveSettings(std::string(), name, {}); return std::string(); } return json_string; } std::string BuffetConfig::LoadSettings() { return LoadSettings(""); } void BuffetConfig::SaveSettings(const std::string& name, const std::string& settings, const weave::DoneCallback& callback) { std::string encrypted_settings; weave::ErrorPtr error; base::FilePath path = CreatePath(name); if (!encryptor_->EncryptWithAuthentication(settings, &encrypted_settings)) { weave::Error::AddTo(&error, FROM_HERE, "file_write_error", "Failed to encrypt settings."); encrypted_settings.clear(); } if (!file_io_->WriteFile(path, encrypted_settings)) { weave::Error::AddTo(&error, FROM_HERE, "file_write_error", "Failed to write \'" + path.value() + "\', proceeding with empty settings."); } if (!callback.is_null()) { base::MessageLoop::current()->PostTask( FROM_HERE, base::Bind(callback, base::Passed(&error))); } } base::FilePath BuffetConfig::CreatePath(const std::string& name) const { return name.empty() ? options_.settings : options_.settings.InsertBeforeExtension( base::FilePath::kExtensionSeparator + name); } bool BuffetConfig::LoadFile(const base::FilePath& file_path, std::string* data, brillo::ErrorPtr* error) { if (!file_io_->ReadFile(file_path, data)) { brillo::errors::system::AddSystemError(error, FROM_HERE, errno); brillo::Error::AddToPrintf(error, FROM_HERE, kErrorDomain, kFileReadError, "Failed to read file '%s'", file_path.value().c_str()); return false; } return true; } } // namespace buffet