Roll src/third_party/WebKit bf18a82:a9cee16 (svn 185297:185304)
[chromium-blink-merge.git] / chrome / browser / about_flags_unittest.cc
blob32fb1ded9a41e528fe855b7147a66646257946e9
1 // Copyright (c) 2011 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.
5 #include <stdint.h>
7 #include "base/files/file_path.h"
8 #include "base/memory/scoped_vector.h"
9 #include "base/path_service.h"
10 #include "base/prefs/pref_registry_simple.h"
11 #include "base/prefs/testing_pref_service.h"
12 #include "base/strings/string_number_conversions.h"
13 #include "base/strings/stringprintf.h"
14 #include "base/strings/utf_string_conversions.h"
15 #include "base/values.h"
16 #include "chrome/browser/about_flags.h"
17 #include "chrome/browser/pref_service_flags_storage.h"
18 #include "chrome/common/chrome_switches.h"
19 #include "chrome/common/pref_names.h"
20 #include "chrome/grit/chromium_strings.h"
21 #include "testing/gtest/include/gtest/gtest.h"
22 #include "third_party/libxml/chromium/libxml_utils.h"
24 namespace {
26 const char kFlags1[] = "flag1";
27 const char kFlags2[] = "flag2";
28 const char kFlags3[] = "flag3";
29 const char kFlags4[] = "flag4";
30 const char kFlags5[] = "flag5";
32 const char kSwitch1[] = "switch";
33 const char kSwitch2[] = "switch2";
34 const char kSwitch3[] = "switch3";
35 const char kValueForSwitch2[] = "value_for_switch2";
37 const char kMultiSwitch1[] = "multi_switch1";
38 const char kMultiSwitch2[] = "multi_switch2";
39 const char kValueForMultiSwitch2[] = "value_for_multi_switch2";
41 const char kEnableDisableValue1[] = "value1";
42 const char kEnableDisableValue2[] = "value2";
44 typedef base::HistogramBase::Sample Sample;
45 typedef std::map<std::string, Sample> SwitchToIdMap;
47 // This is a helper function to the ReadEnumFromHistogramsXml().
48 // Extracts single enum (with integer values) from histograms.xml.
49 // Expects |reader| to point at given enum.
50 // Returns map { value => label }.
51 // Returns empty map on error.
52 std::map<Sample, std::string> ParseEnumFromHistogramsXml(
53 const std::string& enum_name,
54 XmlReader* reader) {
55 int entries_index = -1;
57 std::map<Sample, std::string> result;
58 bool success = true;
60 while (true) {
61 const std::string node_name = reader->NodeName();
62 if (node_name == "enum" && reader->IsClosingElement())
63 break;
65 if (node_name == "int") {
66 ++entries_index;
67 std::string value_str;
68 std::string label;
69 const bool has_value = reader->NodeAttribute("value", &value_str);
70 const bool has_label = reader->NodeAttribute("label", &label);
71 if (!has_value) {
72 ADD_FAILURE() << "Bad " << enum_name << " enum entry (at index "
73 << entries_index << ", label='" << label
74 << "'): No 'value' attribute.";
75 success = false;
77 if (!has_label) {
78 ADD_FAILURE() << "Bad " << enum_name << " enum entry (at index "
79 << entries_index << ", value_str='" << value_str
80 << "'): No 'label' attribute.";
81 success = false;
84 Sample value;
85 if (has_value && !base::StringToInt(value_str, &value)) {
86 ADD_FAILURE() << "Bad " << enum_name << " enum entry (at index "
87 << entries_index << ", label='" << label
88 << "', value_str='" << value_str
89 << "'): 'value' attribute is not integer.";
90 success = false;
92 if (result.count(value)) {
93 ADD_FAILURE() << "Bad " << enum_name << " enum entry (at index "
94 << entries_index << ", label='" << label
95 << "', value_str='" << value_str
96 << "'): duplicate value '" << value_str
97 << "' found in enum. The previous one has label='"
98 << result[value] << "'.";
99 success = false;
101 if (success) {
102 result[value] = label;
105 // All enum entries are on the same level, so it is enough to iterate
106 // until possible.
107 reader->Next();
109 return (success ? result : std::map<Sample, std::string>());
112 // Find and read given enum (with integer values) from histograms.xml.
113 // |enum_name| - enum name.
114 // |histograms_xml| - must be loaded histograms.xml file.
116 // Returns map { value => label } so that:
117 // <int value="9" label="enable-pinch-virtual-viewport"/>
118 // becomes:
119 // { 9 => "enable-pinch-virtual-viewport" }
120 // Returns empty map on error.
121 std::map<Sample, std::string> ReadEnumFromHistogramsXml(
122 const std::string& enum_name,
123 XmlReader* histograms_xml) {
124 std::map<Sample, std::string> login_custom_flags;
126 // Implement simple depth first search.
127 while (true) {
128 const std::string node_name = histograms_xml->NodeName();
129 if (node_name == "enum") {
130 std::string name;
131 if (histograms_xml->NodeAttribute("name", &name) && name == enum_name) {
132 if (!login_custom_flags.empty()) {
133 EXPECT_TRUE(login_custom_flags.empty())
134 << "Duplicate enum '" << enum_name << "' found in histograms.xml";
135 return std::map<Sample, std::string>();
138 const bool got_into_enum = histograms_xml->Read();
139 if (got_into_enum) {
140 login_custom_flags =
141 ParseEnumFromHistogramsXml(enum_name, histograms_xml);
142 EXPECT_FALSE(login_custom_flags.empty())
143 << "Bad enum '" << enum_name
144 << "' found in histograms.xml (format error).";
145 } else {
146 EXPECT_TRUE(got_into_enum)
147 << "Bad enum '" << enum_name
148 << "' (looks empty) found in histograms.xml.";
150 if (login_custom_flags.empty())
151 return std::map<Sample, std::string>();
154 // Go deeper if possible (stops at the closing tag of the deepest node).
155 if (histograms_xml->Read())
156 continue;
158 // Try next node on the same level (skips closing tag).
159 if (histograms_xml->Next())
160 continue;
162 // Go up until next node on the same level exists.
163 while (histograms_xml->Depth() && !histograms_xml->SkipToElement()) {
166 // Reached top. histograms.xml consists of the single top level node
167 // 'histogram-configuration', so this is the end.
168 if (!histograms_xml->Depth())
169 break;
171 EXPECT_FALSE(login_custom_flags.empty())
172 << "Enum '" << enum_name << "' is not found in histograms.xml.";
173 return login_custom_flags;
176 std::string FilePathStringTypeToString(const base::FilePath::StringType& path) {
177 #if defined(OS_WIN)
178 return base::UTF16ToUTF8(path);
179 #else
180 return path;
181 #endif
184 std::set<std::string> GetAllSwitchesForTesting() {
185 std::set<std::string> result;
187 size_t num_experiments = 0;
188 const about_flags::Experiment* experiments =
189 about_flags::testing::GetExperiments(&num_experiments);
191 for (size_t i = 0; i < num_experiments; ++i) {
192 const about_flags::Experiment& experiment = experiments[i];
193 if (experiment.type == about_flags::Experiment::SINGLE_VALUE) {
194 result.insert(experiment.command_line_switch);
195 } else if (experiment.type == about_flags::Experiment::MULTI_VALUE) {
196 for (int j = 0; j < experiment.num_choices; ++j) {
197 result.insert(experiment.choices[j].command_line_switch);
199 } else {
200 DCHECK_EQ(experiment.type, about_flags::Experiment::ENABLE_DISABLE_VALUE);
201 result.insert(experiment.command_line_switch);
202 result.insert(experiment.disable_command_line_switch);
205 return result;
208 } // anonymous namespace
210 namespace about_flags {
212 const Experiment::Choice kMultiChoices[] = {
213 { IDS_PRODUCT_NAME, "", "" },
214 { IDS_PRODUCT_NAME, kMultiSwitch1, "" },
215 { IDS_PRODUCT_NAME, kMultiSwitch2, kValueForMultiSwitch2 },
218 // The experiments that are set for these tests. The 3rd experiment is not
219 // supported on the current platform, all others are.
220 static Experiment kExperiments[] = {
222 kFlags1,
223 IDS_PRODUCT_NAME,
224 IDS_PRODUCT_NAME,
225 0, // Ends up being mapped to the current platform.
226 Experiment::SINGLE_VALUE,
227 kSwitch1,
229 NULL,
230 NULL,
231 NULL,
235 kFlags2,
236 IDS_PRODUCT_NAME,
237 IDS_PRODUCT_NAME,
238 0, // Ends up being mapped to the current platform.
239 Experiment::SINGLE_VALUE,
240 kSwitch2,
241 kValueForSwitch2,
242 NULL,
243 NULL,
244 NULL,
248 kFlags3,
249 IDS_PRODUCT_NAME,
250 IDS_PRODUCT_NAME,
251 0, // This ends up enabling for an OS other than the current.
252 Experiment::SINGLE_VALUE,
253 kSwitch3,
255 NULL,
256 NULL,
257 NULL,
261 kFlags4,
262 IDS_PRODUCT_NAME,
263 IDS_PRODUCT_NAME,
264 0, // Ends up being mapped to the current platform.
265 Experiment::MULTI_VALUE,
270 kMultiChoices,
271 arraysize(kMultiChoices)
274 kFlags5,
275 IDS_PRODUCT_NAME,
276 IDS_PRODUCT_NAME,
277 0, // Ends up being mapped to the current platform.
278 Experiment::ENABLE_DISABLE_VALUE,
279 kSwitch1,
280 kEnableDisableValue1,
281 kSwitch2,
282 kEnableDisableValue2,
283 NULL,
288 class AboutFlagsTest : public ::testing::Test {
289 protected:
290 AboutFlagsTest() : flags_storage_(&prefs_) {
291 prefs_.registry()->RegisterListPref(prefs::kEnabledLabsExperiments);
292 testing::ClearState();
295 void SetUp() override {
296 for (size_t i = 0; i < arraysize(kExperiments); ++i)
297 kExperiments[i].supported_platforms = GetCurrentPlatform();
299 int os_other_than_current = 1;
300 while (os_other_than_current == GetCurrentPlatform())
301 os_other_than_current <<= 1;
302 kExperiments[2].supported_platforms = os_other_than_current;
304 testing::SetExperiments(kExperiments, arraysize(kExperiments));
307 void TearDown() override { testing::SetExperiments(NULL, 0); }
309 TestingPrefServiceSimple prefs_;
310 PrefServiceFlagsStorage flags_storage_;
314 TEST_F(AboutFlagsTest, NoChangeNoRestart) {
315 EXPECT_FALSE(IsRestartNeededToCommitChanges());
316 SetExperimentEnabled(&flags_storage_, kFlags1, false);
317 EXPECT_FALSE(IsRestartNeededToCommitChanges());
320 TEST_F(AboutFlagsTest, ChangeNeedsRestart) {
321 EXPECT_FALSE(IsRestartNeededToCommitChanges());
322 SetExperimentEnabled(&flags_storage_, kFlags1, true);
323 EXPECT_TRUE(IsRestartNeededToCommitChanges());
326 TEST_F(AboutFlagsTest, MultiFlagChangeNeedsRestart) {
327 const Experiment& experiment = kExperiments[3];
328 ASSERT_EQ(kFlags4, experiment.internal_name);
329 EXPECT_FALSE(IsRestartNeededToCommitChanges());
330 // Enable the 2nd choice of the multi-value.
331 SetExperimentEnabled(&flags_storage_, experiment.NameForChoice(2), true);
332 EXPECT_TRUE(IsRestartNeededToCommitChanges());
333 testing::ClearState();
334 EXPECT_FALSE(IsRestartNeededToCommitChanges());
335 // Enable the default choice now.
336 SetExperimentEnabled(&flags_storage_, experiment.NameForChoice(0), true);
337 EXPECT_TRUE(IsRestartNeededToCommitChanges());
340 TEST_F(AboutFlagsTest, AddTwoFlagsRemoveOne) {
341 // Add two experiments, check they're there.
342 SetExperimentEnabled(&flags_storage_, kFlags1, true);
343 SetExperimentEnabled(&flags_storage_, kFlags2, true);
345 const base::ListValue* experiments_list = prefs_.GetList(
346 prefs::kEnabledLabsExperiments);
347 ASSERT_TRUE(experiments_list != NULL);
349 ASSERT_EQ(2u, experiments_list->GetSize());
351 std::string s0;
352 ASSERT_TRUE(experiments_list->GetString(0, &s0));
353 std::string s1;
354 ASSERT_TRUE(experiments_list->GetString(1, &s1));
356 EXPECT_TRUE(s0 == kFlags1 || s1 == kFlags1);
357 EXPECT_TRUE(s0 == kFlags2 || s1 == kFlags2);
359 // Remove one experiment, check the other's still around.
360 SetExperimentEnabled(&flags_storage_, kFlags2, false);
362 experiments_list = prefs_.GetList(prefs::kEnabledLabsExperiments);
363 ASSERT_TRUE(experiments_list != NULL);
364 ASSERT_EQ(1u, experiments_list->GetSize());
365 ASSERT_TRUE(experiments_list->GetString(0, &s0));
366 EXPECT_TRUE(s0 == kFlags1);
369 TEST_F(AboutFlagsTest, AddTwoFlagsRemoveBoth) {
370 // Add two experiments, check the pref exists.
371 SetExperimentEnabled(&flags_storage_, kFlags1, true);
372 SetExperimentEnabled(&flags_storage_, kFlags2, true);
373 const base::ListValue* experiments_list = prefs_.GetList(
374 prefs::kEnabledLabsExperiments);
375 ASSERT_TRUE(experiments_list != NULL);
377 // Remove both, the pref should have been removed completely.
378 SetExperimentEnabled(&flags_storage_, kFlags1, false);
379 SetExperimentEnabled(&flags_storage_, kFlags2, false);
380 experiments_list = prefs_.GetList(prefs::kEnabledLabsExperiments);
381 EXPECT_TRUE(experiments_list == NULL || experiments_list->GetSize() == 0);
384 TEST_F(AboutFlagsTest, ConvertFlagsToSwitches) {
385 SetExperimentEnabled(&flags_storage_, kFlags1, true);
387 CommandLine command_line(CommandLine::NO_PROGRAM);
388 command_line.AppendSwitch("foo");
390 EXPECT_TRUE(command_line.HasSwitch("foo"));
391 EXPECT_FALSE(command_line.HasSwitch(kSwitch1));
393 ConvertFlagsToSwitches(&flags_storage_, &command_line, kAddSentinels);
395 EXPECT_TRUE(command_line.HasSwitch("foo"));
396 EXPECT_TRUE(command_line.HasSwitch(kSwitch1));
397 EXPECT_TRUE(command_line.HasSwitch(switches::kFlagSwitchesBegin));
398 EXPECT_TRUE(command_line.HasSwitch(switches::kFlagSwitchesEnd));
400 CommandLine command_line2(CommandLine::NO_PROGRAM);
402 ConvertFlagsToSwitches(&flags_storage_, &command_line2, kNoSentinels);
404 EXPECT_TRUE(command_line2.HasSwitch(kSwitch1));
405 EXPECT_FALSE(command_line2.HasSwitch(switches::kFlagSwitchesBegin));
406 EXPECT_FALSE(command_line2.HasSwitch(switches::kFlagSwitchesEnd));
409 CommandLine::StringType CreateSwitch(const std::string& value) {
410 #if defined(OS_WIN)
411 return base::ASCIIToUTF16(value);
412 #else
413 return value;
414 #endif
417 TEST_F(AboutFlagsTest, CompareSwitchesToCurrentCommandLine) {
418 SetExperimentEnabled(&flags_storage_, kFlags1, true);
420 const std::string kDoubleDash("--");
422 CommandLine command_line(CommandLine::NO_PROGRAM);
423 command_line.AppendSwitch("foo");
425 CommandLine new_command_line(CommandLine::NO_PROGRAM);
426 ConvertFlagsToSwitches(&flags_storage_, &new_command_line, kAddSentinels);
428 EXPECT_FALSE(AreSwitchesIdenticalToCurrentCommandLine(
429 new_command_line, command_line, NULL));
431 std::set<CommandLine::StringType> difference;
432 EXPECT_FALSE(AreSwitchesIdenticalToCurrentCommandLine(
433 new_command_line, command_line, &difference));
434 EXPECT_EQ(1U, difference.size());
435 EXPECT_EQ(1U, difference.count(CreateSwitch(kDoubleDash + kSwitch1)));
438 ConvertFlagsToSwitches(&flags_storage_, &command_line, kAddSentinels);
440 EXPECT_TRUE(AreSwitchesIdenticalToCurrentCommandLine(
441 new_command_line, command_line, NULL));
443 std::set<CommandLine::StringType> difference;
444 EXPECT_TRUE(AreSwitchesIdenticalToCurrentCommandLine(
445 new_command_line, command_line, &difference));
446 EXPECT_TRUE(difference.empty());
449 // Now both have flags but different.
450 SetExperimentEnabled(&flags_storage_, kFlags1, false);
451 SetExperimentEnabled(&flags_storage_, kFlags2, true);
453 CommandLine another_command_line(CommandLine::NO_PROGRAM);
454 ConvertFlagsToSwitches(&flags_storage_, &another_command_line, kAddSentinels);
456 EXPECT_FALSE(AreSwitchesIdenticalToCurrentCommandLine(
457 new_command_line, another_command_line, NULL));
459 std::set<CommandLine::StringType> difference;
460 EXPECT_FALSE(AreSwitchesIdenticalToCurrentCommandLine(
461 new_command_line, another_command_line, &difference));
462 EXPECT_EQ(2U, difference.size());
463 EXPECT_EQ(1U, difference.count(CreateSwitch(kDoubleDash + kSwitch1)));
464 EXPECT_EQ(1U,
465 difference.count(CreateSwitch(kDoubleDash + kSwitch2 + "=" +
466 kValueForSwitch2)));
470 TEST_F(AboutFlagsTest, RemoveFlagSwitches) {
471 std::map<std::string, CommandLine::StringType> switch_list;
472 switch_list[kSwitch1] = CommandLine::StringType();
473 switch_list[switches::kFlagSwitchesBegin] = CommandLine::StringType();
474 switch_list[switches::kFlagSwitchesEnd] = CommandLine::StringType();
475 switch_list["foo"] = CommandLine::StringType();
477 SetExperimentEnabled(&flags_storage_, kFlags1, true);
479 // This shouldn't do anything before ConvertFlagsToSwitches() wasn't called.
480 RemoveFlagsSwitches(&switch_list);
481 ASSERT_EQ(4u, switch_list.size());
482 EXPECT_TRUE(switch_list.find(kSwitch1) != switch_list.end());
483 EXPECT_TRUE(switch_list.find(switches::kFlagSwitchesBegin) !=
484 switch_list.end());
485 EXPECT_TRUE(switch_list.find(switches::kFlagSwitchesEnd) !=
486 switch_list.end());
487 EXPECT_TRUE(switch_list.find("foo") != switch_list.end());
489 // Call ConvertFlagsToSwitches(), then RemoveFlagsSwitches() again.
490 CommandLine command_line(CommandLine::NO_PROGRAM);
491 command_line.AppendSwitch("foo");
492 ConvertFlagsToSwitches(&flags_storage_, &command_line, kAddSentinels);
493 RemoveFlagsSwitches(&switch_list);
495 // Now the about:flags-related switch should have been removed.
496 ASSERT_EQ(1u, switch_list.size());
497 EXPECT_TRUE(switch_list.find("foo") != switch_list.end());
500 // Tests enabling experiments that aren't supported on the current platform.
501 TEST_F(AboutFlagsTest, PersistAndPrune) {
502 // Enable experiments 1 and 3.
503 SetExperimentEnabled(&flags_storage_, kFlags1, true);
504 SetExperimentEnabled(&flags_storage_, kFlags3, true);
505 CommandLine command_line(CommandLine::NO_PROGRAM);
506 EXPECT_FALSE(command_line.HasSwitch(kSwitch1));
507 EXPECT_FALSE(command_line.HasSwitch(kSwitch3));
509 // Convert the flags to switches. Experiment 3 shouldn't be among the switches
510 // as it is not applicable to the current platform.
511 ConvertFlagsToSwitches(&flags_storage_, &command_line, kAddSentinels);
512 EXPECT_TRUE(command_line.HasSwitch(kSwitch1));
513 EXPECT_FALSE(command_line.HasSwitch(kSwitch3));
515 // Experiment 3 should show still be persisted in preferences though.
516 const base::ListValue* experiments_list =
517 prefs_.GetList(prefs::kEnabledLabsExperiments);
518 ASSERT_TRUE(experiments_list);
519 EXPECT_EQ(2U, experiments_list->GetSize());
520 std::string s0;
521 ASSERT_TRUE(experiments_list->GetString(0, &s0));
522 EXPECT_EQ(kFlags1, s0);
523 std::string s1;
524 ASSERT_TRUE(experiments_list->GetString(1, &s1));
525 EXPECT_EQ(kFlags3, s1);
528 // Tests that switches which should have values get them in the command
529 // line.
530 TEST_F(AboutFlagsTest, CheckValues) {
531 // Enable experiments 1 and 2.
532 SetExperimentEnabled(&flags_storage_, kFlags1, true);
533 SetExperimentEnabled(&flags_storage_, kFlags2, true);
534 CommandLine command_line(CommandLine::NO_PROGRAM);
535 EXPECT_FALSE(command_line.HasSwitch(kSwitch1));
536 EXPECT_FALSE(command_line.HasSwitch(kSwitch2));
538 // Convert the flags to switches.
539 ConvertFlagsToSwitches(&flags_storage_, &command_line, kAddSentinels);
540 EXPECT_TRUE(command_line.HasSwitch(kSwitch1));
541 EXPECT_EQ(std::string(), command_line.GetSwitchValueASCII(kSwitch1));
542 EXPECT_TRUE(command_line.HasSwitch(kSwitch2));
543 EXPECT_EQ(std::string(kValueForSwitch2),
544 command_line.GetSwitchValueASCII(kSwitch2));
546 // Confirm that there is no '=' in the command line for simple switches.
547 std::string switch1_with_equals = std::string("--") +
548 std::string(kSwitch1) +
549 std::string("=");
550 #if defined(OS_WIN)
551 EXPECT_EQ(std::wstring::npos,
552 command_line.GetCommandLineString().find(
553 base::ASCIIToWide(switch1_with_equals)));
554 #else
555 EXPECT_EQ(std::string::npos,
556 command_line.GetCommandLineString().find(switch1_with_equals));
557 #endif
559 // And confirm there is a '=' for switches with values.
560 std::string switch2_with_equals = std::string("--") +
561 std::string(kSwitch2) +
562 std::string("=");
563 #if defined(OS_WIN)
564 EXPECT_NE(std::wstring::npos,
565 command_line.GetCommandLineString().find(
566 base::ASCIIToWide(switch2_with_equals)));
567 #else
568 EXPECT_NE(std::string::npos,
569 command_line.GetCommandLineString().find(switch2_with_equals));
570 #endif
572 // And it should persist.
573 const base::ListValue* experiments_list =
574 prefs_.GetList(prefs::kEnabledLabsExperiments);
575 ASSERT_TRUE(experiments_list);
576 EXPECT_EQ(2U, experiments_list->GetSize());
577 std::string s0;
578 ASSERT_TRUE(experiments_list->GetString(0, &s0));
579 EXPECT_EQ(kFlags1, s0);
580 std::string s1;
581 ASSERT_TRUE(experiments_list->GetString(1, &s1));
582 EXPECT_EQ(kFlags2, s1);
585 // Tests multi-value type experiments.
586 TEST_F(AboutFlagsTest, MultiValues) {
587 const Experiment& experiment = kExperiments[3];
588 ASSERT_EQ(kFlags4, experiment.internal_name);
590 // Initially, the first "deactivated" option of the multi experiment should
591 // be set.
593 CommandLine command_line(CommandLine::NO_PROGRAM);
594 ConvertFlagsToSwitches(&flags_storage_, &command_line, kAddSentinels);
595 EXPECT_FALSE(command_line.HasSwitch(kMultiSwitch1));
596 EXPECT_FALSE(command_line.HasSwitch(kMultiSwitch2));
599 // Enable the 2nd choice of the multi-value.
600 SetExperimentEnabled(&flags_storage_, experiment.NameForChoice(2), true);
602 CommandLine command_line(CommandLine::NO_PROGRAM);
603 ConvertFlagsToSwitches(&flags_storage_, &command_line, kAddSentinels);
604 EXPECT_FALSE(command_line.HasSwitch(kMultiSwitch1));
605 EXPECT_TRUE(command_line.HasSwitch(kMultiSwitch2));
606 EXPECT_EQ(std::string(kValueForMultiSwitch2),
607 command_line.GetSwitchValueASCII(kMultiSwitch2));
610 // Disable the multi-value experiment.
611 SetExperimentEnabled(&flags_storage_, experiment.NameForChoice(0), true);
613 CommandLine command_line(CommandLine::NO_PROGRAM);
614 ConvertFlagsToSwitches(&flags_storage_, &command_line, kAddSentinels);
615 EXPECT_FALSE(command_line.HasSwitch(kMultiSwitch1));
616 EXPECT_FALSE(command_line.HasSwitch(kMultiSwitch2));
620 TEST_F(AboutFlagsTest, EnableDisableValues) {
621 const Experiment& experiment = kExperiments[4];
622 ASSERT_EQ(kFlags5, experiment.internal_name);
624 // Nothing selected.
626 CommandLine command_line(CommandLine::NO_PROGRAM);
627 ConvertFlagsToSwitches(&flags_storage_, &command_line, kAddSentinels);
628 EXPECT_FALSE(command_line.HasSwitch(kSwitch1));
629 EXPECT_FALSE(command_line.HasSwitch(kSwitch2));
632 // "Enable" option selected.
633 SetExperimentEnabled(&flags_storage_, experiment.NameForChoice(1), true);
635 CommandLine command_line(CommandLine::NO_PROGRAM);
636 ConvertFlagsToSwitches(&flags_storage_, &command_line, kAddSentinels);
637 EXPECT_TRUE(command_line.HasSwitch(kSwitch1));
638 EXPECT_FALSE(command_line.HasSwitch(kSwitch2));
639 EXPECT_EQ(kEnableDisableValue1, command_line.GetSwitchValueASCII(kSwitch1));
642 // "Disable" option selected.
643 SetExperimentEnabled(&flags_storage_, experiment.NameForChoice(2), true);
645 CommandLine command_line(CommandLine::NO_PROGRAM);
646 ConvertFlagsToSwitches(&flags_storage_, &command_line, kAddSentinels);
647 EXPECT_FALSE(command_line.HasSwitch(kSwitch1));
648 EXPECT_TRUE(command_line.HasSwitch(kSwitch2));
649 EXPECT_EQ(kEnableDisableValue2, command_line.GetSwitchValueASCII(kSwitch2));
652 // "Default" option selected, same as nothing selected.
653 SetExperimentEnabled(&flags_storage_, experiment.NameForChoice(0), true);
655 CommandLine command_line(CommandLine::NO_PROGRAM);
656 ConvertFlagsToSwitches(&flags_storage_, &command_line, kAddSentinels);
657 EXPECT_FALSE(command_line.HasSwitch(kMultiSwitch1));
658 EXPECT_FALSE(command_line.HasSwitch(kMultiSwitch2));
662 // Makes sure there are no separators in any of the experiment names.
663 TEST_F(AboutFlagsTest, NoSeparators) {
664 testing::SetExperiments(NULL, 0);
665 size_t count;
666 const Experiment* experiments = testing::GetExperiments(&count);
667 for (size_t i = 0; i < count; ++i) {
668 std::string name = experiments->internal_name;
669 EXPECT_EQ(std::string::npos, name.find(testing::kMultiSeparator)) << i;
673 class AboutFlagsHistogramTest : public ::testing::Test {
674 protected:
675 // This is a helper function to check that all IDs in enum LoginCustomFlags in
676 // histograms.xml are unique.
677 void SetSwitchToHistogramIdMapping(const std::string& switch_name,
678 const Sample switch_histogram_id,
679 std::map<std::string, Sample>* out_map) {
680 const std::pair<std::map<std::string, Sample>::iterator, bool> status =
681 out_map->insert(std::make_pair(switch_name, switch_histogram_id));
682 if (!status.second) {
683 EXPECT_TRUE(status.first->second == switch_histogram_id)
684 << "Duplicate switch '" << switch_name
685 << "' found in enum 'LoginCustomFlags' in histograms.xml.";
689 // This method generates a hint for the user for what string should be added
690 // to the enum LoginCustomFlags to make in consistent.
691 std::string GetHistogramEnumEntryText(const std::string& switch_name,
692 Sample value) {
693 return base::StringPrintf(
694 "<int value=\"%d\" label=\"%s\"/>", value, switch_name.c_str());
698 TEST_F(AboutFlagsHistogramTest, CheckHistograms) {
699 base::FilePath histograms_xml_file_path;
700 ASSERT_TRUE(
701 PathService::Get(base::DIR_SOURCE_ROOT, &histograms_xml_file_path));
702 histograms_xml_file_path = histograms_xml_file_path.AppendASCII("tools")
703 .AppendASCII("metrics")
704 .AppendASCII("histograms")
705 .AppendASCII("histograms.xml");
707 XmlReader histograms_xml;
708 ASSERT_TRUE(histograms_xml.LoadFile(
709 FilePathStringTypeToString(histograms_xml_file_path.value())));
710 std::map<Sample, std::string> login_custom_flags =
711 ReadEnumFromHistogramsXml("LoginCustomFlags", &histograms_xml);
712 ASSERT_TRUE(login_custom_flags.size())
713 << "Error reading enum 'LoginCustomFlags' from histograms.xml.";
715 // Build reverse map {switch_name => id} from login_custom_flags.
716 SwitchToIdMap histograms_xml_switches_ids;
718 EXPECT_TRUE(login_custom_flags.count(kBadSwitchFormatHistogramId))
719 << "Entry for UMA ID of incorrect command-line flag is not found in "
720 "histograms.xml enum LoginCustomFlags. "
721 "Consider adding entry:\n"
722 << " " << GetHistogramEnumEntryText("BAD_FLAG_FORMAT", 0);
723 // Check that all LoginCustomFlags entries have correct values.
724 for (std::map<Sample, std::string>::const_iterator it =
725 login_custom_flags.begin();
726 it != login_custom_flags.end();
727 ++it) {
728 if (it->first == kBadSwitchFormatHistogramId) {
729 // Add eror value with empty name.
730 SetSwitchToHistogramIdMapping(
731 "", it->first, &histograms_xml_switches_ids);
732 continue;
734 const Sample uma_id = GetSwitchUMAId(it->second);
735 EXPECT_EQ(uma_id, it->first)
736 << "histograms.xml enum LoginCustomFlags "
737 "entry '" << it->second << "' has incorrect value=" << it->first
738 << ", but " << uma_id << " is expected. Consider changing entry to:\n"
739 << " " << GetHistogramEnumEntryText(it->second, uma_id);
740 SetSwitchToHistogramIdMapping(
741 it->second, it->first, &histograms_xml_switches_ids);
744 // Check that all flags in about_flags.cc have entries in login_custom_flags.
745 std::set<std::string> all_switches = GetAllSwitchesForTesting();
746 for (std::set<std::string>::const_iterator it = all_switches.begin();
747 it != all_switches.end();
748 ++it) {
749 // Skip empty placeholders.
750 if (it->empty())
751 continue;
752 const Sample uma_id = GetSwitchUMAId(*it);
753 EXPECT_NE(kBadSwitchFormatHistogramId, uma_id)
754 << "Command-line switch '" << *it
755 << "' from about_flags.cc has UMA ID equal to reserved value "
756 "kBadSwitchFormatHistogramId=" << kBadSwitchFormatHistogramId
757 << ". Please modify switch name.";
758 SwitchToIdMap::iterator enum_entry =
759 histograms_xml_switches_ids.lower_bound(*it);
761 // Ignore case here when switch ID is incorrect - it has already been
762 // reported in the previous loop.
763 EXPECT_TRUE(enum_entry != histograms_xml_switches_ids.end() &&
764 enum_entry->first == *it)
765 << "histograms.xml enum LoginCustomFlags doesn't contain switch '"
766 << *it << "' (value=" << uma_id
767 << " expected). Consider adding entry:\n"
768 << " " << GetHistogramEnumEntryText(*it, uma_id);
772 } // namespace about_flags