Home | History | Annotate | Download | only in password_manager
      1 // Copyright 2013 The Chromium Authors. All rights reserved.
      2 // Use of this source code is governed by a BSD-style license that can be
      3 // found in the LICENSE file.
      4 
      5 #include "chrome/browser/password_manager/password_manager_metrics_util.h"
      6 
      7 #include "base/basictypes.h"
      8 #include "base/metrics/histogram.h"
      9 #include "base/prefs/pref_service.h"
     10 #include "base/prefs/scoped_user_pref_update.h"
     11 #include "base/rand_util.h"
     12 #include "base/safe_numerics.h"
     13 #include "base/strings/string_number_conversions.h"
     14 #include "base/strings/string_util.h"
     15 #include "base/time/time.h"
     16 #include "base/values.h"
     17 #include "chrome/common/pref_names.h"
     18 #include "url/gurl.h"
     19 
     20 using base::ListValue;
     21 using base::FundamentalValue;
     22 
     23 namespace password_manager_metrics_util {
     24 
     25 namespace {
     26 
     27 // The number of domain groups.
     28 const size_t kNumGroups = 2u * kGroupsPerDomain;
     29 
     30 // |kDomainMapping| contains each monitored website together with all ids of
     31 // groups which contain the website. Each website appears in
     32 // |kGroupsPerDomain| groups, and each group includes an equal number of
     33 // websites, so that no two websites have the same set of groups that they
     34 // belong to. All ids are in the range [1, |kNumGroups|].
     35 // For more information about the algorithm used see http://goo.gl/vUuFd5.
     36 struct DomainGroupsPair {
     37   const char* const domain_name;
     38   const size_t group_ids[kGroupsPerDomain];
     39 };
     40 const DomainGroupsPair kDomainMapping[] = {
     41     {"google.com", {1, 2, 3, 4, 5, 6, 7, 8, 9, 10}},
     42     {"yahoo.com", {1, 2, 3, 4, 5, 11, 12, 13, 14, 15}},
     43     {"baidu.com", {1, 2, 3, 4, 6, 7, 11, 12, 16, 17}},
     44     {"wikipedia.org", {1, 2, 3, 4, 5, 6, 11, 12, 16, 18}},
     45     {"linkedin.com", {1, 6, 8, 11, 13, 14, 15, 16, 17, 19}},
     46     {"twitter.com", {5, 6, 7, 8, 9, 11, 13, 17, 19, 20}},
     47     {"facebook.com", {7, 8, 9, 10, 13, 14, 16, 17, 18, 20}},
     48     {"amazon.com", {2, 5, 9, 10, 12, 14, 15, 18, 19, 20}},
     49     {"ebay.com", {3, 7, 9, 10, 14, 15, 17, 18, 19, 20}},
     50     {"tumblr.com", {4, 8, 10, 12, 13, 15, 16, 18, 19, 20}},
     51 };
     52 const size_t kNumDomains = arraysize(kDomainMapping);
     53 
     54 // For every monitored domain, this function chooses which of the groups
     55 // containing that domain should be used for reporting. That number is chosen
     56 // randomly and stored in the user's preferences.
     57 size_t GetGroupIndex(size_t domain_index, PrefService* pref_service) {
     58   DCHECK_LT(domain_index, kNumDomains);
     59 
     60   const ListValue* group_indices =
     61       pref_service->GetList(prefs::kPasswordManagerGroupsForDomains);
     62   int result = 0;
     63   if (!group_indices->GetInteger(domain_index, &result)) {
     64     ListPrefUpdate group_indices_updater(
     65         pref_service, prefs::kPasswordManagerGroupsForDomains);
     66     // This value has not been generated yet.
     67     result =
     68         base::checked_numeric_cast<int>(base::RandGenerator(kGroupsPerDomain));
     69     group_indices_updater->Set(domain_index, new FundamentalValue(result));
     70   }
     71   return base::checked_numeric_cast<size_t>(result);
     72 }
     73 
     74 }  // namespace
     75 
     76 size_t MonitoredDomainGroupId(const std::string& url_host,
     77                               PrefService* pref_service) {
     78   GURL url(url_host);
     79   for (size_t i = 0; i < kNumDomains; ++i) {
     80     if (url.DomainIs(kDomainMapping[i].domain_name))
     81       return kDomainMapping[i].group_ids[GetGroupIndex(i, pref_service)];
     82   }
     83   return 0;
     84 }
     85 
     86 void LogUMAHistogramEnumeration(const std::string& name,
     87                                 int sample,
     88                                 int boundary_value) {
     89   DCHECK_LT(sample, boundary_value);
     90 
     91   // Note: This leaks memory, which is expected behavior.
     92   base::HistogramBase* histogram =
     93       base::LinearHistogram::FactoryGet(
     94           name,
     95           1,
     96           boundary_value,
     97           boundary_value + 1,
     98           base::HistogramBase::kUmaTargetedHistogramFlag);
     99   histogram->Add(sample);
    100 }
    101 
    102 void LogUMAHistogramBoolean(const std::string& name, bool sample) {
    103   // Note: This leaks memory, which is expected behavior.
    104   base::HistogramBase* histogram =
    105       base::BooleanHistogram::FactoryGet(
    106           name,
    107           base::Histogram::kNoFlags);
    108           histogram->AddBoolean(sample);
    109 }
    110 
    111 std::string GroupIdToString(size_t group_id) {
    112   DCHECK_LE(group_id, kNumGroups);
    113   if (group_id > 0)
    114     return "group_" + base::IntToString(group_id);
    115   return std::string();
    116 }
    117 
    118 }  // namespace password_manager_metrics_util
    119