Home | History | Annotate | Download | only in variations
      1 // Copyright (c) 2012 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/common/metrics/variations/variations_util.h"
      6 
      7 #include <vector>
      8 
      9 #include "base/strings/string16.h"
     10 #include "base/strings/string_number_conversions.h"
     11 #include "base/strings/string_split.h"
     12 #include "base/strings/string_util.h"
     13 #include "base/strings/stringprintf.h"
     14 #include "base/strings/utf_string_conversions.h"
     15 #include "chrome/common/child_process_logging.h"
     16 #include "chrome/installer/util/google_update_experiment_util.h"
     17 
     18 namespace chrome_variations {
     19 
     20 namespace {
     21 
     22 const char kVariationPrefix[] = "CrVar";
     23 const char kExperimentLabelSep[] = ";";
     24 
     25 // Populates |name_group_ids| based on |active_groups|.
     26 void GetFieldTrialActiveGroupIdsForActiveGroups(
     27     const base::FieldTrial::ActiveGroups& active_groups,
     28     std::vector<ActiveGroupId>* name_group_ids) {
     29   DCHECK(name_group_ids->empty());
     30   for (base::FieldTrial::ActiveGroups::const_iterator it =
     31        active_groups.begin(); it != active_groups.end(); ++it) {
     32     name_group_ids->push_back(MakeActiveGroupId(it->trial_name,
     33                                                 it->group_name));
     34   }
     35 }
     36 
     37 // This method builds a single experiment label for a Chrome Variation,
     38 // including a timestamp that is a year in the future from now. Since multiple
     39 // headers can be transmitted, |count| is a number that is appended after the
     40 // label key to differentiate the labels.
     41 string16 CreateSingleExperimentLabel(int count, VariationID id) {
     42   // Build the parts separately so they can be validated.
     43   const string16 key =
     44       ASCIIToUTF16(kVariationPrefix) + base::IntToString16(count);
     45   DCHECK_LE(key.size(), 8U);
     46   const string16 value = base::IntToString16(id);
     47   DCHECK_LE(value.size(), 8U);
     48   string16 label(key);
     49   label += ASCIIToUTF16("=");
     50   label += value;
     51   label += ASCIIToUTF16("|");
     52   label += installer::BuildExperimentDateString();
     53   return label;
     54 }
     55 
     56 }  // namespace
     57 
     58 void GetFieldTrialActiveGroupIds(
     59     std::vector<ActiveGroupId>* name_group_ids) {
     60   DCHECK(name_group_ids->empty());
     61   // A note on thread safety: Since GetActiveFieldTrialGroups() is thread
     62   // safe, and we operate on a separate list of that data, this function is
     63   // technically thread safe as well, with respect to the FieldTriaList data.
     64   base::FieldTrial::ActiveGroups active_groups;
     65   base::FieldTrialList::GetActiveFieldTrialGroups(&active_groups);
     66   GetFieldTrialActiveGroupIdsForActiveGroups(active_groups,
     67                                              name_group_ids);
     68 }
     69 
     70 void GetFieldTrialActiveGroupIdsAsStrings(
     71     std::vector<string16>* output) {
     72   DCHECK(output->empty());
     73   std::vector<ActiveGroupId> name_group_ids;
     74   GetFieldTrialActiveGroupIds(&name_group_ids);
     75   for (size_t i = 0; i < name_group_ids.size(); ++i) {
     76     output->push_back(UTF8ToUTF16(base::StringPrintf(
     77         "%x-%x", name_group_ids[i].name, name_group_ids[i].group)));
     78   }
     79 }
     80 
     81 void GenerateVariationChunks(const std::vector<string16>& experiments,
     82                              std::vector<string16>* chunks) {
     83   string16 current_chunk;
     84   for (size_t i = 0; i < experiments.size(); ++i) {
     85     const size_t needed_length =
     86         (current_chunk.empty() ? 1 : 0) + experiments[i].length();
     87     if (current_chunk.length() + needed_length > kMaxVariationChunkSize) {
     88       chunks->push_back(current_chunk);
     89       current_chunk = experiments[i];
     90     } else {
     91       if (!current_chunk.empty())
     92         current_chunk.push_back(',');
     93       current_chunk += experiments[i];
     94     }
     95   }
     96   if (!current_chunk.empty())
     97     chunks->push_back(current_chunk);
     98 }
     99 
    100 void SetChildProcessLoggingVariationList() {
    101   std::vector<string16> experiment_strings;
    102   GetFieldTrialActiveGroupIdsAsStrings(&experiment_strings);
    103   child_process_logging::SetExperimentList(experiment_strings);
    104 }
    105 
    106 string16 BuildGoogleUpdateExperimentLabel(
    107     const base::FieldTrial::ActiveGroups& active_groups) {
    108   string16 experiment_labels;
    109   int counter = 0;
    110 
    111   // Find all currently active VariationIDs associated with Google Update.
    112   for (base::FieldTrial::ActiveGroups::const_iterator it =
    113        active_groups.begin(); it != active_groups.end(); ++it) {
    114     const VariationID id = GetGoogleVariationID(GOOGLE_UPDATE_SERVICE,
    115                                                 it->trial_name, it->group_name);
    116 
    117     if (id == EMPTY_ID)
    118       continue;
    119 
    120     if (!experiment_labels.empty())
    121       experiment_labels += ASCIIToUTF16(kExperimentLabelSep);
    122     experiment_labels += CreateSingleExperimentLabel(++counter, id);
    123   }
    124 
    125   return experiment_labels;
    126 }
    127 
    128 string16 ExtractNonVariationLabels(const string16& labels) {
    129   const string16 separator = ASCIIToUTF16(kExperimentLabelSep);
    130   string16 non_variation_labels;
    131 
    132   // First, split everything by the label separator.
    133   std::vector<string16> entries;
    134   base::SplitStringUsingSubstr(labels, separator, &entries);
    135 
    136   // For each label, keep the ones that do not look like a Variations label.
    137   for (std::vector<string16>::const_iterator it = entries.begin();
    138        it != entries.end(); ++it) {
    139     if (it->empty() || StartsWith(*it, ASCIIToUTF16(kVariationPrefix), false))
    140       continue;
    141 
    142     // Dump the whole thing, including the timestamp.
    143     if (!non_variation_labels.empty())
    144       non_variation_labels += separator;
    145     non_variation_labels += *it;
    146   }
    147 
    148   return non_variation_labels;
    149 }
    150 
    151 string16 CombineExperimentLabels(const string16& variation_labels,
    152                                  const string16& other_labels) {
    153   const string16 separator = ASCIIToUTF16(kExperimentLabelSep);
    154   DCHECK(!StartsWith(variation_labels, separator, false));
    155   DCHECK(!EndsWith(variation_labels, separator, false));
    156   DCHECK(!StartsWith(other_labels, separator, false));
    157   DCHECK(!EndsWith(other_labels, separator, false));
    158   // Note that if either label is empty, a separator is not necessary.
    159   string16 combined_labels = other_labels;
    160   if (!other_labels.empty() && !variation_labels.empty())
    161     combined_labels += separator;
    162   combined_labels += variation_labels;
    163   return combined_labels;
    164 }
    165 
    166 // Functions below are exposed for testing explicitly behind this namespace.
    167 // They simply wrap existing functions in this file.
    168 namespace testing {
    169 
    170 void TestGetFieldTrialActiveGroupIds(
    171     const base::FieldTrial::ActiveGroups& active_groups,
    172     std::vector<ActiveGroupId>* name_group_ids) {
    173   GetFieldTrialActiveGroupIdsForActiveGroups(active_groups,
    174                                              name_group_ids);
    175 }
    176 
    177 }  // namespace testing
    178 
    179 }  // namespace chrome_variations
    180