OLD | NEW |
1 // Copyright 2015 The Chromium Authors. All rights reserved. | 1 // Copyright 2015 The Chromium Authors. All rights reserved. |
2 // Use of this source code is governed by a BSD-style license that can be | 2 // Use of this source code is governed by a BSD-style license that can be |
3 // found in the LICENSE file. | 3 // found in the LICENSE file. |
4 | 4 |
5 #include "base/feature_list.h" | 5 #include "base/feature_list.h" |
6 | 6 |
| 7 #include "base/format_macros.h" |
| 8 #include "base/metrics/field_trial.h" |
| 9 #include "base/strings/stringprintf.h" |
7 #include "testing/gtest/include/gtest/gtest.h" | 10 #include "testing/gtest/include/gtest/gtest.h" |
8 | 11 |
9 namespace base { | 12 namespace base { |
10 | 13 |
11 namespace { | 14 namespace { |
12 | 15 |
13 const char kFeatureOnByDefaultName[] = "OnByDefault"; | 16 const char kFeatureOnByDefaultName[] = "OnByDefault"; |
14 struct Feature kFeatureOnByDefault { | 17 struct Feature kFeatureOnByDefault { |
15 kFeatureOnByDefaultName, FEATURE_ENABLED_BY_DEFAULT | 18 kFeatureOnByDefaultName, FEATURE_ENABLED_BY_DEFAULT |
16 }; | 19 }; |
(...skipping 46 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
63 {"OffByDefault", "", true, true}, | 66 {"OffByDefault", "", true, true}, |
64 {"OffByDefault", "OnByDefault", false, true}, | 67 {"OffByDefault", "OnByDefault", false, true}, |
65 {"OnByDefault,OffByDefault", "", true, true}, | 68 {"OnByDefault,OffByDefault", "", true, true}, |
66 {"", "OnByDefault,OffByDefault", false, false}, | 69 {"", "OnByDefault,OffByDefault", false, false}, |
67 // In the case an entry is both, disable takes precedence. | 70 // In the case an entry is both, disable takes precedence. |
68 {"OnByDefault", "OnByDefault,OffByDefault", false, false}, | 71 {"OnByDefault", "OnByDefault,OffByDefault", false, false}, |
69 }; | 72 }; |
70 | 73 |
71 for (size_t i = 0; i < arraysize(test_cases); ++i) { | 74 for (size_t i = 0; i < arraysize(test_cases); ++i) { |
72 const auto& test_case = test_cases[i]; | 75 const auto& test_case = test_cases[i]; |
| 76 SCOPED_TRACE(base::StringPrintf("Test[%" PRIuS "]: [%s] [%s]", i, |
| 77 test_case.enable_features, |
| 78 test_case.disable_features)); |
73 | 79 |
74 ClearFeatureListInstance(); | 80 ClearFeatureListInstance(); |
75 scoped_ptr<FeatureList> feature_list(new FeatureList); | 81 scoped_ptr<FeatureList> feature_list(new FeatureList); |
76 feature_list->InitializeFromCommandLine(test_case.enable_features, | 82 feature_list->InitializeFromCommandLine(test_case.enable_features, |
77 test_case.disable_features); | 83 test_case.disable_features); |
78 RegisterFeatureListInstance(feature_list.Pass()); | 84 RegisterFeatureListInstance(feature_list.Pass()); |
79 | 85 |
80 EXPECT_EQ(test_case.expected_feature_on_state, | 86 EXPECT_EQ(test_case.expected_feature_on_state, |
81 FeatureList::IsEnabled(kFeatureOnByDefault)) | 87 FeatureList::IsEnabled(kFeatureOnByDefault)) |
82 << i; | 88 << i; |
(...skipping 15 matching lines...) Expand all Loading... |
98 EXPECT_TRUE(feature_list()->CheckFeatureIdentity(kFeatureOffByDefault)); | 104 EXPECT_TRUE(feature_list()->CheckFeatureIdentity(kFeatureOffByDefault)); |
99 | 105 |
100 // Now, call it with a distinct struct for |kFeatureOnByDefaultName|, which | 106 // Now, call it with a distinct struct for |kFeatureOnByDefaultName|, which |
101 // should return false. | 107 // should return false. |
102 struct Feature kFeatureOnByDefault2 { | 108 struct Feature kFeatureOnByDefault2 { |
103 kFeatureOnByDefaultName, FEATURE_ENABLED_BY_DEFAULT | 109 kFeatureOnByDefaultName, FEATURE_ENABLED_BY_DEFAULT |
104 }; | 110 }; |
105 EXPECT_FALSE(feature_list()->CheckFeatureIdentity(kFeatureOnByDefault2)); | 111 EXPECT_FALSE(feature_list()->CheckFeatureIdentity(kFeatureOnByDefault2)); |
106 } | 112 } |
107 | 113 |
| 114 TEST_F(FeatureListTest, FieldTrialOverrides) { |
| 115 struct { |
| 116 FeatureList::OverrideState trial1_state; |
| 117 FeatureList::OverrideState trial2_state; |
| 118 } test_cases[] = { |
| 119 {FeatureList::OVERRIDE_DISABLE_FEATURE, |
| 120 FeatureList::OVERRIDE_DISABLE_FEATURE}, |
| 121 {FeatureList::OVERRIDE_DISABLE_FEATURE, |
| 122 FeatureList::OVERRIDE_ENABLE_FEATURE}, |
| 123 {FeatureList::OVERRIDE_ENABLE_FEATURE, |
| 124 FeatureList::OVERRIDE_DISABLE_FEATURE}, |
| 125 {FeatureList::OVERRIDE_ENABLE_FEATURE, |
| 126 FeatureList::OVERRIDE_ENABLE_FEATURE}, |
| 127 }; |
| 128 |
| 129 FieldTrial::ActiveGroup active_group; |
| 130 for (size_t i = 0; i < arraysize(test_cases); ++i) { |
| 131 const auto& test_case = test_cases[i]; |
| 132 SCOPED_TRACE(base::StringPrintf("Test[%" PRIuS "]", i)); |
| 133 |
| 134 ClearFeatureListInstance(); |
| 135 |
| 136 FieldTrialList field_trial_list(nullptr); |
| 137 scoped_ptr<FeatureList> feature_list(new FeatureList); |
| 138 |
| 139 FieldTrial* trial1 = FieldTrialList::CreateFieldTrial("TrialExample1", "A"); |
| 140 FieldTrial* trial2 = FieldTrialList::CreateFieldTrial("TrialExample2", "B"); |
| 141 feature_list->RegisterFieldTrialOverride(kFeatureOnByDefaultName, |
| 142 test_case.trial1_state, trial1); |
| 143 feature_list->RegisterFieldTrialOverride(kFeatureOffByDefaultName, |
| 144 test_case.trial2_state, trial2); |
| 145 RegisterFeatureListInstance(feature_list.Pass()); |
| 146 |
| 147 // Initially, neither trial should be active. |
| 148 EXPECT_FALSE(FieldTrialList::IsTrialActive(trial1->trial_name())); |
| 149 EXPECT_FALSE(FieldTrialList::IsTrialActive(trial2->trial_name())); |
| 150 |
| 151 const bool expected_enabled_1 = |
| 152 (test_case.trial1_state == FeatureList::OVERRIDE_ENABLE_FEATURE); |
| 153 EXPECT_EQ(expected_enabled_1, FeatureList::IsEnabled(kFeatureOnByDefault)); |
| 154 // The above should have activated |trial1|. |
| 155 EXPECT_TRUE(FieldTrialList::IsTrialActive(trial1->trial_name())); |
| 156 EXPECT_FALSE(FieldTrialList::IsTrialActive(trial2->trial_name())); |
| 157 |
| 158 const bool expected_enabled_2 = |
| 159 (test_case.trial2_state == FeatureList::OVERRIDE_ENABLE_FEATURE); |
| 160 EXPECT_EQ(expected_enabled_2, FeatureList::IsEnabled(kFeatureOffByDefault)); |
| 161 // The above should have activated |trial2|. |
| 162 EXPECT_TRUE(FieldTrialList::IsTrialActive(trial1->trial_name())); |
| 163 EXPECT_TRUE(FieldTrialList::IsTrialActive(trial2->trial_name())); |
| 164 } |
| 165 } |
| 166 |
| 167 TEST_F(FeatureListTest, CommandLineTakesPrecedenceOverFieldTrial) { |
| 168 ClearFeatureListInstance(); |
| 169 |
| 170 FieldTrialList field_trial_list(nullptr); |
| 171 scoped_ptr<FeatureList> feature_list(new FeatureList); |
| 172 |
| 173 // The feature is explicitly enabled on the command-line. |
| 174 feature_list->InitializeFromCommandLine(kFeatureOffByDefaultName, ""); |
| 175 |
| 176 // But the FieldTrial would set the feature to disabled. |
| 177 FieldTrial* trial = FieldTrialList::CreateFieldTrial("TrialExample2", "A"); |
| 178 feature_list->RegisterFieldTrialOverride( |
| 179 kFeatureOffByDefaultName, FeatureList::OVERRIDE_DISABLE_FEATURE, trial); |
| 180 RegisterFeatureListInstance(feature_list.Pass()); |
| 181 |
| 182 EXPECT_FALSE(FieldTrialList::IsTrialActive(trial->trial_name())); |
| 183 // Command-line should take precedence. |
| 184 EXPECT_TRUE(FeatureList::IsEnabled(kFeatureOffByDefault)); |
| 185 // Since the feature is on due to the command-line, and not as a result of the |
| 186 // field trial, the field trial should not be activated (since the Associate* |
| 187 // API wasn't used.) |
| 188 EXPECT_FALSE(FieldTrialList::IsTrialActive(trial->trial_name())); |
| 189 } |
| 190 |
| 191 TEST_F(FeatureListTest, IsFeatureOverriddenFromCommandLine) { |
| 192 ClearFeatureListInstance(); |
| 193 |
| 194 FieldTrialList field_trial_list(nullptr); |
| 195 scoped_ptr<FeatureList> feature_list(new FeatureList); |
| 196 |
| 197 // No features are overridden from the command line yet |
| 198 EXPECT_FALSE(feature_list->IsFeatureOverriddenFromCommandLine( |
| 199 kFeatureOnByDefaultName, FeatureList::OVERRIDE_DISABLE_FEATURE)); |
| 200 EXPECT_FALSE(feature_list->IsFeatureOverriddenFromCommandLine( |
| 201 kFeatureOnByDefaultName, FeatureList::OVERRIDE_ENABLE_FEATURE)); |
| 202 EXPECT_FALSE(feature_list->IsFeatureOverriddenFromCommandLine( |
| 203 kFeatureOffByDefaultName, FeatureList::OVERRIDE_DISABLE_FEATURE)); |
| 204 EXPECT_FALSE(feature_list->IsFeatureOverriddenFromCommandLine( |
| 205 kFeatureOffByDefaultName, FeatureList::OVERRIDE_ENABLE_FEATURE)); |
| 206 |
| 207 // Now, enable |kFeatureOffByDefaultName| via the command-line. |
| 208 feature_list->InitializeFromCommandLine(kFeatureOffByDefaultName, ""); |
| 209 |
| 210 // It should now be overridden for the enabled group. |
| 211 EXPECT_FALSE(feature_list->IsFeatureOverriddenFromCommandLine( |
| 212 kFeatureOffByDefaultName, FeatureList::OVERRIDE_DISABLE_FEATURE)); |
| 213 EXPECT_TRUE(feature_list->IsFeatureOverriddenFromCommandLine( |
| 214 kFeatureOffByDefaultName, FeatureList::OVERRIDE_ENABLE_FEATURE)); |
| 215 |
| 216 // Register a field trial to associate with the feature and ensure that the |
| 217 // results are still the same. |
| 218 feature_list->AssociateReportingFieldTrial( |
| 219 kFeatureOffByDefaultName, FeatureList::OVERRIDE_ENABLE_FEATURE, |
| 220 FieldTrialList::CreateFieldTrial("Trial1", "A")); |
| 221 EXPECT_FALSE(feature_list->IsFeatureOverriddenFromCommandLine( |
| 222 kFeatureOffByDefaultName, FeatureList::OVERRIDE_DISABLE_FEATURE)); |
| 223 EXPECT_TRUE(feature_list->IsFeatureOverriddenFromCommandLine( |
| 224 kFeatureOffByDefaultName, FeatureList::OVERRIDE_ENABLE_FEATURE)); |
| 225 |
| 226 // Now, register a field trial to override |kFeatureOnByDefaultName| state |
| 227 // and check that the function still returns false for that feature. |
| 228 feature_list->RegisterFieldTrialOverride( |
| 229 kFeatureOnByDefaultName, FeatureList::OVERRIDE_DISABLE_FEATURE, |
| 230 FieldTrialList::CreateFieldTrial("Trial2", "A")); |
| 231 EXPECT_FALSE(feature_list->IsFeatureOverriddenFromCommandLine( |
| 232 kFeatureOnByDefaultName, FeatureList::OVERRIDE_DISABLE_FEATURE)); |
| 233 EXPECT_FALSE(feature_list->IsFeatureOverriddenFromCommandLine( |
| 234 kFeatureOnByDefaultName, FeatureList::OVERRIDE_ENABLE_FEATURE)); |
| 235 RegisterFeatureListInstance(feature_list.Pass()); |
| 236 |
| 237 // Check the expected feature states for good measure. |
| 238 EXPECT_TRUE(FeatureList::IsEnabled(kFeatureOffByDefault)); |
| 239 EXPECT_FALSE(FeatureList::IsEnabled(kFeatureOnByDefault)); |
| 240 } |
| 241 |
| 242 TEST_F(FeatureListTest, AssociateReportingFieldTrial) { |
| 243 struct { |
| 244 const char* enable_features; |
| 245 const char* disable_features; |
| 246 bool expected_enable_trial_created; |
| 247 bool expected_disable_trial_created; |
| 248 } test_cases[] = { |
| 249 // If no enable/disable flags are specified, no trials should be created. |
| 250 {"", "", false, false}, |
| 251 // Enabling the feature should result in the enable trial created. |
| 252 {kFeatureOffByDefaultName, "", true, false}, |
| 253 // Disabling the feature should result in the disable trial created. |
| 254 {"", kFeatureOffByDefaultName, false, true}, |
| 255 }; |
| 256 |
| 257 const char kTrialName[] = "ForcingTrial"; |
| 258 const char kForcedOnGroupName[] = "ForcedOn"; |
| 259 const char kForcedOffGroupName[] = "ForcedOff"; |
| 260 |
| 261 for (size_t i = 0; i < arraysize(test_cases); ++i) { |
| 262 const auto& test_case = test_cases[i]; |
| 263 SCOPED_TRACE(base::StringPrintf("Test[%" PRIuS "]: [%s] [%s]", i, |
| 264 test_case.enable_features, |
| 265 test_case.disable_features)); |
| 266 |
| 267 ClearFeatureListInstance(); |
| 268 |
| 269 FieldTrialList field_trial_list(nullptr); |
| 270 scoped_ptr<FeatureList> feature_list(new FeatureList); |
| 271 feature_list->InitializeFromCommandLine(test_case.enable_features, |
| 272 test_case.disable_features); |
| 273 |
| 274 FieldTrial* enable_trial = nullptr; |
| 275 if (feature_list->IsFeatureOverriddenFromCommandLine( |
| 276 kFeatureOffByDefaultName, FeatureList::OVERRIDE_ENABLE_FEATURE)) { |
| 277 enable_trial = base::FieldTrialList::CreateFieldTrial(kTrialName, |
| 278 kForcedOnGroupName); |
| 279 feature_list->AssociateReportingFieldTrial( |
| 280 kFeatureOffByDefaultName, FeatureList::OVERRIDE_ENABLE_FEATURE, |
| 281 enable_trial); |
| 282 } |
| 283 FieldTrial* disable_trial = nullptr; |
| 284 if (feature_list->IsFeatureOverriddenFromCommandLine( |
| 285 kFeatureOffByDefaultName, FeatureList::OVERRIDE_DISABLE_FEATURE)) { |
| 286 disable_trial = base::FieldTrialList::CreateFieldTrial( |
| 287 kTrialName, kForcedOffGroupName); |
| 288 feature_list->AssociateReportingFieldTrial( |
| 289 kFeatureOffByDefaultName, FeatureList::OVERRIDE_DISABLE_FEATURE, |
| 290 disable_trial); |
| 291 } |
| 292 EXPECT_EQ(test_case.expected_enable_trial_created, enable_trial != nullptr); |
| 293 EXPECT_EQ(test_case.expected_disable_trial_created, |
| 294 disable_trial != nullptr); |
| 295 RegisterFeatureListInstance(feature_list.Pass()); |
| 296 |
| 297 EXPECT_FALSE(FieldTrialList::IsTrialActive(kTrialName)); |
| 298 if (disable_trial) { |
| 299 EXPECT_FALSE(FeatureList::IsEnabled(kFeatureOffByDefault)); |
| 300 EXPECT_TRUE(FieldTrialList::IsTrialActive(kTrialName)); |
| 301 EXPECT_EQ(kForcedOffGroupName, disable_trial->group_name()); |
| 302 } else if (enable_trial) { |
| 303 EXPECT_TRUE(FeatureList::IsEnabled(kFeatureOffByDefault)); |
| 304 EXPECT_TRUE(FieldTrialList::IsTrialActive(kTrialName)); |
| 305 EXPECT_EQ(kForcedOnGroupName, enable_trial->group_name()); |
| 306 } |
| 307 } |
| 308 } |
| 309 |
108 } // namespace base | 310 } // namespace base |
OLD | NEW |