• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
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
3 // found in the LICENSE file.
4 
5 #include "base/feature_list.h"
6 
7 #include <stddef.h>
8 
9 #include <algorithm>
10 #include <utility>
11 
12 #include "base/format_macros.h"
13 #include "base/macros.h"
14 #include "base/memory/ptr_util.h"
15 #include "base/metrics/field_trial.h"
16 #include "base/metrics/persistent_memory_allocator.h"
17 #include "base/strings/string_piece.h"
18 #include "base/strings/string_util.h"
19 #include "base/strings/stringprintf.h"
20 #include "testing/gtest/include/gtest/gtest.h"
21 
22 namespace base {
23 
24 namespace {
25 
26 constexpr char kFeatureOnByDefaultName[] = "OnByDefault";
27 struct Feature kFeatureOnByDefault {
28   kFeatureOnByDefaultName, FEATURE_ENABLED_BY_DEFAULT
29 };
30 
31 constexpr char kFeatureOffByDefaultName[] = "OffByDefault";
32 struct Feature kFeatureOffByDefault {
33   kFeatureOffByDefaultName, FEATURE_DISABLED_BY_DEFAULT
34 };
35 
SortFeatureListString(const std::string & feature_list)36 std::string SortFeatureListString(const std::string& feature_list) {
37   std::vector<base::StringPiece> features =
38       FeatureList::SplitFeatureListString(feature_list);
39   std::sort(features.begin(), features.end());
40   return JoinString(features, ",");
41 }
42 
43 }  // namespace
44 
45 class FeatureListTest : public testing::Test {
46  public:
FeatureListTest()47   FeatureListTest() : feature_list_(nullptr) {
48     RegisterFeatureListInstance(WrapUnique(new FeatureList));
49   }
~FeatureListTest()50   ~FeatureListTest() override { ClearFeatureListInstance(); }
51 
RegisterFeatureListInstance(std::unique_ptr<FeatureList> feature_list)52   void RegisterFeatureListInstance(std::unique_ptr<FeatureList> feature_list) {
53     FeatureList::ClearInstanceForTesting();
54     feature_list_ = feature_list.get();
55     FeatureList::SetInstance(std::move(feature_list));
56   }
ClearFeatureListInstance()57   void ClearFeatureListInstance() {
58     FeatureList::ClearInstanceForTesting();
59     feature_list_ = nullptr;
60   }
61 
feature_list()62   FeatureList* feature_list() { return feature_list_; }
63 
64  private:
65   // Weak. Owned by the FeatureList::SetInstance().
66   FeatureList* feature_list_;
67 
68   DISALLOW_COPY_AND_ASSIGN(FeatureListTest);
69 };
70 
TEST_F(FeatureListTest,DefaultStates)71 TEST_F(FeatureListTest, DefaultStates) {
72   EXPECT_TRUE(FeatureList::IsEnabled(kFeatureOnByDefault));
73   EXPECT_FALSE(FeatureList::IsEnabled(kFeatureOffByDefault));
74 }
75 
TEST_F(FeatureListTest,InitializeFromCommandLine)76 TEST_F(FeatureListTest, InitializeFromCommandLine) {
77   struct {
78     const char* enable_features;
79     const char* disable_features;
80     bool expected_feature_on_state;
81     bool expected_feature_off_state;
82   } test_cases[] = {
83       {"", "", true, false},
84       {"OffByDefault", "", true, true},
85       {"OffByDefault", "OnByDefault", false, true},
86       {"OnByDefault,OffByDefault", "", true, true},
87       {"", "OnByDefault,OffByDefault", false, false},
88       // In the case an entry is both, disable takes precedence.
89       {"OnByDefault", "OnByDefault,OffByDefault", false, false},
90   };
91 
92   for (size_t i = 0; i < arraysize(test_cases); ++i) {
93     const auto& test_case = test_cases[i];
94     SCOPED_TRACE(base::StringPrintf("Test[%" PRIuS "]: [%s] [%s]", i,
95                                     test_case.enable_features,
96                                     test_case.disable_features));
97 
98     ClearFeatureListInstance();
99     std::unique_ptr<FeatureList> feature_list(new FeatureList);
100     feature_list->InitializeFromCommandLine(test_case.enable_features,
101                                             test_case.disable_features);
102     RegisterFeatureListInstance(std::move(feature_list));
103 
104     EXPECT_EQ(test_case.expected_feature_on_state,
105               FeatureList::IsEnabled(kFeatureOnByDefault))
106         << i;
107     EXPECT_EQ(test_case.expected_feature_off_state,
108               FeatureList::IsEnabled(kFeatureOffByDefault))
109         << i;
110   }
111 }
112 
TEST_F(FeatureListTest,CheckFeatureIdentity)113 TEST_F(FeatureListTest, CheckFeatureIdentity) {
114   // Tests that CheckFeatureIdentity() correctly detects when two different
115   // structs with the same feature name are passed to it.
116 
117   // Call it twice for each feature at the top of the file, since the first call
118   // makes it remember the entry and the second call will verify it.
119   EXPECT_TRUE(feature_list()->CheckFeatureIdentity(kFeatureOnByDefault));
120   EXPECT_TRUE(feature_list()->CheckFeatureIdentity(kFeatureOnByDefault));
121   EXPECT_TRUE(feature_list()->CheckFeatureIdentity(kFeatureOffByDefault));
122   EXPECT_TRUE(feature_list()->CheckFeatureIdentity(kFeatureOffByDefault));
123 
124   // Now, call it with a distinct struct for |kFeatureOnByDefaultName|, which
125   // should return false.
126   struct Feature kFeatureOnByDefault2 {
127     kFeatureOnByDefaultName, FEATURE_ENABLED_BY_DEFAULT
128   };
129   EXPECT_FALSE(feature_list()->CheckFeatureIdentity(kFeatureOnByDefault2));
130 }
131 
TEST_F(FeatureListTest,FieldTrialOverrides)132 TEST_F(FeatureListTest, FieldTrialOverrides) {
133   struct {
134     FeatureList::OverrideState trial1_state;
135     FeatureList::OverrideState trial2_state;
136   } test_cases[] = {
137       {FeatureList::OVERRIDE_DISABLE_FEATURE,
138        FeatureList::OVERRIDE_DISABLE_FEATURE},
139       {FeatureList::OVERRIDE_DISABLE_FEATURE,
140        FeatureList::OVERRIDE_ENABLE_FEATURE},
141       {FeatureList::OVERRIDE_ENABLE_FEATURE,
142        FeatureList::OVERRIDE_DISABLE_FEATURE},
143       {FeatureList::OVERRIDE_ENABLE_FEATURE,
144        FeatureList::OVERRIDE_ENABLE_FEATURE},
145   };
146 
147   FieldTrial::ActiveGroup active_group;
148   for (size_t i = 0; i < arraysize(test_cases); ++i) {
149     const auto& test_case = test_cases[i];
150     SCOPED_TRACE(base::StringPrintf("Test[%" PRIuS "]", i));
151 
152     ClearFeatureListInstance();
153 
154     FieldTrialList field_trial_list(nullptr);
155     std::unique_ptr<FeatureList> feature_list(new FeatureList);
156 
157     FieldTrial* trial1 = FieldTrialList::CreateFieldTrial("TrialExample1", "A");
158     FieldTrial* trial2 = FieldTrialList::CreateFieldTrial("TrialExample2", "B");
159     feature_list->RegisterFieldTrialOverride(kFeatureOnByDefaultName,
160                                              test_case.trial1_state, trial1);
161     feature_list->RegisterFieldTrialOverride(kFeatureOffByDefaultName,
162                                              test_case.trial2_state, trial2);
163     RegisterFeatureListInstance(std::move(feature_list));
164 
165     // Initially, neither trial should be active.
166     EXPECT_FALSE(FieldTrialList::IsTrialActive(trial1->trial_name()));
167     EXPECT_FALSE(FieldTrialList::IsTrialActive(trial2->trial_name()));
168 
169     const bool expected_enabled_1 =
170         (test_case.trial1_state == FeatureList::OVERRIDE_ENABLE_FEATURE);
171     EXPECT_EQ(expected_enabled_1, FeatureList::IsEnabled(kFeatureOnByDefault));
172     // The above should have activated |trial1|.
173     EXPECT_TRUE(FieldTrialList::IsTrialActive(trial1->trial_name()));
174     EXPECT_FALSE(FieldTrialList::IsTrialActive(trial2->trial_name()));
175 
176     const bool expected_enabled_2 =
177         (test_case.trial2_state == FeatureList::OVERRIDE_ENABLE_FEATURE);
178     EXPECT_EQ(expected_enabled_2, FeatureList::IsEnabled(kFeatureOffByDefault));
179     // The above should have activated |trial2|.
180     EXPECT_TRUE(FieldTrialList::IsTrialActive(trial1->trial_name()));
181     EXPECT_TRUE(FieldTrialList::IsTrialActive(trial2->trial_name()));
182   }
183 }
184 
TEST_F(FeatureListTest,FieldTrialAssociateUseDefault)185 TEST_F(FeatureListTest, FieldTrialAssociateUseDefault) {
186   FieldTrialList field_trial_list(nullptr);
187   std::unique_ptr<FeatureList> feature_list(new FeatureList);
188 
189   FieldTrial* trial1 = FieldTrialList::CreateFieldTrial("TrialExample1", "A");
190   FieldTrial* trial2 = FieldTrialList::CreateFieldTrial("TrialExample2", "B");
191   feature_list->RegisterFieldTrialOverride(
192       kFeatureOnByDefaultName, FeatureList::OVERRIDE_USE_DEFAULT, trial1);
193   feature_list->RegisterFieldTrialOverride(
194       kFeatureOffByDefaultName, FeatureList::OVERRIDE_USE_DEFAULT, trial2);
195   RegisterFeatureListInstance(std::move(feature_list));
196 
197   // Initially, neither trial should be active.
198   EXPECT_FALSE(FieldTrialList::IsTrialActive(trial1->trial_name()));
199   EXPECT_FALSE(FieldTrialList::IsTrialActive(trial2->trial_name()));
200 
201   // Check the feature enabled state is its default.
202   EXPECT_TRUE(FeatureList::IsEnabled(kFeatureOnByDefault));
203   // The above should have activated |trial1|.
204   EXPECT_TRUE(FieldTrialList::IsTrialActive(trial1->trial_name()));
205   EXPECT_FALSE(FieldTrialList::IsTrialActive(trial2->trial_name()));
206 
207   // Check the feature enabled state is its default.
208   EXPECT_FALSE(FeatureList::IsEnabled(kFeatureOffByDefault));
209   // The above should have activated |trial2|.
210   EXPECT_TRUE(FieldTrialList::IsTrialActive(trial1->trial_name()));
211   EXPECT_TRUE(FieldTrialList::IsTrialActive(trial2->trial_name()));
212 }
213 
TEST_F(FeatureListTest,CommandLineTakesPrecedenceOverFieldTrial)214 TEST_F(FeatureListTest, CommandLineTakesPrecedenceOverFieldTrial) {
215   ClearFeatureListInstance();
216 
217   FieldTrialList field_trial_list(nullptr);
218   std::unique_ptr<FeatureList> feature_list(new FeatureList);
219 
220   // The feature is explicitly enabled on the command-line.
221   feature_list->InitializeFromCommandLine(kFeatureOffByDefaultName, "");
222 
223   // But the FieldTrial would set the feature to disabled.
224   FieldTrial* trial = FieldTrialList::CreateFieldTrial("TrialExample2", "A");
225   feature_list->RegisterFieldTrialOverride(
226       kFeatureOffByDefaultName, FeatureList::OVERRIDE_DISABLE_FEATURE, trial);
227   RegisterFeatureListInstance(std::move(feature_list));
228 
229   EXPECT_FALSE(FieldTrialList::IsTrialActive(trial->trial_name()));
230   // Command-line should take precedence.
231   EXPECT_TRUE(FeatureList::IsEnabled(kFeatureOffByDefault));
232   // Since the feature is on due to the command-line, and not as a result of the
233   // field trial, the field trial should not be activated (since the Associate*
234   // API wasn't used.)
235   EXPECT_FALSE(FieldTrialList::IsTrialActive(trial->trial_name()));
236 }
237 
TEST_F(FeatureListTest,IsFeatureOverriddenFromCommandLine)238 TEST_F(FeatureListTest, IsFeatureOverriddenFromCommandLine) {
239   ClearFeatureListInstance();
240 
241   FieldTrialList field_trial_list(nullptr);
242   std::unique_ptr<FeatureList> feature_list(new FeatureList);
243 
244   // No features are overridden from the command line yet
245   EXPECT_FALSE(feature_list->IsFeatureOverriddenFromCommandLine(
246       kFeatureOnByDefaultName, FeatureList::OVERRIDE_DISABLE_FEATURE));
247   EXPECT_FALSE(feature_list->IsFeatureOverriddenFromCommandLine(
248       kFeatureOnByDefaultName, FeatureList::OVERRIDE_ENABLE_FEATURE));
249   EXPECT_FALSE(feature_list->IsFeatureOverriddenFromCommandLine(
250       kFeatureOffByDefaultName, FeatureList::OVERRIDE_DISABLE_FEATURE));
251   EXPECT_FALSE(feature_list->IsFeatureOverriddenFromCommandLine(
252       kFeatureOffByDefaultName, FeatureList::OVERRIDE_ENABLE_FEATURE));
253 
254   // Now, enable |kFeatureOffByDefaultName| via the command-line.
255   feature_list->InitializeFromCommandLine(kFeatureOffByDefaultName, "");
256 
257   // It should now be overridden for the enabled group.
258   EXPECT_FALSE(feature_list->IsFeatureOverriddenFromCommandLine(
259       kFeatureOffByDefaultName, FeatureList::OVERRIDE_DISABLE_FEATURE));
260   EXPECT_TRUE(feature_list->IsFeatureOverriddenFromCommandLine(
261       kFeatureOffByDefaultName, FeatureList::OVERRIDE_ENABLE_FEATURE));
262 
263   // Register a field trial to associate with the feature and ensure that the
264   // results are still the same.
265   feature_list->AssociateReportingFieldTrial(
266       kFeatureOffByDefaultName, FeatureList::OVERRIDE_ENABLE_FEATURE,
267       FieldTrialList::CreateFieldTrial("Trial1", "A"));
268   EXPECT_FALSE(feature_list->IsFeatureOverriddenFromCommandLine(
269       kFeatureOffByDefaultName, FeatureList::OVERRIDE_DISABLE_FEATURE));
270   EXPECT_TRUE(feature_list->IsFeatureOverriddenFromCommandLine(
271       kFeatureOffByDefaultName, FeatureList::OVERRIDE_ENABLE_FEATURE));
272 
273   // Now, register a field trial to override |kFeatureOnByDefaultName| state
274   // and check that the function still returns false for that feature.
275   feature_list->RegisterFieldTrialOverride(
276       kFeatureOnByDefaultName, FeatureList::OVERRIDE_DISABLE_FEATURE,
277       FieldTrialList::CreateFieldTrial("Trial2", "A"));
278   EXPECT_FALSE(feature_list->IsFeatureOverriddenFromCommandLine(
279       kFeatureOnByDefaultName, FeatureList::OVERRIDE_DISABLE_FEATURE));
280   EXPECT_FALSE(feature_list->IsFeatureOverriddenFromCommandLine(
281       kFeatureOnByDefaultName, FeatureList::OVERRIDE_ENABLE_FEATURE));
282   RegisterFeatureListInstance(std::move(feature_list));
283 
284   // Check the expected feature states for good measure.
285   EXPECT_TRUE(FeatureList::IsEnabled(kFeatureOffByDefault));
286   EXPECT_FALSE(FeatureList::IsEnabled(kFeatureOnByDefault));
287 }
288 
TEST_F(FeatureListTest,AssociateReportingFieldTrial)289 TEST_F(FeatureListTest, AssociateReportingFieldTrial) {
290   struct {
291     const char* enable_features;
292     const char* disable_features;
293     bool expected_enable_trial_created;
294     bool expected_disable_trial_created;
295   } test_cases[] = {
296       // If no enable/disable flags are specified, no trials should be created.
297       {"", "", false, false},
298       // Enabling the feature should result in the enable trial created.
299       {kFeatureOffByDefaultName, "", true, false},
300       // Disabling the feature should result in the disable trial created.
301       {"", kFeatureOffByDefaultName, false, true},
302   };
303 
304   const char kTrialName[] = "ForcingTrial";
305   const char kForcedOnGroupName[] = "ForcedOn";
306   const char kForcedOffGroupName[] = "ForcedOff";
307 
308   for (size_t i = 0; i < arraysize(test_cases); ++i) {
309     const auto& test_case = test_cases[i];
310     SCOPED_TRACE(base::StringPrintf("Test[%" PRIuS "]: [%s] [%s]", i,
311                                     test_case.enable_features,
312                                     test_case.disable_features));
313 
314     ClearFeatureListInstance();
315 
316     FieldTrialList field_trial_list(nullptr);
317     std::unique_ptr<FeatureList> feature_list(new FeatureList);
318     feature_list->InitializeFromCommandLine(test_case.enable_features,
319                                             test_case.disable_features);
320 
321     FieldTrial* enable_trial = nullptr;
322     if (feature_list->IsFeatureOverriddenFromCommandLine(
323             kFeatureOffByDefaultName, FeatureList::OVERRIDE_ENABLE_FEATURE)) {
324       enable_trial = base::FieldTrialList::CreateFieldTrial(kTrialName,
325                                                             kForcedOnGroupName);
326       feature_list->AssociateReportingFieldTrial(
327           kFeatureOffByDefaultName, FeatureList::OVERRIDE_ENABLE_FEATURE,
328           enable_trial);
329     }
330     FieldTrial* disable_trial = nullptr;
331     if (feature_list->IsFeatureOverriddenFromCommandLine(
332             kFeatureOffByDefaultName, FeatureList::OVERRIDE_DISABLE_FEATURE)) {
333       disable_trial = base::FieldTrialList::CreateFieldTrial(
334           kTrialName, kForcedOffGroupName);
335       feature_list->AssociateReportingFieldTrial(
336           kFeatureOffByDefaultName, FeatureList::OVERRIDE_DISABLE_FEATURE,
337           disable_trial);
338     }
339     EXPECT_EQ(test_case.expected_enable_trial_created, enable_trial != nullptr);
340     EXPECT_EQ(test_case.expected_disable_trial_created,
341               disable_trial != nullptr);
342     RegisterFeatureListInstance(std::move(feature_list));
343 
344     EXPECT_FALSE(FieldTrialList::IsTrialActive(kTrialName));
345     if (disable_trial) {
346       EXPECT_FALSE(FeatureList::IsEnabled(kFeatureOffByDefault));
347       EXPECT_TRUE(FieldTrialList::IsTrialActive(kTrialName));
348       EXPECT_EQ(kForcedOffGroupName, disable_trial->group_name());
349     } else if (enable_trial) {
350       EXPECT_TRUE(FeatureList::IsEnabled(kFeatureOffByDefault));
351       EXPECT_TRUE(FieldTrialList::IsTrialActive(kTrialName));
352       EXPECT_EQ(kForcedOnGroupName, enable_trial->group_name());
353     }
354   }
355 }
356 
TEST_F(FeatureListTest,GetFeatureOverrides)357 TEST_F(FeatureListTest, GetFeatureOverrides) {
358   ClearFeatureListInstance();
359   FieldTrialList field_trial_list(nullptr);
360   std::unique_ptr<FeatureList> feature_list(new FeatureList);
361   feature_list->InitializeFromCommandLine("A,X", "D");
362 
363   FieldTrial* trial = FieldTrialList::CreateFieldTrial("Trial", "Group");
364   feature_list->RegisterFieldTrialOverride(kFeatureOffByDefaultName,
365                                            FeatureList::OVERRIDE_ENABLE_FEATURE,
366                                            trial);
367 
368   RegisterFeatureListInstance(std::move(feature_list));
369 
370   std::string enable_features;
371   std::string disable_features;
372   FeatureList::GetInstance()->GetFeatureOverrides(&enable_features,
373                                                   &disable_features);
374   EXPECT_EQ("A,OffByDefault<Trial,X", SortFeatureListString(enable_features));
375   EXPECT_EQ("D", SortFeatureListString(disable_features));
376 }
377 
TEST_F(FeatureListTest,GetFeatureOverrides_UseDefault)378 TEST_F(FeatureListTest, GetFeatureOverrides_UseDefault) {
379   ClearFeatureListInstance();
380   FieldTrialList field_trial_list(nullptr);
381   std::unique_ptr<FeatureList> feature_list(new FeatureList);
382   feature_list->InitializeFromCommandLine("A,X", "D");
383 
384   FieldTrial* trial = FieldTrialList::CreateFieldTrial("Trial", "Group");
385   feature_list->RegisterFieldTrialOverride(
386       kFeatureOffByDefaultName, FeatureList::OVERRIDE_USE_DEFAULT, trial);
387 
388   RegisterFeatureListInstance(std::move(feature_list));
389 
390   std::string enable_features;
391   std::string disable_features;
392   FeatureList::GetInstance()->GetFeatureOverrides(&enable_features,
393                                                   &disable_features);
394   EXPECT_EQ("*OffByDefault<Trial,A,X", SortFeatureListString(enable_features));
395   EXPECT_EQ("D", SortFeatureListString(disable_features));
396 }
397 
TEST_F(FeatureListTest,GetFieldTrial)398 TEST_F(FeatureListTest, GetFieldTrial) {
399   ClearFeatureListInstance();
400   FieldTrialList field_trial_list(nullptr);
401   FieldTrial* trial = FieldTrialList::CreateFieldTrial("Trial", "Group");
402   std::unique_ptr<FeatureList> feature_list(new FeatureList);
403   feature_list->RegisterFieldTrialOverride(
404       kFeatureOnByDefaultName, FeatureList::OVERRIDE_USE_DEFAULT, trial);
405   RegisterFeatureListInstance(std::move(feature_list));
406 
407   EXPECT_EQ(trial, FeatureList::GetFieldTrial(kFeatureOnByDefault));
408   EXPECT_EQ(nullptr, FeatureList::GetFieldTrial(kFeatureOffByDefault));
409 }
410 
TEST_F(FeatureListTest,InitializeFromCommandLine_WithFieldTrials)411 TEST_F(FeatureListTest, InitializeFromCommandLine_WithFieldTrials) {
412   ClearFeatureListInstance();
413   FieldTrialList field_trial_list(nullptr);
414   FieldTrialList::CreateFieldTrial("Trial", "Group");
415   std::unique_ptr<FeatureList> feature_list(new FeatureList);
416   feature_list->InitializeFromCommandLine("A,OffByDefault<Trial,X", "D");
417   RegisterFeatureListInstance(std::move(feature_list));
418 
419   EXPECT_FALSE(FieldTrialList::IsTrialActive("Trial"));
420   EXPECT_TRUE(FeatureList::IsEnabled(kFeatureOffByDefault));
421   EXPECT_TRUE(FieldTrialList::IsTrialActive("Trial"));
422 }
423 
TEST_F(FeatureListTest,InitializeFromCommandLine_UseDefault)424 TEST_F(FeatureListTest, InitializeFromCommandLine_UseDefault) {
425   ClearFeatureListInstance();
426   FieldTrialList field_trial_list(nullptr);
427   FieldTrialList::CreateFieldTrial("T1", "Group");
428   FieldTrialList::CreateFieldTrial("T2", "Group");
429   std::unique_ptr<FeatureList> feature_list(new FeatureList);
430   feature_list->InitializeFromCommandLine(
431       "A,*OffByDefault<T1,*OnByDefault<T2,X", "D");
432   RegisterFeatureListInstance(std::move(feature_list));
433 
434   EXPECT_FALSE(FieldTrialList::IsTrialActive("T1"));
435   EXPECT_FALSE(FeatureList::IsEnabled(kFeatureOffByDefault));
436   EXPECT_TRUE(FieldTrialList::IsTrialActive("T1"));
437 
438   EXPECT_FALSE(FieldTrialList::IsTrialActive("T2"));
439   EXPECT_TRUE(FeatureList::IsEnabled(kFeatureOnByDefault));
440   EXPECT_TRUE(FieldTrialList::IsTrialActive("T2"));
441 }
442 
TEST_F(FeatureListTest,InitializeInstance)443 TEST_F(FeatureListTest, InitializeInstance) {
444   ClearFeatureListInstance();
445 
446   std::unique_ptr<base::FeatureList> feature_list(new base::FeatureList);
447   FeatureList::SetInstance(std::move(feature_list));
448   EXPECT_TRUE(FeatureList::IsEnabled(kFeatureOnByDefault));
449   EXPECT_FALSE(FeatureList::IsEnabled(kFeatureOffByDefault));
450 
451   // Initialize from command line if we haven't yet.
452   FeatureList::InitializeInstance("", kFeatureOnByDefaultName);
453   EXPECT_FALSE(FeatureList::IsEnabled(kFeatureOnByDefault));
454   EXPECT_FALSE(FeatureList::IsEnabled(kFeatureOffByDefault));
455 
456   // Do not initialize from commandline if we have already.
457   FeatureList::InitializeInstance(kFeatureOffByDefaultName, "");
458   EXPECT_FALSE(FeatureList::IsEnabled(kFeatureOnByDefault));
459   EXPECT_FALSE(FeatureList::IsEnabled(kFeatureOffByDefault));
460 }
461 
TEST_F(FeatureListTest,UninitializedInstance_IsEnabledReturnsFalse)462 TEST_F(FeatureListTest, UninitializedInstance_IsEnabledReturnsFalse) {
463   ClearFeatureListInstance();
464   // This test case simulates the calling pattern found in code which does not
465   // explicitly initialize the features list.
466   // All IsEnabled() calls should return the default value in this scenario.
467   EXPECT_EQ(nullptr, FeatureList::GetInstance());
468   EXPECT_TRUE(FeatureList::IsEnabled(kFeatureOnByDefault));
469   EXPECT_EQ(nullptr, FeatureList::GetInstance());
470   EXPECT_FALSE(FeatureList::IsEnabled(kFeatureOffByDefault));
471 }
472 
TEST_F(FeatureListTest,StoreAndRetrieveFeaturesFromSharedMemory)473 TEST_F(FeatureListTest, StoreAndRetrieveFeaturesFromSharedMemory) {
474   std::unique_ptr<base::FeatureList> feature_list(new base::FeatureList);
475 
476   // Create some overrides.
477   feature_list->RegisterOverride(kFeatureOffByDefaultName,
478                                  FeatureList::OVERRIDE_ENABLE_FEATURE, nullptr);
479   feature_list->RegisterOverride(
480       kFeatureOnByDefaultName, FeatureList::OVERRIDE_DISABLE_FEATURE, nullptr);
481   feature_list->FinalizeInitialization();
482 
483   // Create an allocator and store the overrides.
484   std::unique_ptr<SharedMemory> shm(new SharedMemory());
485   shm->CreateAndMapAnonymous(4 << 10);
486   SharedPersistentMemoryAllocator allocator(std::move(shm), 1, "", false);
487   feature_list->AddFeaturesToAllocator(&allocator);
488 
489   std::unique_ptr<base::FeatureList> feature_list2(new base::FeatureList);
490 
491   // Check that the new feature list is empty.
492   EXPECT_FALSE(feature_list2->IsFeatureOverriddenFromCommandLine(
493       kFeatureOffByDefaultName, FeatureList::OVERRIDE_ENABLE_FEATURE));
494   EXPECT_FALSE(feature_list2->IsFeatureOverriddenFromCommandLine(
495       kFeatureOnByDefaultName, FeatureList::OVERRIDE_DISABLE_FEATURE));
496 
497   feature_list2->InitializeFromSharedMemory(&allocator);
498   // Check that the new feature list now has 2 overrides.
499   EXPECT_TRUE(feature_list2->IsFeatureOverriddenFromCommandLine(
500       kFeatureOffByDefaultName, FeatureList::OVERRIDE_ENABLE_FEATURE));
501   EXPECT_TRUE(feature_list2->IsFeatureOverriddenFromCommandLine(
502       kFeatureOnByDefaultName, FeatureList::OVERRIDE_DISABLE_FEATURE));
503 }
504 
TEST_F(FeatureListTest,StoreAndRetrieveAssociatedFeaturesFromSharedMemory)505 TEST_F(FeatureListTest, StoreAndRetrieveAssociatedFeaturesFromSharedMemory) {
506   FieldTrialList field_trial_list(nullptr);
507   std::unique_ptr<base::FeatureList> feature_list(new base::FeatureList);
508 
509   // Create some overrides.
510   FieldTrial* trial1 = FieldTrialList::CreateFieldTrial("TrialExample1", "A");
511   FieldTrial* trial2 = FieldTrialList::CreateFieldTrial("TrialExample2", "B");
512   feature_list->RegisterFieldTrialOverride(
513       kFeatureOnByDefaultName, FeatureList::OVERRIDE_USE_DEFAULT, trial1);
514   feature_list->RegisterFieldTrialOverride(
515       kFeatureOffByDefaultName, FeatureList::OVERRIDE_USE_DEFAULT, trial2);
516   feature_list->FinalizeInitialization();
517 
518   // Create an allocator and store the overrides.
519   std::unique_ptr<SharedMemory> shm(new SharedMemory());
520   shm->CreateAndMapAnonymous(4 << 10);
521   SharedPersistentMemoryAllocator allocator(std::move(shm), 1, "", false);
522   feature_list->AddFeaturesToAllocator(&allocator);
523 
524   std::unique_ptr<base::FeatureList> feature_list2(new base::FeatureList);
525   feature_list2->InitializeFromSharedMemory(&allocator);
526   feature_list2->FinalizeInitialization();
527 
528   // Check that the field trials are still associated.
529   FieldTrial* associated_trial1 =
530       feature_list2->GetAssociatedFieldTrial(kFeatureOnByDefault);
531   FieldTrial* associated_trial2 =
532       feature_list2->GetAssociatedFieldTrial(kFeatureOffByDefault);
533   EXPECT_EQ(associated_trial1, trial1);
534   EXPECT_EQ(associated_trial2, trial2);
535 }
536 
537 }  // namespace base
538