Chromium Code Reviews| OLD | NEW |
|---|---|
| 1 // Copyright 2014 The Chromium Authors. All rights reserved. | 1 // Copyright 2014 The Chromium Authors. All rights reserved. |
| 2 // Use of this source code is governed by a BSD-style license that can be | 2 // Use of this source code is governed by a BSD-style license that can be |
| 3 // found in the LICENSE file. | 3 // found in the LICENSE file. |
| 4 | 4 |
| 5 #include "google_apis/gcm/engine/gservices_settings.h" | 5 #include "google_apis/gcm/engine/gservices_settings.h" |
| 6 | 6 |
| 7 #include <algorithm> | |
| 8 | |
| 7 #include "base/bind.h" | 9 #include "base/bind.h" |
| 10 #include "base/sha1.h" | |
| 8 #include "base/strings/string_number_conversions.h" | 11 #include "base/strings/string_number_conversions.h" |
| 9 #include "base/strings/stringprintf.h" | 12 #include "base/strings/stringprintf.h" |
| 10 | 13 |
| 11 namespace { | 14 namespace { |
| 12 // The expected time in seconds between periodic checkins. | 15 // The expected time in seconds between periodic checkins. |
| 13 const char kCheckinIntervalKey[] = "checkin_interval"; | 16 const char kCheckinIntervalKey[] = "checkin_interval"; |
| 14 // The override URL to the checkin server. | 17 // The override URL to the checkin server. |
| 15 const char kCheckinURLKey[] = "checkin_url"; | 18 const char kCheckinURLKey[] = "checkin_url"; |
| 16 // The MCS machine name to connect to. | 19 // The MCS machine name to connect to. |
| 17 const char kMCSHostnameKey[] = "gcm_hostname"; | 20 const char kMCSHostnameKey[] = "gcm_hostname"; |
| 18 // The MCS port to connect to. | 21 // The MCS port to connect to. |
| 19 const char kMCSSecurePortKey[] = "gcm_secure_port"; | 22 const char kMCSSecurePortKey[] = "gcm_secure_port"; |
| 20 // The URL to get MCS registration IDs. | 23 // The URL to get MCS registration IDs. |
| 21 const char kRegistrationURLKey[] = "gcm_registration_url"; | 24 const char kRegistrationURLKey[] = "gcm_registration_url"; |
| 22 | 25 |
| 23 const int64 kDefaultCheckinInterval = 2 * 24 * 60 * 60; // seconds = 2 days. | 26 const int64 kDefaultCheckinInterval = 2 * 24 * 60 * 60; // seconds = 2 days. |
| 24 const int64 kMinimumCheckinInterval = 12 * 60 * 60; // seconds = 12 hours. | 27 const int64 kMinimumCheckinInterval = 12 * 60 * 60; // seconds = 12 hours. |
| 25 const char kDefaultCheckinURL[] = "https://android.clients.google.com/checkin"; | 28 const char kDefaultCheckinURL[] = "https://android.clients.google.com/checkin"; |
| 26 const char kDefaultMCSHostname[] = "mtalk.google.com"; | 29 const char kDefaultMCSHostname[] = "mtalk.google.com"; |
| 27 const int kDefaultMCSMainSecurePort = 5228; | 30 const int kDefaultMCSMainSecurePort = 5228; |
| 28 const int kDefaultMCSFallbackSecurePort = 443; | 31 const int kDefaultMCSFallbackSecurePort = 443; |
| 29 const char kDefaultRegistrationURL[] = | 32 const char kDefaultRegistrationURL[] = |
| 30 "https://android.clients.google.com/c2dm/register3"; | 33 "https://android.clients.google.com/c2dm/register3"; |
| 34 // Settings that are to be deleted are marked with this prefix in checkin | |
| 35 // response. | |
| 36 const char kDeleteSettingPrefix[] = "delete_"; | |
| 37 // Settings digest starts with verison number followed by '-'. | |
| 38 const char kDigestVersionPrefix[] = "1-"; | |
| 31 const char kMCSEnpointTemplate[] = "https://%s:%d"; | 39 const char kMCSEnpointTemplate[] = "https://%s:%d"; |
| 40 const int kMaxSecurePort = 65535; | |
| 32 | 41 |
| 33 std::string MakeMCSEndpoint(const std::string& mcs_hostname, int port) { | 42 std::string MakeMCSEndpoint(const std::string& mcs_hostname, int port) { |
| 34 return base::StringPrintf(kMCSEnpointTemplate, mcs_hostname.c_str(), port); | 43 return base::StringPrintf(kMCSEnpointTemplate, mcs_hostname.c_str(), port); |
| 35 } | 44 } |
| 36 | 45 |
| 46 bool VerifyCheckinInterval( | |
| 47 const gcm::GServicesSettings::SettingsMap& settings) { | |
| 48 gcm::GServicesSettings::SettingsMap::const_iterator iter = | |
| 49 settings.find(kCheckinIntervalKey); | |
| 50 // Default settings are only present in the map when overwritten. | |
|
jianli
2014/05/19 17:52:27
Comment is a bit hard to understand. Do you mean:
fgorski
2014/05/19 20:25:39
Done.
| |
| 51 if (iter == settings.end()) | |
|
Nicolas Zea
2014/05/19 17:49:48
It would be useful to have some helper method that
fgorski
2014/05/19 20:25:39
I added CanBeOmitted method with a documentation.
| |
| 52 return true; | |
| 53 | |
| 54 int64 checkin_interval = kMinimumCheckinInterval; | |
| 55 if (!base::StringToInt64(iter->second, &checkin_interval)) { | |
| 56 DVLOG(1) << "Failed to parse checkin interval: " << iter->second; | |
| 57 return false; | |
| 58 } | |
| 59 if (checkin_interval == std::numeric_limits<int64>::max()) { | |
| 60 DVLOG(1) << "Checkin interval is too big: " << checkin_interval; | |
| 61 return false; | |
| 62 } | |
| 63 if (checkin_interval < kMinimumCheckinInterval) { | |
| 64 DVLOG(1) << "Checkin interval: " << checkin_interval | |
| 65 << " is less than allowed minimum: " << kMinimumCheckinInterval; | |
| 66 } | |
| 67 | |
| 68 return true; | |
| 69 } | |
| 70 | |
| 71 bool VerifyMCSEndpoint(const gcm::GServicesSettings::SettingsMap& settings) { | |
| 72 // Default settings are only present in the map when overwritten. | |
| 73 std::string mcs_hostname = kDefaultMCSHostname; | |
| 74 gcm::GServicesSettings::SettingsMap::const_iterator iter = | |
| 75 settings.find(kMCSHostnameKey); | |
| 76 if (iter != settings.end()) | |
|
jianli
2014/05/19 17:52:27
This seems to be consistent with other Verify meth
fgorski
2014/05/19 20:25:39
I've updated it to be more consistent, but it is a
| |
| 77 mcs_hostname = iter->second; | |
| 78 | |
| 79 if (mcs_hostname.empty()) { | |
| 80 DVLOG(1) << "Empty MCS hostname provided."; | |
| 81 return false; | |
| 82 } | |
| 83 | |
| 84 int mcs_secure_port = 0; | |
| 85 iter = settings.find(kMCSSecurePortKey); | |
| 86 if (iter == settings.end()) { | |
| 87 mcs_secure_port = kDefaultMCSMainSecurePort; | |
| 88 } else if (!base::StringToInt(iter->second, &mcs_secure_port)) { | |
| 89 DVLOG(1) << "Failed to parse MCS secure port: " << iter->second; | |
| 90 return false; | |
| 91 } | |
| 92 | |
| 93 if (mcs_secure_port < 0 || mcs_secure_port > kMaxSecurePort) { | |
| 94 DVLOG(1) << "Incorrect port value: " << mcs_secure_port; | |
| 95 return false; | |
| 96 } | |
| 97 | |
| 98 GURL mcs_main_endpoint(MakeMCSEndpoint(mcs_hostname, mcs_secure_port)); | |
| 99 if (!mcs_main_endpoint.is_valid()) { | |
| 100 DVLOG(1) << "Invalid main MCS endpoint: " | |
| 101 << mcs_main_endpoint.possibly_invalid_spec(); | |
| 102 return false; | |
| 103 } | |
| 104 GURL mcs_fallback_endpoint( | |
| 105 MakeMCSEndpoint(mcs_hostname, kDefaultMCSFallbackSecurePort)); | |
| 106 if (!mcs_fallback_endpoint.is_valid()) { | |
| 107 DVLOG(1) << "Invalid fallback MCS endpoint: " | |
| 108 << mcs_fallback_endpoint.possibly_invalid_spec(); | |
| 109 return false; | |
| 110 } | |
| 111 | |
| 112 return true; | |
| 113 } | |
| 114 | |
| 115 bool VerifyCheckinURL(const gcm::GServicesSettings::SettingsMap& settings) { | |
| 116 gcm::GServicesSettings::SettingsMap::const_iterator iter = | |
| 117 settings.find(kCheckinURLKey); | |
| 118 // Default settings are only present in the map when overwritten. | |
| 119 if (iter == settings.end()) | |
| 120 return true; | |
| 121 | |
| 122 GURL checkin_url(iter->second); | |
| 123 if (!checkin_url.is_valid()) { | |
| 124 DVLOG(1) << "Invalid checkin URL provided: " << iter->second; | |
| 125 return false; | |
| 126 } | |
| 127 | |
| 128 return true; | |
| 129 } | |
| 130 | |
| 131 bool VerifyRegistrationURL( | |
| 132 const gcm::GServicesSettings::SettingsMap& settings) { | |
| 133 gcm::GServicesSettings::SettingsMap::const_iterator iter = | |
| 134 settings.find(kRegistrationURLKey); | |
| 135 // Default settings are only present in the map when overwritten. | |
| 136 if (iter == settings.end()) | |
| 137 return true; | |
| 138 | |
| 139 GURL registration_url(iter->second); | |
| 140 if (!registration_url.is_valid()) { | |
| 141 DVLOG(1) << "Invalid registration URL provided: " << iter->second; | |
| 142 return false; | |
| 143 } | |
| 144 | |
| 145 return true; | |
| 146 } | |
| 147 | |
| 148 bool VerifySettings(const gcm::GServicesSettings::SettingsMap& settings) { | |
| 149 return VerifyCheckinInterval(settings) && VerifyMCSEndpoint(settings) && | |
| 150 VerifyCheckinURL(settings) && VerifyRegistrationURL(settings); | |
| 151 } | |
| 152 | |
| 37 } // namespace | 153 } // namespace |
| 38 | 154 |
| 39 namespace gcm { | 155 namespace gcm { |
| 40 | 156 |
| 41 // static | 157 // static |
| 42 const base::TimeDelta GServicesSettings::MinimumCheckinInterval() { | 158 const base::TimeDelta GServicesSettings::MinimumCheckinInterval() { |
| 43 return base::TimeDelta::FromSeconds(kMinimumCheckinInterval); | 159 return base::TimeDelta::FromSeconds(kMinimumCheckinInterval); |
| 44 } | 160 } |
| 45 | 161 |
| 46 // static | 162 // static |
| 47 const GURL GServicesSettings::DefaultCheckinURL() { | 163 const GURL GServicesSettings::DefaultCheckinURL() { |
| 48 return GURL(kDefaultCheckinURL); | 164 return GURL(kDefaultCheckinURL); |
| 49 } | 165 } |
| 50 | 166 |
| 51 GServicesSettings::GServicesSettings() | 167 // static |
| 52 : checkin_interval_(base::TimeDelta::FromSeconds(kDefaultCheckinInterval)), | 168 std::string GServicesSettings::CalculateDigest(const SettingsMap& settings) { |
| 53 checkin_url_(kDefaultCheckinURL), | 169 unsigned char hash[base::kSHA1Length]; |
| 54 mcs_main_endpoint_(MakeMCSEndpoint(kDefaultMCSHostname, | 170 std::string data; |
| 55 kDefaultMCSMainSecurePort)), | 171 for (SettingsMap::const_iterator iter = settings.begin(); |
| 56 mcs_fallback_endpoint_(MakeMCSEndpoint(kDefaultMCSHostname, | 172 iter != settings.end(); |
| 57 kDefaultMCSFallbackSecurePort)), | 173 ++iter) { |
| 58 registration_url_(kDefaultRegistrationURL), | 174 data += iter->first; |
| 59 weak_ptr_factory_(this) { | 175 data += '\0'; |
| 176 data += iter->second; | |
| 177 data += '\0'; | |
| 178 } | |
| 179 base::SHA1HashBytes( | |
| 180 reinterpret_cast<const unsigned char*>(&data[0]), data.size(), hash); | |
| 181 std::string digest = | |
| 182 kDigestVersionPrefix + base::HexEncode(hash, base::kSHA1Length); | |
| 183 std::transform(digest.begin(), digest.end(), digest.begin(), ::tolower); | |
|
jianli
2014/05/19 17:52:27
nit: use StringToLowerASCII instead.
fgorski
2014/05/19 20:25:39
Done.
| |
| 184 return digest; | |
| 60 } | 185 } |
| 61 | 186 |
| 62 GServicesSettings::~GServicesSettings() {} | 187 GServicesSettings::GServicesSettings() : weak_ptr_factory_(this) { |
| 188 digest_ = CalculateDigest(settings_); | |
| 189 } | |
| 190 | |
| 191 GServicesSettings::~GServicesSettings() { | |
| 192 } | |
| 63 | 193 |
| 64 bool GServicesSettings::UpdateFromCheckinResponse( | 194 bool GServicesSettings::UpdateFromCheckinResponse( |
| 65 const checkin_proto::AndroidCheckinResponse& checkin_response) { | 195 const checkin_proto::AndroidCheckinResponse& checkin_response) { |
| 66 if (!checkin_response.has_digest() || | 196 if (!checkin_response.has_settings_diff()) { |
| 67 checkin_response.digest() == digest_) { | 197 DVLOG(1) << "Field settings_diff not set in response."; |
| 68 // There are no changes as digest is the same or no settings provided. | |
| 69 return false; | 198 return false; |
| 70 } | 199 } |
| 71 | 200 |
| 72 std::map<std::string, std::string> settings; | 201 bool settings_diff = checkin_response.settings_diff(); |
| 202 SettingsMap new_settings; | |
| 203 // Only reuse the existing settings, if we are given a settings difference. | |
| 204 if (settings_diff) | |
| 205 new_settings = settings_map(); | |
| 206 | |
| 73 for (int i = 0; i < checkin_response.setting_size(); ++i) { | 207 for (int i = 0; i < checkin_response.setting_size(); ++i) { |
| 74 std::string name = checkin_response.setting(i).name(); | 208 std::string name = checkin_response.setting(i).name(); |
| 75 std::string value = checkin_response.setting(i).value(); | 209 if (name.empty()) { |
| 76 settings[name] = value; | 210 DVLOG(1) << "Setting name is empty"; |
| 211 return false; | |
| 212 } | |
| 213 | |
| 214 if (settings_diff && name.find(kDeleteSettingPrefix) == 0) { | |
| 215 std::string setting_to_delete = | |
| 216 name.substr(arraysize(kDeleteSettingPrefix) - 1); | |
| 217 new_settings.erase(setting_to_delete); | |
| 218 DVLOG(1) << "Setting deleted: " << setting_to_delete; | |
| 219 } else { | |
| 220 std::string value = checkin_response.setting(i).value(); | |
| 221 new_settings[name] = value; | |
| 222 DVLOG(1) << "New setting: '" << name << "' : '" << value << "'"; | |
| 223 } | |
| 77 } | 224 } |
| 78 | 225 |
| 79 // Only update the settings in store and digest, if the settings actually | 226 if (!VerifySettings(new_settings)) |
| 80 // passed the verificaiton in update settings. | 227 return false; |
| 81 if (UpdateSettings(settings)) { | |
| 82 digest_ = checkin_response.digest(); | |
| 83 return true; | |
| 84 } | |
| 85 | 228 |
| 86 return false; | 229 settings_.swap(new_settings); |
| 230 digest_ = CalculateDigest(settings_); | |
| 231 return true; | |
| 87 } | 232 } |
| 88 | 233 |
| 89 void GServicesSettings::UpdateFromLoadResult( | 234 void GServicesSettings::UpdateFromLoadResult( |
| 90 const GCMStore::LoadResult& load_result) { | 235 const GCMStore::LoadResult& load_result) { |
| 91 if (UpdateSettings(load_result.gservices_settings)) | 236 // No need to try to update settings when load_result is empty. |
| 92 digest_ = load_result.gservices_digest; | 237 if (load_result.gservices_settings.empty()) |
| 238 return; | |
| 239 if (!VerifySettings(load_result.gservices_settings)) | |
| 240 return; | |
| 241 std::string digest = CalculateDigest(load_result.gservices_settings); | |
| 242 if (digest != load_result.gservices_digest) { | |
| 243 DVLOG(1) << "G-services settings digest mismatch. " | |
| 244 << "Expected digest: " << load_result.gservices_digest | |
| 245 << ". Calculated digest is: " << digest; | |
| 246 return; | |
| 247 } | |
| 248 | |
| 249 settings_ = load_result.gservices_settings; | |
| 250 digest_ = load_result.gservices_digest; | |
| 93 } | 251 } |
| 94 | 252 |
| 95 std::map<std::string, std::string> GServicesSettings::GetSettingsMap() const { | 253 base::TimeDelta GServicesSettings::GetCheckinInterval() const { |
| 96 std::map<std::string, std::string> settings; | 254 int64 checkin_interval = kMinimumCheckinInterval; |
| 97 settings[kCheckinIntervalKey] = | 255 SettingsMap::const_iterator iter = settings_.find(kCheckinIntervalKey); |
| 98 base::Int64ToString(checkin_interval_.InSeconds()); | 256 if (iter == settings_.end() || |
| 99 settings[kCheckinURLKey] = checkin_url_.spec(); | 257 !base::StringToInt64(iter->second, &checkin_interval)) { |
| 100 settings[kMCSHostnameKey] = mcs_main_endpoint_.host(); | 258 checkin_interval = kDefaultCheckinInterval; |
| 101 settings[kMCSSecurePortKey] = mcs_main_endpoint_.port(); | 259 } |
| 102 settings[kRegistrationURLKey] = registration_url_.spec(); | 260 |
| 103 return settings; | 261 if (checkin_interval < kMinimumCheckinInterval) |
| 262 checkin_interval = kMinimumCheckinInterval; | |
| 263 | |
| 264 return base::TimeDelta::FromSeconds(checkin_interval); | |
| 104 } | 265 } |
| 105 | 266 |
| 106 bool GServicesSettings::UpdateSettings( | 267 GURL GServicesSettings::GetCheckinURL() const { |
| 107 const std::map<std::string, std::string>& settings) { | 268 SettingsMap::const_iterator iter = settings_.find(kCheckinURLKey); |
| 108 int64 new_checkin_interval = kMinimumCheckinInterval; | 269 if (iter == settings_.end() || iter->second.empty()) |
| 109 std::map<std::string, std::string>::const_iterator iter = | 270 return GURL(kDefaultCheckinURL); |
| 110 settings.find(kCheckinIntervalKey); | 271 return GURL(iter->second); |
| 111 if (iter == settings.end()) { | 272 } |
| 112 LOG(ERROR) << "Setting not found: " << kCheckinIntervalKey; | 273 |
| 113 return false; | 274 GURL GServicesSettings::GetMCSMainEndpoint() const { |
| 114 } | 275 // Get alternative hostname or use default. |
| 115 if (!base::StringToInt64(iter->second, &new_checkin_interval)) { | 276 std::string mcs_hostname(kDefaultMCSHostname); |
|
jianli
2014/05/19 17:52:27
I think it will be more efficient not to set the d
fgorski
2014/05/19 20:25:39
Done.
| |
| 116 LOG(ERROR) << "Failed to parse checkin interval: " << iter->second; | 277 SettingsMap::const_iterator iter = settings_.find(kMCSHostnameKey); |
| 117 return false; | 278 if (iter != settings_.end() && !iter->second.empty()) |
| 118 } | 279 mcs_hostname = iter->second; |
| 119 if (new_checkin_interval < kMinimumCheckinInterval) { | 280 |
| 120 LOG(ERROR) << "Checkin interval: " << new_checkin_interval | 281 // Get alternative secure port or use defualt. |
| 121 << " is less than allowed minimum: " << kMinimumCheckinInterval; | 282 int mcs_secure_port = 0; |
| 122 new_checkin_interval = kMinimumCheckinInterval; | 283 iter = settings_.find(kMCSSecurePortKey); |
| 123 } | 284 if (iter == settings_.end() || iter->second.empty() || |
| 124 if (new_checkin_interval == std::numeric_limits<int64>::max()) { | 285 !base::StringToInt(iter->second, &mcs_secure_port)) { |
| 125 LOG(ERROR) << "Checkin interval is too big: " << new_checkin_interval; | 286 mcs_secure_port = kDefaultMCSMainSecurePort; |
| 126 return false; | |
| 127 } | 287 } |
| 128 | 288 |
| 129 std::string new_mcs_hostname; | 289 // If constructed address makes sense use it. |
| 130 iter = settings.find(kMCSHostnameKey); | 290 GURL mcs_endpoint(MakeMCSEndpoint(mcs_hostname, mcs_secure_port)); |
| 131 if (iter == settings.end()) { | 291 if (mcs_endpoint.is_valid()) |
| 132 LOG(ERROR) << "Setting not found: " << kMCSHostnameKey; | 292 return mcs_endpoint; |
| 133 return false; | |
| 134 } | |
| 135 new_mcs_hostname = iter->second; | |
| 136 if (new_mcs_hostname.empty()) { | |
| 137 LOG(ERROR) << "Empty MCS hostname provided."; | |
| 138 return false; | |
| 139 } | |
| 140 | 293 |
| 141 int new_mcs_secure_port = -1; | 294 // Otherwise use default settings. |
| 142 iter = settings.find(kMCSSecurePortKey); | 295 return GURL(MakeMCSEndpoint(kDefaultMCSHostname, kDefaultMCSMainSecurePort)); |
| 143 if (iter == settings.end()) { | 296 } |
| 144 LOG(ERROR) << "Setting not found: " << kMCSSecurePortKey; | |
| 145 return false; | |
| 146 } | |
| 147 if (!base::StringToInt(iter->second, &new_mcs_secure_port)) { | |
| 148 LOG(ERROR) << "Failed to parse MCS secure port: " << iter->second; | |
| 149 return false; | |
| 150 } | |
| 151 if (new_mcs_secure_port < 0 || 65535 < new_mcs_secure_port) { | |
| 152 LOG(ERROR) << "Incorrect port value: " << new_mcs_secure_port; | |
| 153 return false; | |
| 154 } | |
| 155 | 297 |
| 156 GURL new_mcs_main_endpoint = | 298 GURL GServicesSettings::GetMCSFallbackEndpoint() const { |
| 157 GURL(MakeMCSEndpoint(new_mcs_hostname, new_mcs_secure_port)); | 299 // Get alternative hostname or use default. |
| 158 GURL new_mcs_fallback_endpoint = | 300 std::string mcs_hostname(kDefaultMCSHostname); |
|
jianli
2014/05/19 17:52:27
ditto
fgorski
2014/05/19 20:25:39
Done.
| |
| 159 GURL(MakeMCSEndpoint(new_mcs_hostname, kDefaultMCSFallbackSecurePort)); | 301 SettingsMap::const_iterator iter = settings_.find(kMCSHostnameKey); |
| 160 if (!new_mcs_main_endpoint.is_valid() || | 302 if (iter != settings_.end() && !iter->second.empty()) |
| 161 !new_mcs_fallback_endpoint.is_valid()) | 303 mcs_hostname = iter->second; |
| 162 return false; | |
| 163 | 304 |
| 164 GURL new_checkin_url; | 305 // If constructed address makes sense use it. |
| 165 iter = settings.find(kCheckinURLKey); | 306 GURL mcs_endpoint( |
| 166 if (iter == settings.end()) { | 307 MakeMCSEndpoint(mcs_hostname, kDefaultMCSFallbackSecurePort)); |
| 167 LOG(ERROR) << "Setting not found: " << kCheckinURLKey; | 308 if (mcs_endpoint.is_valid()) |
| 168 return false; | 309 return mcs_endpoint; |
| 169 } | |
| 170 new_checkin_url = GURL(iter->second); | |
| 171 if (!new_checkin_url.is_valid()) { | |
| 172 LOG(ERROR) << "Invalid checkin URL provided: " | |
| 173 << new_checkin_url.possibly_invalid_spec(); | |
| 174 return false; | |
| 175 } | |
| 176 | 310 |
| 177 GURL new_registration_url; | 311 return GURL( |
| 178 iter = settings.find(kRegistrationURLKey); | 312 MakeMCSEndpoint(kDefaultMCSHostname, kDefaultMCSFallbackSecurePort)); |
| 179 if (iter == settings.end()) { | 313 } |
| 180 LOG(ERROR) << "Setting not found: " << kRegistrationURLKey; | |
| 181 return false; | |
| 182 } | |
| 183 new_registration_url = GURL(iter->second); | |
| 184 if (!new_registration_url.is_valid()) { | |
| 185 LOG(ERROR) << "Invalid registration URL provided: " | |
| 186 << new_registration_url.possibly_invalid_spec(); | |
| 187 return false; | |
| 188 } | |
| 189 | 314 |
| 190 // We only update the settings once all of them are correct. | 315 GURL GServicesSettings::GetRegistrationURL() const { |
| 191 checkin_interval_ = base::TimeDelta::FromSeconds(new_checkin_interval); | 316 SettingsMap::const_iterator iter = settings_.find(kRegistrationURLKey); |
| 192 mcs_main_endpoint_ = new_mcs_main_endpoint; | 317 if (iter == settings_.end() || iter->second.empty()) |
| 193 mcs_fallback_endpoint_ = new_mcs_fallback_endpoint; | 318 return GURL(kDefaultRegistrationURL); |
| 194 checkin_url_ = new_checkin_url; | 319 return GURL(iter->second); |
| 195 registration_url_ = new_registration_url; | |
| 196 return true; | |
| 197 } | 320 } |
| 198 | 321 |
| 199 } // namespace gcm | 322 } // namespace gcm |
| OLD | NEW |