// Copyright 2014 The Chromium Authors. All rights reserved. // Use of this source code is governed by a BSD-style license that can be // found in the LICENSE file. #include "components/suggestions/suggestions_store.h" #include <string> #include "base/base64.h" #include "base/prefs/pref_service.h" #include "base/time/time.h" #include "components/pref_registry/pref_registry_syncable.h" #include "components/suggestions/suggestions_pref_names.h" namespace suggestions { SuggestionsStore::SuggestionsStore(PrefService* profile_prefs) : pref_service_(profile_prefs) { DCHECK(profile_prefs); } SuggestionsStore::~SuggestionsStore() {} bool SuggestionsStore::LoadSuggestions(SuggestionsProfile* suggestions) { DCHECK(suggestions); const std::string base64_suggestions_data = pref_service_->GetString(prefs::kSuggestionsData); if (base64_suggestions_data.empty()) { suggestions->Clear(); return false; } // If the decode process fails, assume the pref value is corrupt and clear it. std::string suggestions_data; if (!base::Base64Decode(base64_suggestions_data, &suggestions_data) || !suggestions->ParseFromString(suggestions_data)) { VLOG(1) << "Suggestions data in profile pref is corrupt, clearing it."; suggestions->Clear(); ClearSuggestions(); return false; } // Filter expired suggestions and update the stored suggestions if at least // one was filtered. Return false if all suggestions are filtered. int unfiltered_size = suggestions->suggestions_size(); FilterExpiredSuggestions(suggestions); if (suggestions->suggestions_size() != unfiltered_size) { if (!suggestions->suggestions_size()) { suggestions->Clear(); ClearSuggestions(); return false; } else { StoreSuggestions(*suggestions); } } return true; } void SuggestionsStore::FilterExpiredSuggestions( SuggestionsProfile* suggestions) { SuggestionsProfile filtered_suggestions; int64 now_usec = (base::Time::NowFromSystemTime() - base::Time::UnixEpoch()) .ToInternalValue(); for (int i = 0; i < suggestions->suggestions_size(); ++i) { ChromeSuggestion* suggestion = suggestions->mutable_suggestions(i); if (!suggestion->has_expiry_ts() || suggestion->expiry_ts() > now_usec) { filtered_suggestions.add_suggestions()->Swap(suggestion); } } suggestions->Swap(&filtered_suggestions); } bool SuggestionsStore::StoreSuggestions(const SuggestionsProfile& suggestions) { std::string suggestions_data; if (!suggestions.SerializeToString(&suggestions_data)) return false; std::string base64_suggestions_data; base::Base64Encode(suggestions_data, &base64_suggestions_data); pref_service_->SetString(prefs::kSuggestionsData, base64_suggestions_data); return true; } void SuggestionsStore::ClearSuggestions() { pref_service_->ClearPref(prefs::kSuggestionsData); } // static void SuggestionsStore::RegisterProfilePrefs( user_prefs::PrefRegistrySyncable* registry) { registry->RegisterStringPref( prefs::kSuggestionsData, std::string(), user_prefs::PrefRegistrySyncable::UNSYNCABLE_PREF); } } // namespace suggestions