// 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 "base/feature_list.h" #include "base/macros.h" #include "base/metrics/field_trial.h" #include "base/metrics/field_trial_param_associator.h" #include "base/test/scoped_feature_list.h" #include "testing/gtest/include/gtest/gtest.h" namespace base { namespace { // Call FieldTrialList::FactoryGetFieldTrial() with a future expiry date. scoped_refptr CreateFieldTrial( const std::string& trial_name, int total_probability, const std::string& default_group_name, int* default_group_number) { return FieldTrialList::FactoryGetFieldTrial( trial_name, total_probability, default_group_name, FieldTrialList::kNoExpirationYear, 1, 1, FieldTrial::SESSION_RANDOMIZED, default_group_number); } } // namespace class FieldTrialParamsTest : public ::testing::Test { public: FieldTrialParamsTest() : field_trial_list_(nullptr) {} ~FieldTrialParamsTest() override { // Ensure that the maps are cleared between tests, since they are stored as // process singletons. FieldTrialParamAssociator::GetInstance()->ClearAllParamsForTesting(); } void CreateFeatureWithTrial(const Feature& feature, FeatureList::OverrideState override_state, FieldTrial* trial) { std::unique_ptr feature_list(new FeatureList); feature_list->RegisterFieldTrialOverride(feature.name, override_state, trial); scoped_feature_list_.InitWithFeatureList(std::move(feature_list)); } private: FieldTrialList field_trial_list_; test::ScopedFeatureList scoped_feature_list_; DISALLOW_COPY_AND_ASSIGN(FieldTrialParamsTest); }; TEST_F(FieldTrialParamsTest, AssociateFieldTrialParams) { const std::string kTrialName = "AssociateFieldTrialParams"; { std::map params; params["a"] = "10"; params["b"] = "test"; ASSERT_TRUE(AssociateFieldTrialParams(kTrialName, "A", params)); } { std::map params; params["a"] = "5"; ASSERT_TRUE(AssociateFieldTrialParams(kTrialName, "B", params)); } FieldTrialList::CreateFieldTrial(kTrialName, "B"); EXPECT_EQ("5", GetFieldTrialParamValue(kTrialName, "a")); EXPECT_EQ(std::string(), GetFieldTrialParamValue(kTrialName, "b")); EXPECT_EQ(std::string(), GetFieldTrialParamValue(kTrialName, "x")); std::map params; EXPECT_TRUE(GetFieldTrialParams(kTrialName, ¶ms)); EXPECT_EQ(1U, params.size()); EXPECT_EQ("5", params["a"]); } TEST_F(FieldTrialParamsTest, AssociateFieldTrialParams_Fail) { const std::string kTrialName = "AssociateFieldTrialParams_Fail"; const std::string kGroupName = "A"; std::map params; params["a"] = "10"; ASSERT_TRUE(AssociateFieldTrialParams(kTrialName, kGroupName, params)); params["a"] = "1"; params["b"] = "2"; ASSERT_FALSE(AssociateFieldTrialParams(kTrialName, kGroupName, params)); FieldTrialList::CreateFieldTrial(kTrialName, kGroupName); EXPECT_EQ("10", GetFieldTrialParamValue(kTrialName, "a")); EXPECT_EQ(std::string(), GetFieldTrialParamValue(kTrialName, "b")); } TEST_F(FieldTrialParamsTest, AssociateFieldTrialParams_TrialActiveFail) { const std::string kTrialName = "AssociateFieldTrialParams_TrialActiveFail"; FieldTrialList::CreateFieldTrial(kTrialName, "A"); ASSERT_EQ("A", FieldTrialList::FindFullName(kTrialName)); std::map params; params["a"] = "10"; EXPECT_FALSE(AssociateFieldTrialParams(kTrialName, "B", params)); EXPECT_FALSE(AssociateFieldTrialParams(kTrialName, "A", params)); } TEST_F(FieldTrialParamsTest, AssociateFieldTrialParams_DoesntActivateTrial) { const std::string kTrialName = "AssociateFieldTrialParams_DoesntActivateTrial"; ASSERT_FALSE(FieldTrialList::IsTrialActive(kTrialName)); scoped_refptr trial( CreateFieldTrial(kTrialName, 100, "A", nullptr)); ASSERT_FALSE(FieldTrialList::IsTrialActive(kTrialName)); std::map params; params["a"] = "10"; EXPECT_TRUE(AssociateFieldTrialParams(kTrialName, "A", params)); ASSERT_FALSE(FieldTrialList::IsTrialActive(kTrialName)); } TEST_F(FieldTrialParamsTest, GetFieldTrialParams_NoTrial) { const std::string kTrialName = "GetFieldTrialParams_NoParams"; std::map params; EXPECT_FALSE(GetFieldTrialParams(kTrialName, ¶ms)); EXPECT_EQ(std::string(), GetFieldTrialParamValue(kTrialName, "x")); EXPECT_EQ(std::string(), GetFieldTrialParamValue(kTrialName, "y")); } TEST_F(FieldTrialParamsTest, GetFieldTrialParams_NoParams) { const std::string kTrialName = "GetFieldTrialParams_NoParams"; FieldTrialList::CreateFieldTrial(kTrialName, "A"); std::map params; EXPECT_FALSE(GetFieldTrialParams(kTrialName, ¶ms)); EXPECT_EQ(std::string(), GetFieldTrialParamValue(kTrialName, "x")); EXPECT_EQ(std::string(), GetFieldTrialParamValue(kTrialName, "y")); } TEST_F(FieldTrialParamsTest, GetFieldTrialParams_ActivatesTrial) { const std::string kTrialName = "GetFieldTrialParams_ActivatesTrial"; ASSERT_FALSE(FieldTrialList::IsTrialActive(kTrialName)); scoped_refptr trial( CreateFieldTrial(kTrialName, 100, "A", nullptr)); ASSERT_FALSE(FieldTrialList::IsTrialActive(kTrialName)); std::map params; EXPECT_FALSE(GetFieldTrialParams(kTrialName, ¶ms)); ASSERT_TRUE(FieldTrialList::IsTrialActive(kTrialName)); } TEST_F(FieldTrialParamsTest, GetFieldTrialParamValue_ActivatesTrial) { const std::string kTrialName = "GetFieldTrialParamValue_ActivatesTrial"; ASSERT_FALSE(FieldTrialList::IsTrialActive(kTrialName)); scoped_refptr trial( CreateFieldTrial(kTrialName, 100, "A", nullptr)); ASSERT_FALSE(FieldTrialList::IsTrialActive(kTrialName)); std::map params; EXPECT_EQ(std::string(), GetFieldTrialParamValue(kTrialName, "x")); ASSERT_TRUE(FieldTrialList::IsTrialActive(kTrialName)); } TEST_F(FieldTrialParamsTest, GetFieldTrialParamsByFeature) { const std::string kTrialName = "GetFieldTrialParamsByFeature"; const Feature kFeature{"TestFeature", FEATURE_DISABLED_BY_DEFAULT}; std::map params; params["x"] = "1"; AssociateFieldTrialParams(kTrialName, "A", params); scoped_refptr trial( CreateFieldTrial(kTrialName, 100, "A", nullptr)); CreateFeatureWithTrial(kFeature, FeatureList::OVERRIDE_ENABLE_FEATURE, trial.get()); std::map actualParams; EXPECT_TRUE(GetFieldTrialParamsByFeature(kFeature, &actualParams)); EXPECT_EQ(params, actualParams); } TEST_F(FieldTrialParamsTest, GetFieldTrialParamValueByFeature) { const std::string kTrialName = "GetFieldTrialParamsByFeature"; const Feature kFeature{"TestFeature", FEATURE_DISABLED_BY_DEFAULT}; std::map params; params["x"] = "1"; AssociateFieldTrialParams(kTrialName, "A", params); scoped_refptr trial( CreateFieldTrial(kTrialName, 100, "A", nullptr)); CreateFeatureWithTrial(kFeature, FeatureList::OVERRIDE_ENABLE_FEATURE, trial.get()); std::map actualParams; EXPECT_EQ(params["x"], GetFieldTrialParamValueByFeature(kFeature, "x")); } TEST_F(FieldTrialParamsTest, GetFieldTrialParamsByFeature_Disable) { const std::string kTrialName = "GetFieldTrialParamsByFeature"; const Feature kFeature{"TestFeature", FEATURE_DISABLED_BY_DEFAULT}; std::map params; params["x"] = "1"; AssociateFieldTrialParams(kTrialName, "A", params); scoped_refptr trial( CreateFieldTrial(kTrialName, 100, "A", nullptr)); CreateFeatureWithTrial(kFeature, FeatureList::OVERRIDE_DISABLE_FEATURE, trial.get()); std::map actualParams; EXPECT_FALSE(GetFieldTrialParamsByFeature(kFeature, &actualParams)); } TEST_F(FieldTrialParamsTest, GetFieldTrialParamValueByFeature_Disable) { const std::string kTrialName = "GetFieldTrialParamsByFeature"; const Feature kFeature{"TestFeature", FEATURE_DISABLED_BY_DEFAULT}; std::map params; params["x"] = "1"; AssociateFieldTrialParams(kTrialName, "A", params); scoped_refptr trial( CreateFieldTrial(kTrialName, 100, "A", nullptr)); CreateFeatureWithTrial(kFeature, FeatureList::OVERRIDE_DISABLE_FEATURE, trial.get()); std::map actualParams; EXPECT_EQ(std::string(), GetFieldTrialParamValueByFeature(kFeature, "x")); } TEST_F(FieldTrialParamsTest, FeatureParamString) { const std::string kTrialName = "GetFieldTrialParamsByFeature"; static const Feature kFeature{"TestFeature", FEATURE_DISABLED_BY_DEFAULT}; static const FeatureParam a{&kFeature, "a", "default"}; static const FeatureParam b{&kFeature, "b", ""}; static const FeatureParam c{&kFeature, "c", "default"}; static const FeatureParam d{&kFeature, "d", ""}; static const FeatureParam e{&kFeature, "e", "default"}; static const FeatureParam f{&kFeature, "f", ""}; std::map params; params["a"] = ""; params["b"] = "non-default"; params["c"] = "non-default"; params["d"] = ""; // "e" is not registered // "f" is not registered AssociateFieldTrialParams(kTrialName, "A", params); scoped_refptr trial( CreateFieldTrial(kTrialName, 100, "A", nullptr)); CreateFeatureWithTrial(kFeature, FeatureList::OVERRIDE_ENABLE_FEATURE, trial.get()); EXPECT_EQ("default", a.Get()); // empty EXPECT_EQ("non-default", b.Get()); EXPECT_EQ("non-default", c.Get()); EXPECT_EQ("", d.Get()); // empty EXPECT_EQ("default", e.Get()); // not registered EXPECT_EQ("", f.Get()); // not registered } TEST_F(FieldTrialParamsTest, FeatureParamInt) { const std::string kTrialName = "GetFieldTrialParamsByFeature"; static const Feature kFeature{"TestFeature", FEATURE_DISABLED_BY_DEFAULT}; static const FeatureParam a{&kFeature, "a", 0}; static const FeatureParam b{&kFeature, "b", 0}; static const FeatureParam c{&kFeature, "c", 0}; static const FeatureParam d{&kFeature, "d", 0}; static const FeatureParam e{&kFeature, "e", 0}; std::map params; params["a"] = "1"; params["b"] = "1.5"; params["c"] = "foo"; params["d"] = ""; // "e" is not registered AssociateFieldTrialParams(kTrialName, "A", params); scoped_refptr trial( CreateFieldTrial(kTrialName, 100, "A", nullptr)); CreateFeatureWithTrial(kFeature, FeatureList::OVERRIDE_ENABLE_FEATURE, trial.get()); EXPECT_EQ(1, GetFieldTrialParamByFeatureAsInt(kFeature, "a", 0)); EXPECT_EQ(0, GetFieldTrialParamByFeatureAsInt(kFeature, "b", 0)); // invalid EXPECT_EQ(0, GetFieldTrialParamByFeatureAsInt(kFeature, "c", 0)); // invalid EXPECT_EQ(0, GetFieldTrialParamByFeatureAsInt(kFeature, "d", 0)); // empty EXPECT_EQ(0, GetFieldTrialParamByFeatureAsInt(kFeature, "e", 0)); // empty EXPECT_EQ(1, a.Get()); EXPECT_EQ(0, b.Get()); // invalid EXPECT_EQ(0, c.Get()); // invalid EXPECT_EQ(0, d.Get()); // empty EXPECT_EQ(0, e.Get()); // empty } TEST_F(FieldTrialParamsTest, FeatureParamDouble) { const std::string kTrialName = "GetFieldTrialParamsByFeature"; static const Feature kFeature{"TestFeature", FEATURE_DISABLED_BY_DEFAULT}; static const FeatureParam a{&kFeature, "a", 0.0}; static const FeatureParam b{&kFeature, "b", 0.0}; static const FeatureParam c{&kFeature, "c", 0.0}; static const FeatureParam d{&kFeature, "d", 0.0}; static const FeatureParam e{&kFeature, "e", 0.0}; static const FeatureParam f{&kFeature, "f", 0.0}; std::map params; params["a"] = "1"; params["b"] = "1.5"; params["c"] = "1.0e-10"; params["d"] = "foo"; params["e"] = ""; // "f" is not registered AssociateFieldTrialParams(kTrialName, "A", params); scoped_refptr trial( CreateFieldTrial(kTrialName, 100, "A", nullptr)); CreateFeatureWithTrial(kFeature, FeatureList::OVERRIDE_ENABLE_FEATURE, trial.get()); EXPECT_EQ(1, GetFieldTrialParamByFeatureAsDouble(kFeature, "a", 0)); EXPECT_EQ(1.5, GetFieldTrialParamByFeatureAsDouble(kFeature, "b", 0)); EXPECT_EQ(1.0e-10, GetFieldTrialParamByFeatureAsDouble(kFeature, "c", 0)); EXPECT_EQ(0, GetFieldTrialParamByFeatureAsDouble(kFeature, "d", 0)); // invalid EXPECT_EQ(0, GetFieldTrialParamByFeatureAsDouble(kFeature, "e", 0)); // empty EXPECT_EQ(0, GetFieldTrialParamByFeatureAsDouble(kFeature, "f", 0)); // empty EXPECT_EQ(1, a.Get()); EXPECT_EQ(1.5, b.Get()); EXPECT_EQ(1.0e-10, c.Get()); EXPECT_EQ(0, d.Get()); // invalid EXPECT_EQ(0, e.Get()); // empty EXPECT_EQ(0, f.Get()); // empty } TEST_F(FieldTrialParamsTest, FeatureParamBool) { const std::string kTrialName = "GetFieldTrialParamsByFeature"; static const Feature kFeature{"TestFeature", FEATURE_DISABLED_BY_DEFAULT}; static const FeatureParam a{&kFeature, "a", false}; static const FeatureParam b{&kFeature, "b", true}; static const FeatureParam c{&kFeature, "c", false}; static const FeatureParam d{&kFeature, "d", true}; static const FeatureParam e{&kFeature, "e", true}; static const FeatureParam f{&kFeature, "f", true}; std::map params; params["a"] = "true"; params["b"] = "false"; params["c"] = "1"; params["d"] = "False"; params["e"] = ""; // "f" is not registered AssociateFieldTrialParams(kTrialName, "A", params); scoped_refptr trial( CreateFieldTrial(kTrialName, 100, "A", nullptr)); CreateFeatureWithTrial(kFeature, FeatureList::OVERRIDE_ENABLE_FEATURE, trial.get()); EXPECT_TRUE(a.Get()); EXPECT_FALSE(b.Get()); EXPECT_FALSE(c.Get()); // invalid EXPECT_TRUE(d.Get()); // invalid EXPECT_TRUE(e.Get()); // empty EXPECT_TRUE(f.Get()); // empty } enum Hand { ROCK, PAPER, SCISSORS }; TEST_F(FieldTrialParamsTest, FeatureParamEnum) { const std::string kTrialName = "GetFieldTrialParamsByFeature"; static const FeatureParam::Option hands[] = { {ROCK, "rock"}, {PAPER, "paper"}, {SCISSORS, "scissors"}}; static const Feature kFeature{"TestFeature", FEATURE_DISABLED_BY_DEFAULT}; static const FeatureParam a{&kFeature, "a", ROCK, &hands}; static const FeatureParam b{&kFeature, "b", ROCK, &hands}; static const FeatureParam c{&kFeature, "c", ROCK, &hands}; static const FeatureParam d{&kFeature, "d", ROCK, &hands}; static const FeatureParam e{&kFeature, "e", PAPER, &hands}; static const FeatureParam f{&kFeature, "f", SCISSORS, &hands}; std::map params; params["a"] = "rock"; params["b"] = "paper"; params["c"] = "scissors"; params["d"] = "lizard"; params["e"] = ""; // "f" is not registered AssociateFieldTrialParams(kTrialName, "A", params); scoped_refptr trial( CreateFieldTrial(kTrialName, 100, "A", nullptr)); CreateFeatureWithTrial(kFeature, FeatureList::OVERRIDE_ENABLE_FEATURE, trial.get()); EXPECT_EQ(ROCK, a.Get()); EXPECT_EQ(PAPER, b.Get()); EXPECT_EQ(SCISSORS, c.Get()); EXPECT_EQ(ROCK, d.Get()); // invalid EXPECT_EQ(PAPER, e.Get()); // invalid/empty EXPECT_EQ(SCISSORS, f.Get()); // not registered } enum class UI { ONE_D, TWO_D, THREE_D }; TEST_F(FieldTrialParamsTest, FeatureParamEnumClass) { const std::string kTrialName = "GetFieldTrialParamsByFeature"; static const FeatureParam::Option uis[] = { {UI::ONE_D, "1d"}, {UI::TWO_D, "2d"}, {UI::THREE_D, "3d"}}; static const Feature kFeature{"TestFeature", FEATURE_DISABLED_BY_DEFAULT}; static const FeatureParam a{&kFeature, "a", UI::ONE_D, &uis}; static const FeatureParam b{&kFeature, "b", UI::ONE_D, &uis}; static const FeatureParam c{&kFeature, "c", UI::ONE_D, &uis}; static const FeatureParam d{&kFeature, "d", UI::ONE_D, &uis}; static const FeatureParam e{&kFeature, "e", UI::TWO_D, &uis}; static const FeatureParam f{&kFeature, "f", UI::THREE_D, &uis}; std::map params; params["a"] = "1d"; params["b"] = "2d"; params["c"] = "3d"; params["d"] = "4d"; params["e"] = ""; // "f" is not registered AssociateFieldTrialParams(kTrialName, "A", params); scoped_refptr trial( CreateFieldTrial(kTrialName, 100, "A", nullptr)); CreateFeatureWithTrial(kFeature, FeatureList::OVERRIDE_ENABLE_FEATURE, trial.get()); EXPECT_EQ(UI::ONE_D, a.Get()); EXPECT_EQ(UI::TWO_D, b.Get()); EXPECT_EQ(UI::THREE_D, c.Get()); EXPECT_EQ(UI::ONE_D, d.Get()); // invalid EXPECT_EQ(UI::TWO_D, e.Get()); // invalid/empty EXPECT_EQ(UI::THREE_D, f.Get()); // not registered } } // namespace base