Vitaly Buka | 4615e0d | 2015-10-14 15:35:12 -0700 | [diff] [blame] | 1 | // Copyright 2015 The Weave Authors. All rights reserved. |
Vitaly Buka | d322a3c | 2015-08-16 01:03:02 -0700 | [diff] [blame] | 2 | // Use of this source code is governed by a BSD-style license that can be |
| 3 | // found in the LICENSE file. |
| 4 | |
Stefan Sauer | 2d16dfa | 2015-09-25 17:08:35 +0200 | [diff] [blame] | 5 | #include "src/config.h" |
Vitaly Buka | d322a3c | 2015-08-16 01:03:02 -0700 | [diff] [blame] | 6 | |
| 7 | #include <set> |
| 8 | |
Vitaly Buka | c11a17d | 2015-08-15 10:36:10 -0700 | [diff] [blame] | 9 | #include <base/bind.h> |
Johan Euphrosine | 0b7bb9f | 2015-09-29 01:11:21 -0700 | [diff] [blame] | 10 | #include <base/guid.h> |
Vitaly Buka | c11a17d | 2015-08-15 10:36:10 -0700 | [diff] [blame] | 11 | #include <base/json/json_reader.h> |
| 12 | #include <base/json/json_writer.h> |
Vitaly Buka | d322a3c | 2015-08-16 01:03:02 -0700 | [diff] [blame] | 13 | #include <base/logging.h> |
| 14 | #include <base/strings/string_number_conversions.h> |
Vitaly Buka | c11a17d | 2015-08-15 10:36:10 -0700 | [diff] [blame] | 15 | #include <base/values.h> |
Vitaly Buka | d322a3c | 2015-08-16 01:03:02 -0700 | [diff] [blame] | 16 | #include <weave/enum_to_string.h> |
| 17 | |
Vitaly Buka | f08caeb | 2015-12-02 13:47:48 -0800 | [diff] [blame] | 18 | #include "src/data_encoding.h" |
Stefan Sauer | 2d16dfa | 2015-09-25 17:08:35 +0200 | [diff] [blame] | 19 | #include "src/privet/privet_types.h" |
| 20 | #include "src/string_utils.h" |
Vitaly Buka | ac18fcf | 2016-01-15 14:48:54 -0800 | [diff] [blame^] | 21 | #include "src/bind_lambda.h" |
Vitaly Buka | d322a3c | 2015-08-16 01:03:02 -0700 | [diff] [blame] | 22 | |
Vitaly Buka | d322a3c | 2015-08-16 01:03:02 -0700 | [diff] [blame] | 23 | namespace weave { |
| 24 | |
Vitaly Buka | d1e6c4f | 2016-01-15 12:19:17 -0800 | [diff] [blame] | 25 | const char kConfigName[] = "config"; |
| 26 | |
Vitaly Buka | d322a3c | 2015-08-16 01:03:02 -0700 | [diff] [blame] | 27 | namespace config_keys { |
| 28 | |
Vitaly Buka | bced5af | 2015-10-12 17:42:30 -0700 | [diff] [blame] | 29 | const char kVersion[] = "version"; |
| 30 | |
Vitaly Buka | d322a3c | 2015-08-16 01:03:02 -0700 | [diff] [blame] | 31 | const char kClientId[] = "client_id"; |
| 32 | const char kClientSecret[] = "client_secret"; |
| 33 | const char kApiKey[] = "api_key"; |
| 34 | const char kOAuthURL[] = "oauth_url"; |
| 35 | const char kServiceURL[] = "service_url"; |
| 36 | const char kName[] = "name"; |
| 37 | const char kDescription[] = "description"; |
| 38 | const char kLocation[] = "location"; |
| 39 | const char kLocalAnonymousAccessRole[] = "local_anonymous_access_role"; |
| 40 | const char kLocalDiscoveryEnabled[] = "local_discovery_enabled"; |
| 41 | const char kLocalPairingEnabled[] = "local_pairing_enabled"; |
Vitaly Buka | d322a3c | 2015-08-16 01:03:02 -0700 | [diff] [blame] | 42 | const char kRefreshToken[] = "refresh_token"; |
Johan Euphrosine | 312c2f5 | 2015-09-29 00:04:29 -0700 | [diff] [blame] | 43 | const char kCloudId[] = "cloud_id"; |
Johan Euphrosine | 0b7bb9f | 2015-09-29 01:11:21 -0700 | [diff] [blame] | 44 | const char kDeviceId[] = "device_id"; |
Vitaly Buka | d322a3c | 2015-08-16 01:03:02 -0700 | [diff] [blame] | 45 | const char kRobotAccount[] = "robot_account"; |
Vitaly Buka | d322a3c | 2015-08-16 01:03:02 -0700 | [diff] [blame] | 46 | const char kLastConfiguredSsid[] = "last_configured_ssid"; |
Vitaly Buka | 8589b05 | 2015-09-29 00:46:14 -0700 | [diff] [blame] | 47 | const char kSecret[] = "secret"; |
Vitaly Buka | 63c9862 | 2015-12-11 11:06:04 -0800 | [diff] [blame] | 48 | const char kRootClientTokenOwner[] = "root_client_token_owner"; |
Vitaly Buka | d322a3c | 2015-08-16 01:03:02 -0700 | [diff] [blame] | 49 | |
| 50 | } // namespace config_keys |
| 51 | |
Vitaly Buka | 0d7aac8 | 2015-11-16 23:02:24 -0800 | [diff] [blame] | 52 | const char kWeaveUrl[] = "https://www.googleapis.com/weave/v1/"; |
| 53 | const char kDeprecatedUrl[] = "https://www.googleapis.com/clouddevices/v1/"; |
| 54 | |
Vitaly Buka | c11a17d | 2015-08-15 10:36:10 -0700 | [diff] [blame] | 55 | namespace { |
| 56 | |
Vitaly Buka | bced5af | 2015-10-12 17:42:30 -0700 | [diff] [blame] | 57 | const int kCurrentConfigVersion = 1; |
| 58 | |
| 59 | void MigrateFromV0(base::DictionaryValue* dict) { |
| 60 | std::string cloud_id; |
| 61 | if (dict->GetString(config_keys::kCloudId, &cloud_id) && !cloud_id.empty()) |
| 62 | return; |
| 63 | scoped_ptr<base::Value> tmp; |
| 64 | if (dict->Remove(config_keys::kDeviceId, &tmp)) |
| 65 | dict->Set(config_keys::kCloudId, std::move(tmp)); |
| 66 | } |
| 67 | |
Vitaly Buka | 238db69 | 2015-09-29 16:58:39 -0700 | [diff] [blame] | 68 | Config::Settings CreateDefaultSettings() { |
| 69 | Config::Settings result; |
Vitaly Buka | d322a3c | 2015-08-16 01:03:02 -0700 | [diff] [blame] | 70 | result.oauth_url = "https://accounts.google.com/o/oauth2/"; |
Vitaly Buka | 0d7aac8 | 2015-11-16 23:02:24 -0800 | [diff] [blame] | 71 | result.service_url = kWeaveUrl; |
Vitaly Buka | b624bc4 | 2015-09-29 19:13:55 -0700 | [diff] [blame] | 72 | result.local_anonymous_access_role = AuthScope::kViewer; |
Vitaly Buka | 52d006a | 2015-11-21 17:14:51 -0800 | [diff] [blame] | 73 | result.pairing_modes.insert(PairingType::kPinCode); |
Johan Euphrosine | 0b7bb9f | 2015-09-29 01:11:21 -0700 | [diff] [blame] | 74 | result.device_id = base::GenerateGUID(); |
Vitaly Buka | d322a3c | 2015-08-16 01:03:02 -0700 | [diff] [blame] | 75 | return result; |
| 76 | } |
| 77 | |
Vitaly Buka | 63c9862 | 2015-12-11 11:06:04 -0800 | [diff] [blame] | 78 | const EnumToStringMap<RootClientTokenOwner>::Map kRootClientTokenOwnerMap[] = { |
| 79 | {RootClientTokenOwner::kNone, "none"}, |
| 80 | {RootClientTokenOwner::kClient, "client"}, |
| 81 | {RootClientTokenOwner::kCloud, "cloud"}, |
| 82 | }; |
| 83 | |
Vitaly Buka | c11a17d | 2015-08-15 10:36:10 -0700 | [diff] [blame] | 84 | } // namespace |
Vitaly Buka | d322a3c | 2015-08-16 01:03:02 -0700 | [diff] [blame] | 85 | |
Vitaly Buka | 63c9862 | 2015-12-11 11:06:04 -0800 | [diff] [blame] | 86 | template <> |
| 87 | LIBWEAVE_EXPORT EnumToStringMap<RootClientTokenOwner>::EnumToStringMap() |
| 88 | : EnumToStringMap(kRootClientTokenOwnerMap) {} |
| 89 | |
Vitaly Buka | 1e36367 | 2015-09-25 14:01:16 -0700 | [diff] [blame] | 90 | Config::Config(provider::ConfigStore* config_store) |
Vitaly Buka | 3d6b552 | 2015-12-22 13:14:16 -0800 | [diff] [blame] | 91 | : settings_{CreateDefaultSettings()}, config_store_{config_store} { |
| 92 | Load(); |
| 93 | } |
Vitaly Buka | d322a3c | 2015-08-16 01:03:02 -0700 | [diff] [blame] | 94 | |
| 95 | void Config::AddOnChangedCallback(const OnChangedCallback& callback) { |
| 96 | on_changed_.push_back(callback); |
| 97 | // Force to read current state. |
| 98 | callback.Run(settings_); |
| 99 | } |
| 100 | |
Vitaly Buka | 238db69 | 2015-09-29 16:58:39 -0700 | [diff] [blame] | 101 | const Config::Settings& Config::GetSettings() const { |
Vitaly Buka | d322a3c | 2015-08-16 01:03:02 -0700 | [diff] [blame] | 102 | return settings_; |
| 103 | } |
| 104 | |
Vitaly Buka | c11a17d | 2015-08-15 10:36:10 -0700 | [diff] [blame] | 105 | void Config::Load() { |
Vitaly Buka | d322a3c | 2015-08-16 01:03:02 -0700 | [diff] [blame] | 106 | Transaction change{this}; |
| 107 | change.save_ = false; |
| 108 | |
Vitaly Buka | c11a17d | 2015-08-15 10:36:10 -0700 | [diff] [blame] | 109 | settings_ = CreateDefaultSettings(); |
| 110 | |
| 111 | if (!config_store_) |
| 112 | return; |
| 113 | |
| 114 | // Crash on any mistakes in defaults. |
| 115 | CHECK(config_store_->LoadDefaults(&settings_)); |
| 116 | |
Vitaly Buka | d322a3c | 2015-08-16 01:03:02 -0700 | [diff] [blame] | 117 | CHECK(!settings_.client_id.empty()); |
Vitaly Buka | d322a3c | 2015-08-16 01:03:02 -0700 | [diff] [blame] | 118 | CHECK(!settings_.client_secret.empty()); |
Vitaly Buka | d322a3c | 2015-08-16 01:03:02 -0700 | [diff] [blame] | 119 | CHECK(!settings_.api_key.empty()); |
Vitaly Buka | d322a3c | 2015-08-16 01:03:02 -0700 | [diff] [blame] | 120 | CHECK(!settings_.oauth_url.empty()); |
Vitaly Buka | d322a3c | 2015-08-16 01:03:02 -0700 | [diff] [blame] | 121 | CHECK(!settings_.service_url.empty()); |
Vitaly Buka | d322a3c | 2015-08-16 01:03:02 -0700 | [diff] [blame] | 122 | CHECK(!settings_.oem_name.empty()); |
Vitaly Buka | d322a3c | 2015-08-16 01:03:02 -0700 | [diff] [blame] | 123 | CHECK(!settings_.model_name.empty()); |
Vitaly Buka | d322a3c | 2015-08-16 01:03:02 -0700 | [diff] [blame] | 124 | CHECK(!settings_.model_id.empty()); |
Vitaly Buka | d322a3c | 2015-08-16 01:03:02 -0700 | [diff] [blame] | 125 | CHECK(!settings_.name.empty()); |
Johan Euphrosine | 0b7bb9f | 2015-09-29 01:11:21 -0700 | [diff] [blame] | 126 | CHECK(!settings_.device_id.empty()); |
Alex Vakulenko | 674f0eb | 2016-01-20 08:10:48 -0800 | [diff] [blame] | 127 | CHECK_EQ(settings_.embedded_code.empty(), |
| 128 | (settings_.pairing_modes.find(PairingType::kEmbeddedCode) == |
| 129 | settings_.pairing_modes.end())); |
Vitaly Buka | d322a3c | 2015-08-16 01:03:02 -0700 | [diff] [blame] | 130 | |
Vitaly Buka | 8589b05 | 2015-09-29 00:46:14 -0700 | [diff] [blame] | 131 | // Values below will be generated at runtime. |
| 132 | CHECK(settings_.cloud_id.empty()); |
| 133 | CHECK(settings_.refresh_token.empty()); |
| 134 | CHECK(settings_.robot_account.empty()); |
| 135 | CHECK(settings_.last_configured_ssid.empty()); |
| 136 | CHECK(settings_.secret.empty()); |
Vitaly Buka | 63c9862 | 2015-12-11 11:06:04 -0800 | [diff] [blame] | 137 | CHECK(settings_.root_client_token_owner == RootClientTokenOwner::kNone); |
Vitaly Buka | 8589b05 | 2015-09-29 00:46:14 -0700 | [diff] [blame] | 138 | |
Vitaly Buka | d322a3c | 2015-08-16 01:03:02 -0700 | [diff] [blame] | 139 | change.LoadState(); |
| 140 | } |
| 141 | |
| 142 | void Config::Transaction::LoadState() { |
Vitaly Buka | c11a17d | 2015-08-15 10:36:10 -0700 | [diff] [blame] | 143 | if (!config_->config_store_) |
Vitaly Buka | d322a3c | 2015-08-16 01:03:02 -0700 | [diff] [blame] | 144 | return; |
Vitaly Buka | d1e6c4f | 2016-01-15 12:19:17 -0800 | [diff] [blame] | 145 | std::string json_string = config_->config_store_->LoadSettings(kConfigName); |
| 146 | if (json_string.empty()) { |
| 147 | json_string = config_->config_store_->LoadSettings(); |
| 148 | if (json_string.empty()) |
| 149 | return; |
| 150 | } |
Vitaly Buka | c11a17d | 2015-08-15 10:36:10 -0700 | [diff] [blame] | 151 | |
| 152 | auto value = base::JSONReader::Read(json_string); |
Vitaly Buka | bced5af | 2015-10-12 17:42:30 -0700 | [diff] [blame] | 153 | base::DictionaryValue* dict = nullptr; |
Vitaly Buka | c11a17d | 2015-08-15 10:36:10 -0700 | [diff] [blame] | 154 | if (!value || !value->GetAsDictionary(&dict)) { |
| 155 | LOG(ERROR) << "Failed to parse settings."; |
Vitaly Buka | d322a3c | 2015-08-16 01:03:02 -0700 | [diff] [blame] | 156 | return; |
Vitaly Buka | c11a17d | 2015-08-15 10:36:10 -0700 | [diff] [blame] | 157 | } |
Vitaly Buka | d322a3c | 2015-08-16 01:03:02 -0700 | [diff] [blame] | 158 | |
Vitaly Buka | bced5af | 2015-10-12 17:42:30 -0700 | [diff] [blame] | 159 | int loaded_version = 0; |
| 160 | dict->GetInteger(config_keys::kVersion, &loaded_version); |
| 161 | |
| 162 | if (loaded_version != kCurrentConfigVersion) { |
| 163 | LOG(INFO) << "State version mismatch. expected: " << kCurrentConfigVersion |
| 164 | << ", loaded: " << loaded_version; |
| 165 | save_ = true; |
| 166 | } |
| 167 | |
| 168 | if (loaded_version == 0) { |
| 169 | MigrateFromV0(dict); |
| 170 | } |
| 171 | |
Vitaly Buka | d322a3c | 2015-08-16 01:03:02 -0700 | [diff] [blame] | 172 | std::string tmp; |
| 173 | bool tmp_bool{false}; |
| 174 | |
| 175 | if (dict->GetString(config_keys::kClientId, &tmp)) |
| 176 | set_client_id(tmp); |
| 177 | |
| 178 | if (dict->GetString(config_keys::kClientSecret, &tmp)) |
| 179 | set_client_secret(tmp); |
| 180 | |
| 181 | if (dict->GetString(config_keys::kApiKey, &tmp)) |
| 182 | set_api_key(tmp); |
| 183 | |
| 184 | if (dict->GetString(config_keys::kOAuthURL, &tmp)) |
| 185 | set_oauth_url(tmp); |
| 186 | |
Vitaly Buka | 0d7aac8 | 2015-11-16 23:02:24 -0800 | [diff] [blame] | 187 | if (dict->GetString(config_keys::kServiceURL, &tmp)) { |
| 188 | if (tmp == kDeprecatedUrl) |
| 189 | tmp = kWeaveUrl; |
Vitaly Buka | d322a3c | 2015-08-16 01:03:02 -0700 | [diff] [blame] | 190 | set_service_url(tmp); |
Vitaly Buka | 0d7aac8 | 2015-11-16 23:02:24 -0800 | [diff] [blame] | 191 | } |
Vitaly Buka | d322a3c | 2015-08-16 01:03:02 -0700 | [diff] [blame] | 192 | |
| 193 | if (dict->GetString(config_keys::kName, &tmp)) |
| 194 | set_name(tmp); |
| 195 | |
| 196 | if (dict->GetString(config_keys::kDescription, &tmp)) |
| 197 | set_description(tmp); |
| 198 | |
| 199 | if (dict->GetString(config_keys::kLocation, &tmp)) |
| 200 | set_location(tmp); |
| 201 | |
Vitaly Buka | b624bc4 | 2015-09-29 19:13:55 -0700 | [diff] [blame] | 202 | AuthScope scope{AuthScope::kNone}; |
| 203 | if (dict->GetString(config_keys::kLocalAnonymousAccessRole, &tmp) && |
| 204 | StringToEnum(tmp, &scope)) { |
| 205 | set_local_anonymous_access_role(scope); |
| 206 | } |
Vitaly Buka | d322a3c | 2015-08-16 01:03:02 -0700 | [diff] [blame] | 207 | |
| 208 | if (dict->GetBoolean(config_keys::kLocalDiscoveryEnabled, &tmp_bool)) |
| 209 | set_local_discovery_enabled(tmp_bool); |
| 210 | |
| 211 | if (dict->GetBoolean(config_keys::kLocalPairingEnabled, &tmp_bool)) |
| 212 | set_local_pairing_enabled(tmp_bool); |
| 213 | |
Vitaly Buka | 8589b05 | 2015-09-29 00:46:14 -0700 | [diff] [blame] | 214 | if (dict->GetString(config_keys::kCloudId, &tmp)) |
| 215 | set_cloud_id(tmp); |
| 216 | |
Johan Euphrosine | 0b7bb9f | 2015-09-29 01:11:21 -0700 | [diff] [blame] | 217 | if (dict->GetString(config_keys::kDeviceId, &tmp)) |
| 218 | set_device_id(tmp); |
| 219 | |
Vitaly Buka | d322a3c | 2015-08-16 01:03:02 -0700 | [diff] [blame] | 220 | if (dict->GetString(config_keys::kRefreshToken, &tmp)) |
| 221 | set_refresh_token(tmp); |
| 222 | |
| 223 | if (dict->GetString(config_keys::kRobotAccount, &tmp)) |
| 224 | set_robot_account(tmp); |
| 225 | |
| 226 | if (dict->GetString(config_keys::kLastConfiguredSsid, &tmp)) |
| 227 | set_last_configured_ssid(tmp); |
| 228 | |
Vitaly Buka | f08caeb | 2015-12-02 13:47:48 -0800 | [diff] [blame] | 229 | std::vector<uint8_t> secret; |
| 230 | if (dict->GetString(config_keys::kSecret, &tmp) && Base64Decode(tmp, &secret)) |
| 231 | set_secret(secret); |
Vitaly Buka | a580328 | 2015-12-06 20:11:55 -0800 | [diff] [blame] | 232 | |
Vitaly Buka | 63c9862 | 2015-12-11 11:06:04 -0800 | [diff] [blame] | 233 | RootClientTokenOwner token_owner{RootClientTokenOwner::kNone}; |
| 234 | if (dict->GetString(config_keys::kRootClientTokenOwner, &tmp) && |
| 235 | StringToEnum(tmp, &token_owner)) { |
| 236 | set_root_client_token_owner(token_owner); |
| 237 | } |
Vitaly Buka | d322a3c | 2015-08-16 01:03:02 -0700 | [diff] [blame] | 238 | } |
| 239 | |
Vitaly Buka | c11a17d | 2015-08-15 10:36:10 -0700 | [diff] [blame] | 240 | void Config::Save() { |
| 241 | if (!config_store_) |
| 242 | return; |
| 243 | |
Vitaly Buka | d322a3c | 2015-08-16 01:03:02 -0700 | [diff] [blame] | 244 | base::DictionaryValue dict; |
Vitaly Buka | bced5af | 2015-10-12 17:42:30 -0700 | [diff] [blame] | 245 | dict.SetInteger(config_keys::kVersion, kCurrentConfigVersion); |
| 246 | |
Vitaly Buka | d322a3c | 2015-08-16 01:03:02 -0700 | [diff] [blame] | 247 | dict.SetString(config_keys::kClientId, settings_.client_id); |
| 248 | dict.SetString(config_keys::kClientSecret, settings_.client_secret); |
| 249 | dict.SetString(config_keys::kApiKey, settings_.api_key); |
| 250 | dict.SetString(config_keys::kOAuthURL, settings_.oauth_url); |
| 251 | dict.SetString(config_keys::kServiceURL, settings_.service_url); |
| 252 | dict.SetString(config_keys::kRefreshToken, settings_.refresh_token); |
Johan Euphrosine | 312c2f5 | 2015-09-29 00:04:29 -0700 | [diff] [blame] | 253 | dict.SetString(config_keys::kCloudId, settings_.cloud_id); |
Johan Euphrosine | 0b7bb9f | 2015-09-29 01:11:21 -0700 | [diff] [blame] | 254 | dict.SetString(config_keys::kDeviceId, settings_.device_id); |
Vitaly Buka | d322a3c | 2015-08-16 01:03:02 -0700 | [diff] [blame] | 255 | dict.SetString(config_keys::kRobotAccount, settings_.robot_account); |
| 256 | dict.SetString(config_keys::kLastConfiguredSsid, |
| 257 | settings_.last_configured_ssid); |
Vitaly Buka | f08caeb | 2015-12-02 13:47:48 -0800 | [diff] [blame] | 258 | dict.SetString(config_keys::kSecret, Base64Encode(settings_.secret)); |
Vitaly Buka | 63c9862 | 2015-12-11 11:06:04 -0800 | [diff] [blame] | 259 | dict.SetString(config_keys::kRootClientTokenOwner, |
| 260 | EnumToString(settings_.root_client_token_owner)); |
Vitaly Buka | d322a3c | 2015-08-16 01:03:02 -0700 | [diff] [blame] | 261 | dict.SetString(config_keys::kName, settings_.name); |
| 262 | dict.SetString(config_keys::kDescription, settings_.description); |
| 263 | dict.SetString(config_keys::kLocation, settings_.location); |
| 264 | dict.SetString(config_keys::kLocalAnonymousAccessRole, |
Vitaly Buka | b624bc4 | 2015-09-29 19:13:55 -0700 | [diff] [blame] | 265 | EnumToString(settings_.local_anonymous_access_role)); |
Vitaly Buka | d322a3c | 2015-08-16 01:03:02 -0700 | [diff] [blame] | 266 | dict.SetBoolean(config_keys::kLocalDiscoveryEnabled, |
| 267 | settings_.local_discovery_enabled); |
| 268 | dict.SetBoolean(config_keys::kLocalPairingEnabled, |
| 269 | settings_.local_pairing_enabled); |
| 270 | |
Vitaly Buka | c11a17d | 2015-08-15 10:36:10 -0700 | [diff] [blame] | 271 | std::string json_string; |
| 272 | base::JSONWriter::WriteWithOptions( |
| 273 | dict, base::JSONWriter::OPTIONS_PRETTY_PRINT, &json_string); |
| 274 | |
Vitaly Buka | ac18fcf | 2016-01-15 14:48:54 -0800 | [diff] [blame^] | 275 | config_store_->SaveSettings( |
| 276 | kConfigName, json_string, |
| 277 | base::Bind([](ErrorPtr error) { CHECK(!error); })); |
Vitaly Buka | d322a3c | 2015-08-16 01:03:02 -0700 | [diff] [blame] | 278 | } |
| 279 | |
| 280 | Config::Transaction::~Transaction() { |
| 281 | Commit(); |
| 282 | } |
| 283 | |
Vitaly Buka | d322a3c | 2015-08-16 01:03:02 -0700 | [diff] [blame] | 284 | void Config::Transaction::Commit() { |
| 285 | if (!config_) |
| 286 | return; |
| 287 | if (save_) |
| 288 | config_->Save(); |
| 289 | for (const auto& cb : config_->on_changed_) |
| 290 | cb.Run(*settings_); |
| 291 | config_ = nullptr; |
| 292 | } |
| 293 | |
| 294 | } // namespace weave |