Chromium Code Reviews
chromiumcodereview-hr@appspot.gserviceaccount.com (chromiumcodereview-hr) | Please choose your nickname with Settings | Help | Chromium Project | Gerrit Changes | Sign out
(40)

Side by Side Diff: chrome/browser/metrics/variations/variations_seed_store.cc

Issue 1200233005: Support delta-compressed variations server responses. (Closed) Base URL: https://chromium.googlesource.com/chromium/src.git@master
Patch Set: Rebase. Created 5 years, 4 months ago
Use n/p to move between diff chunks; N/P to move between comments. Draft comments are only viewable by you.
Jump to:
View unified diff | Download patch
OLDNEW
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 "chrome/browser/metrics/variations/variations_seed_store.h" 5 #include "chrome/browser/metrics/variations/variations_seed_store.h"
6 6
7 #include "base/base64.h" 7 #include "base/base64.h"
8 #include "base/metrics/histogram_macros.h" 8 #include "base/metrics/histogram_macros.h"
9 #include "base/numerics/safe_math.h"
9 #include "base/prefs/pref_registry_simple.h" 10 #include "base/prefs/pref_registry_simple.h"
10 #include "base/prefs/pref_service.h" 11 #include "base/prefs/pref_service.h"
11 #include "base/sha1.h" 12 #include "base/sha1.h"
12 #include "base/strings/string_number_conversions.h" 13 #include "base/strings/string_number_conversions.h"
13 #include "chrome/common/pref_names.h" 14 #include "chrome/common/pref_names.h"
14 #include "components/metrics/compression_utils.h" 15 #include "components/metrics/compression_utils.h"
15 #include "components/variations/proto/variations_seed.pb.h" 16 #include "components/variations/proto/variations_seed.pb.h"
16 #include "crypto/signature_verifier.h" 17 #include "crypto/signature_verifier.h"
18 #include "third_party/protobuf/src/google/protobuf/io/coded_stream.h"
17 19
18 namespace chrome_variations { 20 namespace chrome_variations {
19 21
20 namespace { 22 namespace {
21 23
22 // Signature verification is disabled on mobile platforms for now, since it 24 // Signature verification is disabled on mobile platforms for now, since it
23 // adds about ~15ms to the startup time on mobile (vs. a couple ms on desktop). 25 // adds about ~15ms to the startup time on mobile (vs. a couple ms on desktop).
24 bool SignatureVerificationEnabled() { 26 bool SignatureVerificationEnabled() {
25 #if defined(OS_IOS) || defined(OS_ANDROID) 27 #if defined(OS_IOS) || defined(OS_ANDROID)
26 return false; 28 return false;
(...skipping 48 matching lines...) Expand 10 before | Expand all | Expand 10 after
75 UMA_HISTOGRAM_ENUMERATION("Variations.SeedEmpty", state, 77 UMA_HISTOGRAM_ENUMERATION("Variations.SeedEmpty", state,
76 VARIATIONS_SEED_EMPTY_ENUM_SIZE); 78 VARIATIONS_SEED_EMPTY_ENUM_SIZE);
77 } 79 }
78 80
79 enum VariationsSeedStoreResult { 81 enum VariationsSeedStoreResult {
80 VARIATIONS_SEED_STORE_SUCCESS, 82 VARIATIONS_SEED_STORE_SUCCESS,
81 VARIATIONS_SEED_STORE_FAILED_EMPTY, 83 VARIATIONS_SEED_STORE_FAILED_EMPTY,
82 VARIATIONS_SEED_STORE_FAILED_PARSE, 84 VARIATIONS_SEED_STORE_FAILED_PARSE,
83 VARIATIONS_SEED_STORE_FAILED_SIGNATURE, 85 VARIATIONS_SEED_STORE_FAILED_SIGNATURE,
84 VARIATIONS_SEED_STORE_FAILED_GZIP, 86 VARIATIONS_SEED_STORE_FAILED_GZIP,
87 VARIATIONS_SEED_STORE_DELTA_COUNT,
rkaplow 2015/08/04 22:37:05 DELTA_COUNT is a bit unlike the rest (not really a
Alexei Svitkine (slow) 2015/08/05 15:46:15 Done.
88 VARIATIONS_SEED_STORE_FAILED_DELTA_READ_SEED,
89 VARIATIONS_SEED_STORE_FAILED_DELTA_APPLY,
90 VARIATIONS_SEED_STORE_FAILED_DELTA_STORE,
85 VARIATIONS_SEED_STORE_RESULT_ENUM_SIZE, 91 VARIATIONS_SEED_STORE_RESULT_ENUM_SIZE,
86 }; 92 };
87 93
88 void RecordVariationsSeedStoreHistogram(VariationsSeedStoreResult result) { 94 void RecordSeedStoreHistogram(VariationsSeedStoreResult result) {
89 UMA_HISTOGRAM_ENUMERATION("Variations.SeedStoreResult", result, 95 UMA_HISTOGRAM_ENUMERATION("Variations.SeedStoreResult", result,
90 VARIATIONS_SEED_STORE_RESULT_ENUM_SIZE); 96 VARIATIONS_SEED_STORE_RESULT_ENUM_SIZE);
91 } 97 }
92 98
93 // Note: UMA histogram enum - don't re-order or remove entries. 99 // Note: UMA histogram enum - don't re-order or remove entries.
94 enum VariationsSeedDateChangeState { 100 enum VariationsSeedDateChangeState {
95 SEED_DATE_NO_OLD_DATE, 101 SEED_DATE_NO_OLD_DATE,
96 SEED_DATE_NEW_DATE_OLDER, 102 SEED_DATE_NEW_DATE_OLDER,
97 SEED_DATE_SAME_DAY, 103 SEED_DATE_SAME_DAY,
98 SEED_DATE_NEW_DAY, 104 SEED_DATE_NEW_DAY,
(...skipping 25 matching lines...) Expand all
124 // The server date is earlier than the stored date, and they are from 130 // The server date is earlier than the stored date, and they are from
125 // different UTC days, so |server_seed_date| is a valid new day. 131 // different UTC days, so |server_seed_date| is a valid new day.
126 return SEED_DATE_NEW_DAY; 132 return SEED_DATE_NEW_DAY;
127 } 133 }
128 return SEED_DATE_SAME_DAY; 134 return SEED_DATE_SAME_DAY;
129 } 135 }
130 136
131 } // namespace 137 } // namespace
132 138
133 VariationsSeedStore::VariationsSeedStore(PrefService* local_state) 139 VariationsSeedStore::VariationsSeedStore(PrefService* local_state)
134 : local_state_(local_state) { 140 : local_state_(local_state), seed_has_country_code_(false) {
135 } 141 }
136 142
137 VariationsSeedStore::~VariationsSeedStore() { 143 VariationsSeedStore::~VariationsSeedStore() {
138 } 144 }
139 145
140 bool VariationsSeedStore::LoadSeed(variations::VariationsSeed* seed) { 146 bool VariationsSeedStore::LoadSeed(variations::VariationsSeed* seed) {
141 invalid_base64_signature_.clear(); 147 invalid_base64_signature_.clear();
142 148
143 std::string seed_data; 149 std::string seed_data;
144 if (!ReadSeedData(&seed_data)) 150 if (!ReadSeedData(&seed_data))
(...skipping 22 matching lines...) Expand all
167 } 173 }
168 174
169 // Migrate any existing country code from the seed to the pref, if the pref is 175 // Migrate any existing country code from the seed to the pref, if the pref is
170 // empty. TODO(asvitkine): Clean up the code in M50+ when sufficient number 176 // empty. TODO(asvitkine): Clean up the code in M50+ when sufficient number
171 // of clients have migrated. 177 // of clients have migrated.
172 if (seed->has_country_code() && 178 if (seed->has_country_code() &&
173 local_state_->GetString(prefs::kVariationsCountry).empty()) { 179 local_state_->GetString(prefs::kVariationsCountry).empty()) {
174 local_state_->SetString(prefs::kVariationsCountry, seed->country_code()); 180 local_state_->SetString(prefs::kVariationsCountry, seed->country_code());
175 } 181 }
176 variations_serial_number_ = seed->serial_number(); 182 variations_serial_number_ = seed->serial_number();
183 seed_has_country_code_ = seed->has_country_code();
177 RecordVariationSeedEmptyHistogram(VARIATIONS_SEED_NOT_EMPTY); 184 RecordVariationSeedEmptyHistogram(VARIATIONS_SEED_NOT_EMPTY);
178 return true; 185 return true;
179 } 186 }
180 187
181 bool VariationsSeedStore::StoreSeedData( 188 bool VariationsSeedStore::StoreSeedData(
189 const std::string& data,
190 const std::string& base64_seed_signature,
191 const std::string& country_code,
192 const base::Time& date_fetched,
193 bool is_delta_compressed,
194 variations::VariationsSeed* parsed_seed) {
195 if (!is_delta_compressed) {
196 return StoreSeedDataNoDelta(data, base64_seed_signature, country_code,
197 date_fetched, parsed_seed);
198 }
199
200 // If the data is delta compressed, first decode it.
201 DCHECK(invalid_base64_signature_.empty());
202 RecordSeedStoreHistogram(VARIATIONS_SEED_STORE_DELTA_COUNT);
203
204 std::string existing_seed_data;
205 std::string updated_seed_data;
206 if (!ReadSeedData(&existing_seed_data)) {
207 RecordSeedStoreHistogram(VARIATIONS_SEED_STORE_FAILED_DELTA_READ_SEED);
208 return false;
209 }
210 if (!ApplyDeltaPatch(existing_seed_data, data, &updated_seed_data)) {
211 RecordSeedStoreHistogram(VARIATIONS_SEED_STORE_FAILED_DELTA_APPLY);
212 return false;
213 }
214
215 const bool result =
216 StoreSeedDataNoDelta(updated_seed_data, base64_seed_signature,
217 country_code, date_fetched, parsed_seed);
218 if (result) {
219 // Note: |updated_seed_data.length()| is guaranteed to be non-zero, else
220 // result would be false.
221 int size_reduction = updated_seed_data.length() - data.length();
222 UMA_HISTOGRAM_PERCENTAGE("Variations.SeedDelta.SizeReductionPercent",
223 100 * size_reduction / updated_seed_data.length());
224 } else {
225 RecordSeedStoreHistogram(VARIATIONS_SEED_STORE_FAILED_DELTA_STORE);
226 }
227 return result;
228 }
229
230 // TODO(asvitkine): Move this method down to match declaration order in a
rkaplow 2015/08/04 22:37:05 did you leave it as is as to make the diff more re
Alexei Svitkine (slow) 2015/08/05 15:46:15 Yep. :)
231 // follow-up CL.
232 bool VariationsSeedStore::StoreSeedDataNoDelta(
182 const std::string& seed_data, 233 const std::string& seed_data,
183 const std::string& base64_seed_signature, 234 const std::string& base64_seed_signature,
235 const std::string& country_code,
184 const base::Time& date_fetched, 236 const base::Time& date_fetched,
185 variations::VariationsSeed* parsed_seed) { 237 variations::VariationsSeed* parsed_seed) {
186 if (seed_data.empty()) { 238 if (seed_data.empty()) {
187 RecordVariationsSeedStoreHistogram(VARIATIONS_SEED_STORE_FAILED_EMPTY); 239 RecordSeedStoreHistogram(VARIATIONS_SEED_STORE_FAILED_EMPTY);
188 return false; 240 return false;
189 } 241 }
190 242
191 // Only store the seed data if it parses correctly. 243 // Only store the seed data if it parses correctly.
192 variations::VariationsSeed seed; 244 variations::VariationsSeed seed;
193 if (!seed.ParseFromString(seed_data)) { 245 if (!seed.ParseFromString(seed_data)) {
194 RecordVariationsSeedStoreHistogram(VARIATIONS_SEED_STORE_FAILED_PARSE); 246 RecordSeedStoreHistogram(VARIATIONS_SEED_STORE_FAILED_PARSE);
195 return false; 247 return false;
196 } 248 }
197 249
198 const VerifySignatureResult result = 250 const VerifySignatureResult result =
199 VerifySeedSignature(seed_data, base64_seed_signature); 251 VerifySeedSignature(seed_data, base64_seed_signature);
200 if (result != VARIATIONS_SEED_SIGNATURE_ENUM_SIZE) { 252 if (result != VARIATIONS_SEED_SIGNATURE_ENUM_SIZE) {
201 UMA_HISTOGRAM_ENUMERATION("Variations.StoreSeedSignature", result, 253 UMA_HISTOGRAM_ENUMERATION("Variations.StoreSeedSignature", result,
202 VARIATIONS_SEED_SIGNATURE_ENUM_SIZE); 254 VARIATIONS_SEED_SIGNATURE_ENUM_SIZE);
203 if (result != VARIATIONS_SEED_SIGNATURE_VALID) { 255 if (result != VARIATIONS_SEED_SIGNATURE_VALID) {
204 RecordVariationsSeedStoreHistogram( 256 RecordSeedStoreHistogram(VARIATIONS_SEED_STORE_FAILED_SIGNATURE);
205 VARIATIONS_SEED_STORE_FAILED_SIGNATURE);
206 return false; 257 return false;
207 } 258 }
208 } 259 }
209 260
210 // Compress the seed before base64-encoding and storing. 261 // Compress the seed before base64-encoding and storing.
211 std::string compressed_seed_data; 262 std::string compressed_seed_data;
212 if (!metrics::GzipCompress(seed_data, &compressed_seed_data)) { 263 if (!metrics::GzipCompress(seed_data, &compressed_seed_data)) {
213 RecordVariationsSeedStoreHistogram(VARIATIONS_SEED_STORE_FAILED_GZIP); 264 RecordSeedStoreHistogram(VARIATIONS_SEED_STORE_FAILED_GZIP);
214 return false; 265 return false;
215 } 266 }
216 267
217 std::string base64_seed_data; 268 std::string base64_seed_data;
218 base::Base64Encode(compressed_seed_data, &base64_seed_data); 269 base::Base64Encode(compressed_seed_data, &base64_seed_data);
219 270
220 // TODO(asvitkine): This pref is no longer being used. Remove it completely 271 // TODO(asvitkine): This pref is no longer being used. Remove it completely
221 // in M45+. 272 // in M45+.
222 local_state_->ClearPref(prefs::kVariationsSeed); 273 local_state_->ClearPref(prefs::kVariationsSeed);
223 274
224 // Update the saved country code only if one was returned from the server. 275 // Update the saved country code only if one was returned from the server.
225 if (seed.has_country_code()) 276 // Prefer the country code that was transmitted in the header over the one in
277 // the seed (which is deprecated).
278 if (!country_code.empty())
279 local_state_->SetString(prefs::kVariationsCountry, country_code);
280 else if (seed.has_country_code())
226 local_state_->SetString(prefs::kVariationsCountry, seed.country_code()); 281 local_state_->SetString(prefs::kVariationsCountry, seed.country_code());
227 282
228 local_state_->SetString(prefs::kVariationsCompressedSeed, base64_seed_data); 283 local_state_->SetString(prefs::kVariationsCompressedSeed, base64_seed_data);
229 UpdateSeedDateAndLogDayChange(date_fetched); 284 UpdateSeedDateAndLogDayChange(date_fetched);
230 local_state_->SetString(prefs::kVariationsSeedSignature, 285 local_state_->SetString(prefs::kVariationsSeedSignature,
231 base64_seed_signature); 286 base64_seed_signature);
232 variations_serial_number_ = seed.serial_number(); 287 variations_serial_number_ = seed.serial_number();
233 if (parsed_seed) 288 if (parsed_seed)
234 seed.Swap(parsed_seed); 289 seed.Swap(parsed_seed);
235 290
236 RecordVariationsSeedStoreHistogram(VARIATIONS_SEED_STORE_SUCCESS); 291 RecordSeedStoreHistogram(VARIATIONS_SEED_STORE_SUCCESS);
237 return true; 292 return true;
238 } 293 }
239 294
240 void VariationsSeedStore::UpdateSeedDateAndLogDayChange( 295 void VariationsSeedStore::UpdateSeedDateAndLogDayChange(
241 const base::Time& server_date_fetched) { 296 const base::Time& server_date_fetched) {
242 VariationsSeedDateChangeState date_change = SEED_DATE_NO_OLD_DATE; 297 VariationsSeedDateChangeState date_change = SEED_DATE_NO_OLD_DATE;
243 298
244 if (local_state_->HasPrefPath(prefs::kVariationsSeedDate)) { 299 if (local_state_->HasPrefPath(prefs::kVariationsSeedDate)) {
245 const int64 stored_date_value = 300 const int64 stored_date_value =
246 local_state_->GetInt64(prefs::kVariationsSeedDate); 301 local_state_->GetInt64(prefs::kVariationsSeedDate);
247 const base::Time stored_date = 302 const base::Time stored_date =
248 base::Time::FromInternalValue(stored_date_value); 303 base::Time::FromInternalValue(stored_date_value);
249 304
250 date_change = GetSeedDateChangeState(server_date_fetched, stored_date); 305 date_change = GetSeedDateChangeState(server_date_fetched, stored_date);
251 } 306 }
252 307
253 UMA_HISTOGRAM_ENUMERATION("Variations.SeedDateChange", date_change, 308 UMA_HISTOGRAM_ENUMERATION("Variations.SeedDateChange", date_change,
254 SEED_DATE_ENUM_SIZE); 309 SEED_DATE_ENUM_SIZE);
255 310
256 local_state_->SetInt64(prefs::kVariationsSeedDate, 311 local_state_->SetInt64(prefs::kVariationsSeedDate,
257 server_date_fetched.ToInternalValue()); 312 server_date_fetched.ToInternalValue());
258 } 313 }
259 314
315
316 std::string VariationsSeedStore::GetInvalidSignature() const {
317 return invalid_base64_signature_;
318 }
319
260 // static 320 // static
261 void VariationsSeedStore::RegisterPrefs(PrefRegistrySimple* registry) { 321 void VariationsSeedStore::RegisterPrefs(PrefRegistrySimple* registry) {
262 registry->RegisterStringPref(prefs::kVariationsCompressedSeed, std::string()); 322 registry->RegisterStringPref(prefs::kVariationsCompressedSeed, std::string());
263 registry->RegisterStringPref(prefs::kVariationsSeed, std::string()); 323 registry->RegisterStringPref(prefs::kVariationsSeed, std::string());
264 registry->RegisterInt64Pref(prefs::kVariationsSeedDate, 324 registry->RegisterInt64Pref(prefs::kVariationsSeedDate,
265 base::Time().ToInternalValue()); 325 base::Time().ToInternalValue());
266 registry->RegisterStringPref(prefs::kVariationsSeedSignature, std::string()); 326 registry->RegisterStringPref(prefs::kVariationsSeedSignature, std::string());
267 } 327 }
268 328
269 void VariationsSeedStore::ClearPrefs() { 329 void VariationsSeedStore::ClearPrefs() {
(...skipping 57 matching lines...) Expand 10 before | Expand all | Expand 10 after
327 return VARIATIONS_SEED_SIGNATURE_INVALID_SIGNATURE; 387 return VARIATIONS_SEED_SIGNATURE_INVALID_SIGNATURE;
328 } 388 }
329 389
330 verifier.VerifyUpdate(reinterpret_cast<const uint8*>(seed_bytes.data()), 390 verifier.VerifyUpdate(reinterpret_cast<const uint8*>(seed_bytes.data()),
331 seed_bytes.size()); 391 seed_bytes.size());
332 if (verifier.VerifyFinal()) 392 if (verifier.VerifyFinal())
333 return VARIATIONS_SEED_SIGNATURE_VALID; 393 return VARIATIONS_SEED_SIGNATURE_VALID;
334 return VARIATIONS_SEED_SIGNATURE_INVALID_SEED; 394 return VARIATIONS_SEED_SIGNATURE_INVALID_SEED;
335 } 395 }
336 396
337 std::string VariationsSeedStore::GetInvalidSignature() const { 397 // static
338 return invalid_base64_signature_; 398 bool VariationsSeedStore::ApplyDeltaPatch(const std::string& existing_data,
399 const std::string& patch,
400 std::string* output) {
401 output->clear();
402
403 google::protobuf::io::CodedInputStream in(
404 reinterpret_cast<const uint8*>(patch.data()), patch.length());
405 // Temporary string declared outside the loop so it can be re-used between
406 // different iterations (rather than allocating new ones).
407 std::string temp;
408
409 const uint32 existing_data_size = static_cast<uint32>(existing_data.size());
410 while (in.CurrentPosition() != static_cast<int>(patch.length())) {
411 uint32 value;
412 if (!in.ReadVarint32(&value))
413 return false;
414
415 if (value == 0) {
rkaplow 2015/08/04 22:37:05 hm, maybe I'm not familiar enough with data diffs,
Alexei Svitkine (slow) 2015/08/05 15:46:15 It's actually just the format of the diff produced
416 uint32 offset;
417 uint32 length;
418 if (!in.ReadVarint32(&offset) || !in.ReadVarint32(&length))
419 return false;
420
421 // Check for |offset + length| being out of range and for overflow.
422 base::CheckedNumeric<uint32> end_offset(offset);
423 end_offset += length;
424 if (!end_offset.IsValid() || end_offset.ValueOrDie() > existing_data_size)
425 return false;
426 output->append(existing_data, offset, length);
427 } else {
428 // No need to guard against bad data (i.e. very large |value|) because the
429 // call below will fail if |value| is greater than the size of the patch.
430 if (!in.ReadString(&temp, value))
431 return false;
432 output->append(temp);
433 }
434 }
435 return true;
339 } 436 }
340 437
341 } // namespace chrome_variations 438 } // namespace chrome_variations
OLDNEW

Powered by Google App Engine
This is Rietveld 408576698