// Copyright 2017 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 "base/metrics/field_trial_params.h" #include #include #include #include "base/feature_list.h" #include "base/metrics/field_trial.h" #include "base/metrics/field_trial_param_associator.h" #include "base/strings/string_number_conversions.h" #include "base/strings/string_split.h" #include "base/strings/stringprintf.h" namespace base { bool AssociateFieldTrialParams(const std::string& trial_name, const std::string& group_name, const FieldTrialParams& params) { return FieldTrialParamAssociator::GetInstance()->AssociateFieldTrialParams( trial_name, group_name, params); } bool AssociateFieldTrialParamsFromString( const std::string& params_string, FieldTrialParamsDecodeStringFunc decode_data_func) { // Format: Trial1.Group1:k1/v1/k2/v2,Trial2.Group2:k1/v1/k2/v2 std::set> trial_groups; for (StringPiece experiment_group : SplitStringPiece(params_string, ",", TRIM_WHITESPACE, SPLIT_WANT_ALL)) { std::vector experiment = SplitStringPiece( experiment_group, ":", TRIM_WHITESPACE, SPLIT_WANT_ALL); if (experiment.size() != 2) { DLOG(ERROR) << "Experiment and params should be separated by ':'"; return false; } std::vector group_parts = SplitString(experiment[0], ".", TRIM_WHITESPACE, SPLIT_WANT_ALL); if (group_parts.size() != 2) { DLOG(ERROR) << "Trial and group name should be separated by '.'"; return false; } std::vector key_values = SplitString(experiment[1], "/", TRIM_WHITESPACE, SPLIT_WANT_ALL); if (key_values.size() % 2 != 0) { DLOG(ERROR) << "Param name and param value should be separated by '/'"; return false; } std::string trial = decode_data_func(group_parts[0]); std::string group = decode_data_func(group_parts[1]); auto trial_group = std::make_pair(trial, group); if (trial_groups.find(trial_group) != trial_groups.end()) { DLOG(ERROR) << StringPrintf( "A (trial, group) pair listed more than once. (%s, %s)", trial.c_str(), group.c_str()); return false; } trial_groups.insert(trial_group); std::map params; for (size_t i = 0; i < key_values.size(); i += 2) { std::string key = decode_data_func(key_values[i]); std::string value = decode_data_func(key_values[i + 1]); params[key] = value; } bool result = AssociateFieldTrialParams(trial, group, params); if (!result) { DLOG(ERROR) << "Failed to associate field trial params for group \"" << group << "\" in trial \"" << trial << "\""; return false; } } return true; } bool GetFieldTrialParams(const std::string& trial_name, FieldTrialParams* params) { return FieldTrialParamAssociator::GetInstance()->GetFieldTrialParams( trial_name, params); } bool GetFieldTrialParamsByFeature(const Feature& feature, FieldTrialParams* params) { if (!FeatureList::IsEnabled(feature)) return false; FieldTrial* trial = FeatureList::GetFieldTrial(feature); if (!trial) return false; return GetFieldTrialParams(trial->trial_name(), params); } std::string GetFieldTrialParamValue(const std::string& trial_name, const std::string& param_name) { FieldTrialParams params; if (GetFieldTrialParams(trial_name, ¶ms)) { auto it = params.find(param_name); if (it != params.end()) return it->second; } return std::string(); } std::string GetFieldTrialParamValueByFeature(const Feature& feature, const std::string& param_name) { if (!FeatureList::IsEnabled(feature)) return std::string(); FieldTrial* trial = FeatureList::GetFieldTrial(feature); if (!trial) return std::string(); return GetFieldTrialParamValue(trial->trial_name(), param_name); } int GetFieldTrialParamByFeatureAsInt(const Feature& feature, const std::string& param_name, int default_value) { std::string value_as_string = GetFieldTrialParamValueByFeature(feature, param_name); int value_as_int = 0; if (!StringToInt(value_as_string, &value_as_int)) { if (!value_as_string.empty()) { DLOG(WARNING) << "Failed to parse field trial param " << param_name << " with string value " << value_as_string << " under feature " << feature.name << " into an int. Falling back to default value of " << default_value; } value_as_int = default_value; } return value_as_int; } double GetFieldTrialParamByFeatureAsDouble(const Feature& feature, const std::string& param_name, double default_value) { std::string value_as_string = GetFieldTrialParamValueByFeature(feature, param_name); double value_as_double = 0; if (!StringToDouble(value_as_string, &value_as_double)) { if (!value_as_string.empty()) { DLOG(WARNING) << "Failed to parse field trial param " << param_name << " with string value " << value_as_string << " under feature " << feature.name << " into a double. Falling back to default value of " << default_value; } value_as_double = default_value; } return value_as_double; } bool GetFieldTrialParamByFeatureAsBool(const Feature& feature, const std::string& param_name, bool default_value) { std::string value_as_string = GetFieldTrialParamValueByFeature(feature, param_name); if (value_as_string == "true") return true; if (value_as_string == "false") return false; if (!value_as_string.empty()) { DLOG(WARNING) << "Failed to parse field trial param " << param_name << " with string value " << value_as_string << " under feature " << feature.name << " into a bool. Falling back to default value of " << default_value; } return default_value; } std::string FeatureParam::Get() const { const std::string value = GetFieldTrialParamValueByFeature(*feature, name); return value.empty() ? default_value : value; } double FeatureParam::Get() const { return GetFieldTrialParamByFeatureAsDouble(*feature, name, default_value); } int FeatureParam::Get() const { return GetFieldTrialParamByFeatureAsInt(*feature, name, default_value); } bool FeatureParam::Get() const { return GetFieldTrialParamByFeatureAsBool(*feature, name, default_value); } void LogInvalidEnumValue(const Feature& feature, const std::string& param_name, const std::string& value_as_string, int default_value_as_int) { DLOG(WARNING) << "Failed to parse field trial param " << param_name << " with string value " << value_as_string << " under feature " << feature.name << " into an enum. Falling back to default value of " << default_value_as_int; } } // namespace base