blob: d5bb64e27a3339cc61dfc33e081ef9b9fbe4bd76 [file] [log] [blame]
[email protected]3828d6f2011-02-24 18:32:211// Copyright (c) 2011 The Chromium Authors. All rights reserved.
[email protected]ad2a3ded2010-08-27 13:19:052// Use of this source code is governed by a BSD-style license that can be
3// found in the LICENSE file.
4
[email protected]1a47d7e2010-10-15 00:37:245#include "chrome/browser/about_flags.h"
[email protected]ad2a3ded2010-08-27 13:19:056
7#include <algorithm>
8#include <iterator>
9#include <map>
10#include <set>
11
[email protected]ad2a3ded2010-08-27 13:19:0512#include "base/command_line.h"
[email protected]3b63f8f42011-03-28 01:54:1513#include "base/memory/singleton.h"
[email protected]8a6ff28d2010-12-02 16:35:1914#include "base/string_number_conversions.h"
[email protected]ad2a3ded2010-08-27 13:19:0515#include "base/values.h"
[email protected]4bc5050c2010-11-18 17:55:5416#include "chrome/browser/metrics/user_metrics.h"
[email protected]ad2a3ded2010-08-27 13:19:0517#include "chrome/browser/prefs/pref_service.h"
[email protected]1bc78422011-03-31 08:41:3818#include "chrome/browser/prefs/scoped_user_pref_update.h"
[email protected]ad2a3ded2010-08-27 13:19:0519#include "chrome/common/chrome_switches.h"
20#include "chrome/common/pref_names.h"
21#include "grit/generated_resources.h"
[email protected]c051a1b2011-01-21 23:30:1722#include "ui/base/l10n/l10n_util.h"
[email protected]ad2a3ded2010-08-27 13:19:0523
[email protected]1a47d7e2010-10-15 00:37:2424namespace about_flags {
[email protected]ad2a3ded2010-08-27 13:19:0525
[email protected]8a6ff28d2010-12-02 16:35:1926// Macros to simplify specifying the type.
[email protected]a82744532011-02-11 16:15:5327#define SINGLE_VALUE_TYPE_AND_VALUE(command_line_switch, switch_value) \
28 Experiment::SINGLE_VALUE, command_line_switch, switch_value, NULL, 0
29#define SINGLE_VALUE_TYPE(command_line_switch) \
30 SINGLE_VALUE_TYPE_AND_VALUE(command_line_switch, "")
31#define MULTI_VALUE_TYPE(choices) \
[email protected]0e6f56d2011-02-12 23:45:1532 Experiment::MULTI_VALUE, "", "", choices, arraysize(choices)
[email protected]8a6ff28d2010-12-02 16:35:1933
[email protected]e2ddbc92010-10-15 20:02:0734namespace {
35
[email protected]a314ee5a2010-10-26 21:23:2836const unsigned kOsAll = kOsMac | kOsWin | kOsLinux | kOsCrOS;
[email protected]ad2a3ded2010-08-27 13:19:0537
[email protected]ba8164242010-11-16 21:31:0038// Names for former Chrome OS Labs experiments, shared with prefs migration
39// code.
40const char kMediaPlayerExperimentName[] = "media-player";
41const char kAdvancedFileSystemExperimentName[] = "advanced-file-system";
42const char kVerticalTabsExperimentName[] = "vertical-tabs";
43
[email protected]4bc5050c2010-11-18 17:55:5444// RECORDING USER METRICS FOR FLAGS:
45// -----------------------------------------------------------------------------
46// The first line of the experiment is the internal name. If you'd like to
47// gather statistics about the usage of your flag, you should append a marker
48// comment to the end of the feature name, like so:
49// "my-special-feature", // FLAGS:RECORD_UMA
50//
51// After doing that, run //chrome/tools/extract_actions.py (see instructions at
52// the top of that file for details) to update the chromeactions.txt file, which
53// will enable UMA to record your feature flag.
54//
55// After your feature has shipped under a flag, you can locate the metrics
56// under the action name AboutFlags_internal-action-name. Actions are recorded
57// once per startup, so you should divide this number by AboutFlags_StartupTick
58// to get a sense of usage. Note that this will not be the same as number of
59// users with a given feature enabled because users can quit and relaunch
60// the application multiple times over a given time interval.
61// TODO(rsesek): See if there's a way to count per-user, rather than
62// per-startup.
63
[email protected]8a6ff28d2010-12-02 16:35:1964// To add a new experiment add to the end of kExperiments. There are two
65// distinct types of experiments:
66// . SINGLE_VALUE: experiment is either on or off. Use the SINGLE_VALUE_TYPE
67// macro for this type supplying the command line to the macro.
[email protected]28e35af2011-02-09 12:56:2268// . MULTI_VALUE: a list of choices, the first of which should correspond to a
69// deactivated state for this lab (i.e. no command line option). To specify
70// this type of experiment use the macro MULTI_VALUE_TYPE supplying it the
71// array of choices.
[email protected]8a6ff28d2010-12-02 16:35:1972// See the documentation of Experiment for details on the fields.
73//
74// When adding a new choice, add it to the end of the list.
[email protected]ad2a3ded2010-08-27 13:19:0575const Experiment kExperiments[] = {
76 {
[email protected]aac169d2011-03-18 19:53:0377 "expose-for-tabs", // FLAGS:RECORD_UMA
78 IDS_FLAGS_TABPOSE_NAME,
79 IDS_FLAGS_TABPOSE_DESCRIPTION,
80 kOsMac,
81#if defined(OS_MACOSX)
82 // The switch exists only on OS X.
83 SINGLE_VALUE_TYPE(switches::kEnableExposeForTabs)
84#else
85 SINGLE_VALUE_TYPE("")
86#endif
87 },
88 {
[email protected]ba8164242010-11-16 21:31:0089 kMediaPlayerExperimentName,
90 IDS_FLAGS_MEDIA_PLAYER_NAME,
91 IDS_FLAGS_MEDIA_PLAYER_DESCRIPTION,
92 kOsCrOS,
93#if defined(OS_CHROMEOS)
94 // The switch exists only on Chrome OS.
[email protected]8a6ff28d2010-12-02 16:35:1995 SINGLE_VALUE_TYPE(switches::kEnableMediaPlayer)
[email protected]ba8164242010-11-16 21:31:0096#else
[email protected]8a6ff28d2010-12-02 16:35:1997 SINGLE_VALUE_TYPE("")
[email protected]ba8164242010-11-16 21:31:0098#endif
99 },
100 {
101 kAdvancedFileSystemExperimentName,
102 IDS_FLAGS_ADVANCED_FS_NAME,
103 IDS_FLAGS_ADVANCED_FS_DESCRIPTION,
104 kOsCrOS,
105#if defined(OS_CHROMEOS)
106 // The switch exists only on Chrome OS.
[email protected]8a6ff28d2010-12-02 16:35:19107 SINGLE_VALUE_TYPE(switches::kEnableAdvancedFileSystem)
[email protected]ba8164242010-11-16 21:31:00108#else
[email protected]8a6ff28d2010-12-02 16:35:19109 SINGLE_VALUE_TYPE("")
[email protected]ba8164242010-11-16 21:31:00110#endif
111 },
112 {
[email protected]4bc5050c2010-11-18 17:55:54113 "vertical-tabs", // FLAGS:RECORD_UMA
[email protected]9486c1f2010-10-14 19:52:12114 IDS_FLAGS_SIDE_TABS_NAME,
115 IDS_FLAGS_SIDE_TABS_DESCRIPTION,
[email protected]ba8164242010-11-16 21:31:00116 kOsWin | kOsCrOS,
[email protected]8a6ff28d2010-12-02 16:35:19117 SINGLE_VALUE_TYPE(switches::kEnableVerticalTabs)
[email protected]654151872010-09-13 22:43:05118 },
119 {
[email protected]4bc5050c2010-11-18 17:55:54120 "remoting", // FLAGS:RECORD_UMA
[email protected]9486c1f2010-10-14 19:52:12121 IDS_FLAGS_REMOTING_NAME,
[email protected]d99599862011-01-20 22:43:59122 IDS_FLAGS_REMOTING_DESCRIPTION,
123 kOsAll,
[email protected]8a6ff28d2010-12-02 16:35:19124 SINGLE_VALUE_TYPE(switches::kEnableRemoting)
[email protected]52fa2d52010-09-25 14:08:56125 },
[email protected]a6f03652010-09-28 15:59:07126 {
[email protected]4bc5050c2010-11-18 17:55:54127 "conflicting-modules-check", // FLAGS:RECORD_UMA
[email protected]c1bbaa82010-11-08 11:17:05128 IDS_FLAGS_CONFLICTS_CHECK_NAME,
129 IDS_FLAGS_CONFLICTS_CHECK_DESCRIPTION,
130 kOsWin,
[email protected]8a6ff28d2010-12-02 16:35:19131 SINGLE_VALUE_TYPE(switches::kConflictingModulesCheck)
[email protected]c1bbaa82010-11-08 11:17:05132 },
133 {
[email protected]4bc5050c2010-11-18 17:55:54134 "cloud-print-proxy", // FLAGS:RECORD_UMA
[email protected]9486c1f2010-10-14 19:52:12135 IDS_FLAGS_CLOUD_PRINT_PROXY_NAME,
136 IDS_FLAGS_CLOUD_PRINT_PROXY_DESCRIPTION,
[email protected]fa6d2a2f2010-11-30 21:47:19137#if defined(GOOGLE_CHROME_BUILD)
[email protected]9f8872b32011-03-04 19:44:45138 // For a Chrome build, we know we have a PDF plug-in on Windows, so it's
[email protected]be51dd252011-03-05 00:23:44139 // fully enabled. Linux still need some final polish.
140 kOsLinux,
[email protected]fa6d2a2f2010-11-30 21:47:19141#else
[email protected]be51dd252011-03-05 00:23:44142 // Otherwise, where we know Windows could be working if a viable PDF
143 // plug-in could be supplied, we'll keep the lab enabled. Mac always has
144 // PDF rasterization available, so no flag needed there.
145 kOsWin | kOsLinux,
[email protected]fa6d2a2f2010-11-30 21:47:19146#endif
[email protected]8a6ff28d2010-12-02 16:35:19147 SINGLE_VALUE_TYPE(switches::kEnableCloudPrintProxy)
[email protected]8b6588a2010-10-12 02:39:42148 },
[email protected]580939a2010-10-12 18:54:37149 {
[email protected]bb461532010-11-26 21:50:23150 "crxless-web-apps",
151 IDS_FLAGS_CRXLESS_WEB_APPS_NAME,
152 IDS_FLAGS_CRXLESS_WEB_APPS_DESCRIPTION,
153 kOsAll,
[email protected]8a6ff28d2010-12-02 16:35:19154 SINGLE_VALUE_TYPE(switches::kEnableCrxlessWebApps)
[email protected]bb461532010-11-26 21:50:23155 },
156 {
[email protected]5963b772011-02-09 22:55:38157 "composited-layer-borders",
158 IDS_FLAGS_COMPOSITED_LAYER_BORDERS,
159 IDS_FLAGS_COMPOSITED_LAYER_BORDERS_DESCRIPTION,
160 kOsAll,
161 SINGLE_VALUE_TYPE(switches::kShowCompositedLayerBorders)
162 },
163 {
[email protected]a8f1eaa2011-03-07 19:00:58164 "show-fps-counter",
165 IDS_FLAGS_SHOW_FPS_COUNTER,
166 IDS_FLAGS_SHOW_FPS_COUNTER_DESCRIPTION,
167 kOsAll,
168 SINGLE_VALUE_TYPE(switches::kShowFPSCounter)
169 },
170 {
[email protected]4bc5050c2010-11-18 17:55:54171 "gpu-canvas-2d", // FLAGS:RECORD_UMA
[email protected]9486c1f2010-10-14 19:52:12172 IDS_FLAGS_ACCELERATED_CANVAS_2D_NAME,
173 IDS_FLAGS_ACCELERATED_CANVAS_2D_DESCRIPTION,
[email protected]1af1dfe2010-10-19 23:49:14174 kOsWin | kOsLinux | kOsCrOS,
[email protected]8a6ff28d2010-12-02 16:35:19175 SINGLE_VALUE_TYPE(switches::kEnableAccelerated2dCanvas)
[email protected]8d260e52010-10-13 01:03:05176 },
[email protected]11c4c812010-10-22 19:50:12177 // FIXME(scheib): Add Flags entry for WebGL,
178 // or pull it and the strings in generated_resources.grd by Dec 2010
179 // {
[email protected]4bc5050c2010-11-18 17:55:54180 // "webgl",
[email protected]11c4c812010-10-22 19:50:12181 // IDS_FLAGS_WEBGL_NAME,
182 // IDS_FLAGS_WEBGL_DESCRIPTION,
183 // kOsAll,
[email protected]8a6ff28d2010-12-02 16:35:19184 // SINGLE_VALUE_TYPE(switches::kDisableExperimentalWebGL)
[email protected]11c4c812010-10-22 19:50:12185 // }
[email protected]8d260e52010-10-13 01:03:05186 {
[email protected]4bc5050c2010-11-18 17:55:54187 "print-preview", // FLAGS:RECORD_UMA
[email protected]9486c1f2010-10-14 19:52:12188 IDS_FLAGS_PRINT_PREVIEW_NAME,
189 IDS_FLAGS_PRINT_PREVIEW_DESCRIPTION,
[email protected]411c37f2011-03-25 22:55:34190 kOsMac | kOsWin | kOsLinux, // This switch is not available in CrOS.
[email protected]8a6ff28d2010-12-02 16:35:19191 SINGLE_VALUE_TYPE(switches::kEnablePrintPreview)
[email protected]2fe15fcb2010-10-21 20:39:53192 },
193 {
[email protected]4bc5050c2010-11-18 17:55:54194 "enable-nacl", // FLAGS:RECORD_UMA
[email protected]4ff87d202010-11-06 01:28:40195 IDS_FLAGS_ENABLE_NACL_NAME,
196 IDS_FLAGS_ENABLE_NACL_DESCRIPTION,
197 kOsAll,
[email protected]8a6ff28d2010-12-02 16:35:19198 SINGLE_VALUE_TYPE(switches::kEnableNaCl)
[email protected]4ff87d202010-11-06 01:28:40199 },
200 {
[email protected]4bc5050c2010-11-18 17:55:54201 "dns-server", // FLAGS:RECORD_UMA
[email protected]2fe15fcb2010-10-21 20:39:53202 IDS_FLAGS_DNS_SERVER_NAME,
203 IDS_FLAGS_DNS_SERVER_DESCRIPTION,
204 kOsLinux,
[email protected]8a6ff28d2010-12-02 16:35:19205 SINGLE_VALUE_TYPE(switches::kDnsServer)
[email protected]2fe15fcb2010-10-21 20:39:53206 },
[email protected]177aceb2010-11-03 16:17:41207 {
[email protected]4bc5050c2010-11-18 17:55:54208 "extension-apis", // FLAGS:RECORD_UMA
[email protected]11dd68cd52010-11-12 01:15:32209 IDS_FLAGS_EXPERIMENTAL_EXTENSION_APIS_NAME,
210 IDS_FLAGS_EXPERIMENTAL_EXTENSION_APIS_DESCRIPTION,
211 kOsAll,
[email protected]8a6ff28d2010-12-02 16:35:19212 SINGLE_VALUE_TYPE(switches::kEnableExperimentalExtensionApis)
[email protected]11dd68cd52010-11-12 01:15:32213 },
[email protected]3627b06d2010-11-12 16:36:16214 {
[email protected]4bc5050c2010-11-18 17:55:54215 "click-to-play", // FLAGS:RECORD_UMA
[email protected]3627b06d2010-11-12 16:36:16216 IDS_FLAGS_CLICK_TO_PLAY_NAME,
217 IDS_FLAGS_CLICK_TO_PLAY_DESCRIPTION,
218 kOsAll,
[email protected]8a6ff28d2010-12-02 16:35:19219 SINGLE_VALUE_TYPE(switches::kEnableClickToPlay)
[email protected]3627b06d2010-11-12 16:36:16220 },
[email protected]80bd24e2010-11-30 09:34:38221 {
222 "disable-hyperlink-auditing",
223 IDS_FLAGS_DISABLE_HYPERLINK_AUDITING_NAME,
224 IDS_FLAGS_DISABLE_HYPERLINK_AUDITING_DESCRIPTION,
225 kOsAll,
[email protected]8a6ff28d2010-12-02 16:35:19226 SINGLE_VALUE_TYPE(switches::kNoPings)
[email protected]feb28fef2010-12-01 10:52:51227 },
228 {
229 "experimental-location-features", // FLAGS:RECORD_UMA
230 IDS_FLAGS_EXPERIMENTAL_LOCATION_FEATURES_NAME,
231 IDS_FLAGS_EXPERIMENTAL_LOCATION_FEATURES_DESCRIPTION,
232 kOsMac | kOsWin | kOsLinux, // Currently does nothing on CrOS.
[email protected]8a6ff28d2010-12-02 16:35:19233 SINGLE_VALUE_TYPE(switches::kExperimentalLocationFeatures)
234 },
235 {
[email protected]4c6452d2011-01-12 08:47:57236 "block-reading-third-party-cookies",
237 IDS_FLAGS_BLOCK_ALL_THIRD_PARTY_COOKIES_NAME,
238 IDS_FLAGS_BLOCK_ALL_THIRD_PARTY_COOKIES_DESCRIPTION,
239 kOsAll,
240 SINGLE_VALUE_TYPE(switches::kBlockReadingThirdPartyCookies)
241 },
[email protected]04227962011-01-20 02:03:09242 {
243 "disable-interactive-form-validation",
244 IDS_FLAGS_DISABLE_INTERACTIVE_FORM_VALIDATION_NAME,
245 IDS_FLAGS_DISABLE_INTERACTIVE_FORM_VALIDATION_DESCRIPTION,
246 kOsAll,
247 SINGLE_VALUE_TYPE(switches::kDisableInteractiveFormValidation)
248 },
[email protected]8df51192011-01-22 20:05:03249 {
250 "webaudio",
251 IDS_FLAGS_WEBAUDIO_NAME,
252 IDS_FLAGS_WEBAUDIO_DESCRIPTION,
253 kOsMac, // TODO(crogers): add windows and linux when FFT is ready.
254 SINGLE_VALUE_TYPE(switches::kEnableWebAudio)
255 },
[email protected]3828d6f2011-02-24 18:32:21256 {
[email protected]6c54e7e42011-03-02 20:52:34257 "p2papi",
258 IDS_FLAGS_P2P_API_NAME,
259 IDS_FLAGS_P2P_API_DESCRIPTION,
260 kOsAll,
261 SINGLE_VALUE_TYPE(switches::kEnableP2PApi)
262 },
[email protected]07d490bc2011-03-07 17:05:26263 {
264 "focus-existing-tab-on-open", // FLAGS:RECORD_UMA
265 IDS_FLAGS_FOCUS_EXISTING_TAB_ON_OPEN_NAME,
266 IDS_FLAGS_FOCUS_EXISTING_TAB_ON_OPEN_DESCRIPTION,
267 kOsAll,
268 SINGLE_VALUE_TYPE(switches::kFocusExistingTabOnOpen)
269 },
[email protected]d5dcfb32011-03-19 00:49:24270 {
271 "new-tab-page-4",
272 IDS_FLAGS_NEW_TAB_PAGE_4_NAME,
273 IDS_FLAGS_NEW_TAB_PAGE_4_DESCRIPTION,
274 kOsAll,
275 SINGLE_VALUE_TYPE(switches::kNewTabPage4)
276 },
[email protected]5aea1862011-03-23 23:55:39277 {
278 "tab-groups-context-menu",
279 IDS_FLAGS_TAB_GROUPS_CONTEXT_MENU_NAME,
280 IDS_FLAGS_TAB_GROUPS_CONTEXT_MENU_DESCRIPTION,
281 kOsWin,
282 SINGLE_VALUE_TYPE(switches::kEnableTabGroupsContextMenu)
283 },
[email protected]e9bf9d92011-03-31 20:57:15284 {
285 "ppapi-flash-in-process",
286 IDS_FLAGS_PPAPI_FLASH_IN_PROCESS_NAME,
287 IDS_FLAGS_PPAPI_FLASH_IN_PROCESS_DESCRIPTION,
[email protected]f2d3ce0d2011-04-01 18:19:30288 kOsAll,
[email protected]e9bf9d92011-03-31 20:57:15289 SINGLE_VALUE_TYPE(switches::kPpapiFlashInProcess)
290 },
[email protected]360d2ec2011-04-11 22:53:36291#if defined(TOOLKIT_GTK)
292 {
293 "global-gnome-menu",
294 IDS_FLAGS_LINUX_GLOBAL_MENUBAR_NAME,
295 IDS_FLAGS_LINUX_GLOBAL_MENUBAR_DESCRIPTION,
296 kOsLinux,
297 SINGLE_VALUE_TYPE(switches::kGlobalGnomeMenu)
298 },
299#endif
[email protected]ce5737142011-04-14 23:35:09300 {
301 "enable-experimental-eap",
302 IDS_FLAGS_ENABLE_EXPERIMENTAL_EAP_NAME,
303 IDS_FLAGS_ENABLE_EXPERIMENTAL_EAP_DESCRIPTION,
304 kOsCrOS,
305#if defined(OS_CHROMEOS)
306 // The switch exists only on Chrome OS.
307 SINGLE_VALUE_TYPE(switches::kEnableExperimentalEap)
308#else
309 SINGLE_VALUE_TYPE("")
310#endif
311 },
[email protected]c3647902011-04-15 06:40:33312 {
313 "enable-vpn",
314 IDS_FLAGS_ENABLE_VPN_NAME,
315 IDS_FLAGS_ENABLE_VPN_DESCRIPTION,
316 kOsCrOS,
317#if defined(OS_CHROMEOS)
318 // The switch exists only on Chrome OS.
319 SINGLE_VALUE_TYPE(switches::kEnableVPN)
320#else
321 SINGLE_VALUE_TYPE("")
322#endif
323 },
[email protected]ad2a3ded2010-08-27 13:19:05324};
325
[email protected]a314ee5a2010-10-26 21:23:28326const Experiment* experiments = kExperiments;
327size_t num_experiments = arraysize(kExperiments);
328
[email protected]e2ddbc92010-10-15 20:02:07329// Stores and encapsulates the little state that about:flags has.
330class FlagsState {
331 public:
332 FlagsState() : needs_restart_(false) {}
333 void ConvertFlagsToSwitches(PrefService* prefs, CommandLine* command_line);
334 bool IsRestartNeededToCommitChanges();
335 void SetExperimentEnabled(
[email protected]a314ee5a2010-10-26 21:23:28336 PrefService* prefs, const std::string& internal_name, bool enable);
[email protected]e2ddbc92010-10-15 20:02:07337 void RemoveFlagsSwitches(
338 std::map<std::string, CommandLine::StringType>* switch_list);
339 void reset();
340
341 // Returns the singleton instance of this class
[email protected]8e8bb6d2010-12-13 08:18:55342 static FlagsState* GetInstance() {
[email protected]e2ddbc92010-10-15 20:02:07343 return Singleton<FlagsState>::get();
344 }
345
346 private:
347 bool needs_restart_;
[email protected]a82744532011-02-11 16:15:53348 std::map<std::string, std::string> flags_switches_;
[email protected]e2ddbc92010-10-15 20:02:07349
350 DISALLOW_COPY_AND_ASSIGN(FlagsState);
351};
352
[email protected]c7b7800a2010-10-07 18:51:35353// Extracts the list of enabled lab experiments from preferences and stores them
[email protected]ad2a3ded2010-08-27 13:19:05354// in a set.
[email protected]1a47d7e2010-10-15 00:37:24355void GetEnabledFlags(const PrefService* prefs, std::set<std::string>* result) {
[email protected]ad2a3ded2010-08-27 13:19:05356 const ListValue* enabled_experiments = prefs->GetList(
357 prefs::kEnabledLabsExperiments);
358 if (!enabled_experiments)
359 return;
360
361 for (ListValue::const_iterator it = enabled_experiments->begin();
362 it != enabled_experiments->end();
363 ++it) {
364 std::string experiment_name;
365 if (!(*it)->GetAsString(&experiment_name)) {
366 LOG(WARNING) << "Invalid entry in " << prefs::kEnabledLabsExperiments;
367 continue;
368 }
369 result->insert(experiment_name);
370 }
371}
372
373// Takes a set of enabled lab experiments
[email protected]1a47d7e2010-10-15 00:37:24374void SetEnabledFlags(
[email protected]ad2a3ded2010-08-27 13:19:05375 PrefService* prefs, const std::set<std::string>& enabled_experiments) {
[email protected]1bc78422011-03-31 08:41:38376 ListPrefUpdate update(prefs, prefs::kEnabledLabsExperiments);
377 ListValue* experiments_list = update.Get();
[email protected]ad2a3ded2010-08-27 13:19:05378
379 experiments_list->Clear();
380 for (std::set<std::string>::const_iterator it = enabled_experiments.begin();
381 it != enabled_experiments.end();
382 ++it) {
383 experiments_list->Append(new StringValue(*it));
384 }
385}
386
[email protected]8a6ff28d2010-12-02 16:35:19387// Returns the name used in prefs for the choice at the specified index.
388std::string NameForChoice(const Experiment& e, int index) {
[email protected]2ce9c89752011-02-25 18:24:34389 DCHECK_EQ(Experiment::MULTI_VALUE, e.type);
390 DCHECK_LT(index, e.num_choices);
[email protected]8a6ff28d2010-12-02 16:35:19391 return std::string(e.internal_name) + about_flags::testing::kMultiSeparator +
392 base::IntToString(index);
393}
394
395// Adds the internal names for the specified experiment to |names|.
396void AddInternalName(const Experiment& e, std::set<std::string>* names) {
397 if (e.type == Experiment::SINGLE_VALUE) {
398 names->insert(e.internal_name);
399 } else {
[email protected]2ce9c89752011-02-25 18:24:34400 DCHECK_EQ(Experiment::MULTI_VALUE, e.type);
[email protected]8a6ff28d2010-12-02 16:35:19401 for (int i = 0; i < e.num_choices; ++i)
402 names->insert(NameForChoice(e, i));
403 }
404}
405
[email protected]28e35af2011-02-09 12:56:22406// Confirms that an experiment is valid, used in a DCHECK in
407// SanitizeList below.
408bool ValidateExperiment(const Experiment& e) {
409 switch (e.type) {
410 case Experiment::SINGLE_VALUE:
411 DCHECK_EQ(0, e.num_choices);
412 DCHECK(!e.choices);
413 break;
414 case Experiment::MULTI_VALUE:
415 DCHECK_GT(e.num_choices, 0);
416 DCHECK(e.choices);
[email protected]a82744532011-02-11 16:15:53417 DCHECK(e.choices[0].command_line_switch);
418 DCHECK_EQ('\0', e.choices[0].command_line_switch[0]);
[email protected]28e35af2011-02-09 12:56:22419 break;
420 default:
421 NOTREACHED();
422 }
423 return true;
424}
425
[email protected]ad2a3ded2010-08-27 13:19:05426// Removes all experiments from prefs::kEnabledLabsExperiments that are
427// unknown, to prevent this list to become very long as experiments are added
428// and removed.
429void SanitizeList(PrefService* prefs) {
430 std::set<std::string> known_experiments;
[email protected]28e35af2011-02-09 12:56:22431 for (size_t i = 0; i < num_experiments; ++i) {
432 DCHECK(ValidateExperiment(experiments[i]));
[email protected]8a6ff28d2010-12-02 16:35:19433 AddInternalName(experiments[i], &known_experiments);
[email protected]28e35af2011-02-09 12:56:22434 }
[email protected]ad2a3ded2010-08-27 13:19:05435
436 std::set<std::string> enabled_experiments;
[email protected]1a47d7e2010-10-15 00:37:24437 GetEnabledFlags(prefs, &enabled_experiments);
[email protected]ad2a3ded2010-08-27 13:19:05438
439 std::set<std::string> new_enabled_experiments;
440 std::set_intersection(
441 known_experiments.begin(), known_experiments.end(),
442 enabled_experiments.begin(), enabled_experiments.end(),
443 std::inserter(new_enabled_experiments, new_enabled_experiments.begin()));
444
[email protected]1a47d7e2010-10-15 00:37:24445 SetEnabledFlags(prefs, new_enabled_experiments);
[email protected]ad2a3ded2010-08-27 13:19:05446}
447
[email protected]1a47d7e2010-10-15 00:37:24448void GetSanitizedEnabledFlags(
[email protected]ad2a3ded2010-08-27 13:19:05449 PrefService* prefs, std::set<std::string>* result) {
450 SanitizeList(prefs);
[email protected]1a47d7e2010-10-15 00:37:24451 GetEnabledFlags(prefs, result);
[email protected]ad2a3ded2010-08-27 13:19:05452}
453
[email protected]a314ee5a2010-10-26 21:23:28454// Variant of GetSanitizedEnabledFlags that also removes any flags that aren't
455// enabled on the current platform.
456void GetSanitizedEnabledFlagsForCurrentPlatform(
457 PrefService* prefs, std::set<std::string>* result) {
458 GetSanitizedEnabledFlags(prefs, result);
459
460 // Filter out any experiments that aren't enabled on the current platform. We
461 // don't remove these from prefs else syncing to a platform with a different
462 // set of experiments would be lossy.
463 std::set<std::string> platform_experiments;
464 int current_platform = GetCurrentPlatform();
465 for (size_t i = 0; i < num_experiments; ++i) {
466 if (experiments[i].supported_platforms & current_platform)
[email protected]8a6ff28d2010-12-02 16:35:19467 AddInternalName(experiments[i], &platform_experiments);
[email protected]a314ee5a2010-10-26 21:23:28468 }
469
470 std::set<std::string> new_enabled_experiments;
471 std::set_intersection(
472 platform_experiments.begin(), platform_experiments.end(),
473 result->begin(), result->end(),
474 std::inserter(new_enabled_experiments, new_enabled_experiments.begin()));
475
476 result->swap(new_enabled_experiments);
[email protected]ad2a3ded2010-08-27 13:19:05477}
478
[email protected]8a6ff28d2010-12-02 16:35:19479// Returns the Value representing the choice data in the specified experiment.
[email protected]8a6ff28d2010-12-02 16:35:19480Value* CreateChoiceData(const Experiment& experiment,
[email protected]28e35af2011-02-09 12:56:22481 const std::set<std::string>& enabled_experiments) {
[email protected]2ce9c89752011-02-25 18:24:34482 DCHECK_EQ(Experiment::MULTI_VALUE, experiment.type);
[email protected]8a6ff28d2010-12-02 16:35:19483 ListValue* result = new ListValue;
484 for (int i = 0; i < experiment.num_choices; ++i) {
485 const Experiment::Choice& choice = experiment.choices[i];
486 DictionaryValue* value = new DictionaryValue;
487 std::string name = NameForChoice(experiment, i);
488 value->SetString("description",
489 l10n_util::GetStringUTF16(choice.description_id));
490 value->SetString("internal_name", name);
[email protected]28e35af2011-02-09 12:56:22491 value->SetBoolean("selected", enabled_experiments.count(name) > 0);
[email protected]8a6ff28d2010-12-02 16:35:19492 result->Append(value);
493 }
494 return result;
495}
496
[email protected]e2ddbc92010-10-15 20:02:07497} // namespace
498
[email protected]1a47d7e2010-10-15 00:37:24499void ConvertFlagsToSwitches(PrefService* prefs, CommandLine* command_line) {
[email protected]8e8bb6d2010-12-13 08:18:55500 FlagsState::GetInstance()->ConvertFlagsToSwitches(prefs, command_line);
[email protected]ad2a3ded2010-08-27 13:19:05501}
502
[email protected]1a47d7e2010-10-15 00:37:24503ListValue* GetFlagsExperimentsData(PrefService* prefs) {
[email protected]ad2a3ded2010-08-27 13:19:05504 std::set<std::string> enabled_experiments;
[email protected]1a47d7e2010-10-15 00:37:24505 GetSanitizedEnabledFlags(prefs, &enabled_experiments);
[email protected]ad2a3ded2010-08-27 13:19:05506
507 int current_platform = GetCurrentPlatform();
508
509 ListValue* experiments_data = new ListValue();
[email protected]a314ee5a2010-10-26 21:23:28510 for (size_t i = 0; i < num_experiments; ++i) {
511 const Experiment& experiment = experiments[i];
[email protected]ad2a3ded2010-08-27 13:19:05512 if (!(experiment.supported_platforms & current_platform))
513 continue;
514
515 DictionaryValue* data = new DictionaryValue();
516 data->SetString("internal_name", experiment.internal_name);
517 data->SetString("name",
518 l10n_util::GetStringUTF16(experiment.visible_name_id));
519 data->SetString("description",
520 l10n_util::GetStringUTF16(
521 experiment.visible_description_id));
[email protected]ad2a3ded2010-08-27 13:19:05522
[email protected]28e35af2011-02-09 12:56:22523 switch (experiment.type) {
524 case Experiment::SINGLE_VALUE:
525 data->SetBoolean(
526 "enabled",
527 enabled_experiments.count(experiment.internal_name) > 0);
528 break;
529 case Experiment::MULTI_VALUE:
530 data->Set("choices", CreateChoiceData(experiment, enabled_experiments));
531 break;
532 default:
533 NOTREACHED();
[email protected]8a6ff28d2010-12-02 16:35:19534 }
535
[email protected]ad2a3ded2010-08-27 13:19:05536 experiments_data->Append(data);
537 }
538 return experiments_data;
539}
540
[email protected]ad2a3ded2010-08-27 13:19:05541bool IsRestartNeededToCommitChanges() {
[email protected]8e8bb6d2010-12-13 08:18:55542 return FlagsState::GetInstance()->IsRestartNeededToCommitChanges();
[email protected]ad2a3ded2010-08-27 13:19:05543}
544
545void SetExperimentEnabled(
[email protected]c7b7800a2010-10-07 18:51:35546 PrefService* prefs, const std::string& internal_name, bool enable) {
[email protected]8e8bb6d2010-12-13 08:18:55547 FlagsState::GetInstance()->SetExperimentEnabled(prefs, internal_name, enable);
[email protected]e2ddbc92010-10-15 20:02:07548}
549
550void RemoveFlagsSwitches(
551 std::map<std::string, CommandLine::StringType>* switch_list) {
[email protected]8e8bb6d2010-12-13 08:18:55552 FlagsState::GetInstance()->RemoveFlagsSwitches(switch_list);
[email protected]e2ddbc92010-10-15 20:02:07553}
554
[email protected]a314ee5a2010-10-26 21:23:28555int GetCurrentPlatform() {
556#if defined(OS_MACOSX)
557 return kOsMac;
558#elif defined(OS_WIN)
559 return kOsWin;
560#elif defined(OS_CHROMEOS) // Needs to be before the OS_LINUX check.
561 return kOsCrOS;
562#elif defined(OS_LINUX)
563 return kOsLinux;
564#else
565#error Unknown platform
566#endif
567}
568
[email protected]4bc5050c2010-11-18 17:55:54569void RecordUMAStatistics(const PrefService* prefs) {
570 std::set<std::string> flags;
571 GetEnabledFlags(prefs, &flags);
572 for (std::set<std::string>::iterator it = flags.begin(); it != flags.end();
573 ++it) {
574 std::string action("AboutFlags_");
575 action += *it;
576 UserMetrics::RecordComputedAction(action);
577 }
578 // Since flag metrics are recorded every startup, add a tick so that the
579 // stats can be made meaningful.
580 if (flags.size())
581 UserMetrics::RecordAction(UserMetricsAction("AboutFlags_StartupTick"));
[email protected]970b2fd2011-01-22 18:06:45582 UserMetrics::RecordAction(UserMetricsAction("StartupTick"));
[email protected]4bc5050c2010-11-18 17:55:54583}
584
[email protected]e2ddbc92010-10-15 20:02:07585//////////////////////////////////////////////////////////////////////////////
586// FlagsState implementation.
587
588namespace {
589
590void FlagsState::ConvertFlagsToSwitches(
591 PrefService* prefs, CommandLine* command_line) {
[email protected]e2ddbc92010-10-15 20:02:07592 if (command_line->HasSwitch(switches::kNoExperiments))
593 return;
594
595 std::set<std::string> enabled_experiments;
[email protected]ba8164242010-11-16 21:31:00596
[email protected]a314ee5a2010-10-26 21:23:28597 GetSanitizedEnabledFlagsForCurrentPlatform(prefs, &enabled_experiments);
[email protected]e2ddbc92010-10-15 20:02:07598
[email protected]a82744532011-02-11 16:15:53599 typedef std::map<std::string, std::pair<std::string, std::string> >
600 NameToSwitchAndValueMap;
601 NameToSwitchAndValueMap name_to_switch_map;
[email protected]8a6ff28d2010-12-02 16:35:19602 for (size_t i = 0; i < num_experiments; ++i) {
603 const Experiment& e = experiments[i];
604 if (e.type == Experiment::SINGLE_VALUE) {
[email protected]a82744532011-02-11 16:15:53605 name_to_switch_map[e.internal_name] =
606 std::pair<std::string, std::string>(e.command_line_switch,
607 e.command_line_value);
[email protected]8a6ff28d2010-12-02 16:35:19608 } else {
609 for (int j = 0; j < e.num_choices; ++j)
[email protected]a82744532011-02-11 16:15:53610 name_to_switch_map[NameForChoice(e, j)] =
611 std::pair<std::string, std::string>(
612 e.choices[j].command_line_switch,
613 e.choices[j].command_line_value);
[email protected]8a6ff28d2010-12-02 16:35:19614 }
615 }
[email protected]e2ddbc92010-10-15 20:02:07616
617 command_line->AppendSwitch(switches::kFlagSwitchesBegin);
[email protected]a82744532011-02-11 16:15:53618 flags_switches_.insert(
619 std::pair<std::string, std::string>(switches::kFlagSwitchesBegin,
620 std::string()));
[email protected]e2ddbc92010-10-15 20:02:07621 for (std::set<std::string>::iterator it = enabled_experiments.begin();
622 it != enabled_experiments.end();
623 ++it) {
624 const std::string& experiment_name = *it;
[email protected]a82744532011-02-11 16:15:53625 NameToSwitchAndValueMap::const_iterator name_to_switch_it =
[email protected]8a6ff28d2010-12-02 16:35:19626 name_to_switch_map.find(experiment_name);
627 if (name_to_switch_it == name_to_switch_map.end()) {
628 NOTREACHED();
[email protected]e2ddbc92010-10-15 20:02:07629 continue;
[email protected]8a6ff28d2010-12-02 16:35:19630 }
[email protected]e2ddbc92010-10-15 20:02:07631
[email protected]a82744532011-02-11 16:15:53632 const std::pair<std::string, std::string>&
633 switch_and_value_pair = name_to_switch_it->second;
634
635 command_line->AppendSwitchASCII(switch_and_value_pair.first,
636 switch_and_value_pair.second);
637 flags_switches_[switch_and_value_pair.first] = switch_and_value_pair.second;
[email protected]e2ddbc92010-10-15 20:02:07638 }
639 command_line->AppendSwitch(switches::kFlagSwitchesEnd);
[email protected]a82744532011-02-11 16:15:53640 flags_switches_.insert(
641 std::pair<std::string, std::string>(switches::kFlagSwitchesEnd,
642 std::string()));
[email protected]e2ddbc92010-10-15 20:02:07643}
644
645bool FlagsState::IsRestartNeededToCommitChanges() {
646 return needs_restart_;
647}
648
649void FlagsState::SetExperimentEnabled(
650 PrefService* prefs, const std::string& internal_name, bool enable) {
[email protected]ad2a3ded2010-08-27 13:19:05651 needs_restart_ = true;
652
[email protected]8a6ff28d2010-12-02 16:35:19653 size_t at_index = internal_name.find(about_flags::testing::kMultiSeparator);
654 if (at_index != std::string::npos) {
655 DCHECK(enable);
656 // We're being asked to enable a multi-choice experiment. Disable the
657 // currently selected choice.
658 DCHECK_NE(at_index, 0u);
[email protected]28e35af2011-02-09 12:56:22659 const std::string experiment_name = internal_name.substr(0, at_index);
660 SetExperimentEnabled(prefs, experiment_name, false);
[email protected]8a6ff28d2010-12-02 16:35:19661
[email protected]28e35af2011-02-09 12:56:22662 // And enable the new choice, if it is not the default first choice.
663 if (internal_name != experiment_name + "@0") {
664 std::set<std::string> enabled_experiments;
665 GetSanitizedEnabledFlags(prefs, &enabled_experiments);
666 enabled_experiments.insert(internal_name);
667 SetEnabledFlags(prefs, enabled_experiments);
668 }
[email protected]8a6ff28d2010-12-02 16:35:19669 return;
670 }
671
[email protected]ad2a3ded2010-08-27 13:19:05672 std::set<std::string> enabled_experiments;
[email protected]1a47d7e2010-10-15 00:37:24673 GetSanitizedEnabledFlags(prefs, &enabled_experiments);
[email protected]ad2a3ded2010-08-27 13:19:05674
[email protected]8a6ff28d2010-12-02 16:35:19675 const Experiment* e = NULL;
676 for (size_t i = 0; i < num_experiments; ++i) {
677 if (experiments[i].internal_name == internal_name) {
678 e = experiments + i;
679 break;
680 }
681 }
682 DCHECK(e);
683
684 if (e->type == Experiment::SINGLE_VALUE) {
685 if (enable)
686 enabled_experiments.insert(internal_name);
687 else
688 enabled_experiments.erase(internal_name);
689 } else {
690 if (enable) {
691 // Enable the first choice.
692 enabled_experiments.insert(NameForChoice(*e, 0));
693 } else {
694 // Find the currently enabled choice and disable it.
695 for (int i = 0; i < e->num_choices; ++i) {
696 std::string choice_name = NameForChoice(*e, i);
697 if (enabled_experiments.find(choice_name) !=
698 enabled_experiments.end()) {
699 enabled_experiments.erase(choice_name);
700 // Continue on just in case there's a bug and more than one
701 // experiment for this choice was enabled.
702 }
703 }
704 }
705 }
[email protected]ad2a3ded2010-08-27 13:19:05706
[email protected]1a47d7e2010-10-15 00:37:24707 SetEnabledFlags(prefs, enabled_experiments);
[email protected]ad2a3ded2010-08-27 13:19:05708}
709
[email protected]e2ddbc92010-10-15 20:02:07710void FlagsState::RemoveFlagsSwitches(
711 std::map<std::string, CommandLine::StringType>* switch_list) {
[email protected]a82744532011-02-11 16:15:53712 for (std::map<std::string, std::string>::const_iterator
713 it = flags_switches_.begin(); it != flags_switches_.end(); ++it) {
714 switch_list->erase(it->first);
[email protected]e2ddbc92010-10-15 20:02:07715 }
716}
717
718void FlagsState::reset() {
719 needs_restart_ = false;
720 flags_switches_.clear();
721}
722
723} // namespace
724
725namespace testing {
[email protected]8a6ff28d2010-12-02 16:35:19726
727// WARNING: '@' is also used in the html file. If you update this constant you
728// also need to update the html file.
729const char kMultiSeparator[] = "@";
730
[email protected]e2ddbc92010-10-15 20:02:07731void ClearState() {
[email protected]8e8bb6d2010-12-13 08:18:55732 FlagsState::GetInstance()->reset();
[email protected]e2ddbc92010-10-15 20:02:07733}
[email protected]a314ee5a2010-10-26 21:23:28734
735void SetExperiments(const Experiment* e, size_t count) {
736 if (!e) {
737 experiments = kExperiments;
738 num_experiments = arraysize(kExperiments);
739 } else {
740 experiments = e;
741 num_experiments = count;
742 }
743}
744
[email protected]8a6ff28d2010-12-02 16:35:19745const Experiment* GetExperiments(size_t* count) {
746 *count = num_experiments;
747 return experiments;
748}
749
[email protected]e2ddbc92010-10-15 20:02:07750} // namespace testing
751
[email protected]1a47d7e2010-10-15 00:37:24752} // namespace about_flags