1 # Copyright (c) 2012 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.
7 import pylib
.android_commands
8 import pylib
.device
.device_utils
11 class FlagChanger(object):
12 """Changes the flags Chrome runs with.
14 There are two different use cases for this file:
15 * Flags are permanently set by calling Set().
16 * Flags can be temporarily set for a particular set of unit tests. These
17 tests should call Restore() to revert the flags to their original state
18 once the tests have completed.
21 def __init__(self
, device
, cmdline_file
):
22 """Initializes the FlagChanger and records the original arguments.
25 device: A DeviceUtils instance.
26 cmdline_file: Path to the command line file on the device.
28 # TODO(jbudorick) Remove once telemetry switches over.
29 if isinstance(device
, pylib
.android_commands
.AndroidCommands
):
30 device
= pylib
.device
.device_utils
.DeviceUtils(device
)
32 self
._cmdline
_file
= cmdline_file
34 # Save the original flags.
35 self
._orig
_line
= self
._device
.ReadFile(self
._cmdline
_file
)
37 self
._orig
_line
= self
._orig
_line
[0].strip()
39 # Parse out the flags into a list to facilitate adding and removing flags.
40 self
._current
_flags
= self
._TokenizeFlags
(self
._orig
_line
)
43 """Returns list of current flags."""
44 return self
._current
_flags
47 """Replaces all flags on the current command line with the flags given.
50 flags: A list of flags to set, eg. ['--single-process'].
53 assert flags
[0] != 'chrome'
55 self
._current
_flags
= flags
56 self
._UpdateCommandLineFile
()
58 def AddFlags(self
, flags
):
59 """Appends flags to the command line if they aren't already there.
62 flags: A list of flags to add on, eg. ['--single-process'].
65 assert flags
[0] != 'chrome'
67 # Avoid appending flags that are already present.
69 if flag
not in self
._current
_flags
:
70 self
._current
_flags
.append(flag
)
71 self
._UpdateCommandLineFile
()
73 def RemoveFlags(self
, flags
):
74 """Removes flags from the command line, if they exist.
77 flags: A list of flags to remove, eg. ['--single-process']. Note that we
78 expect a complete match when removing flags; if you want to remove
79 a switch with a value, you must use the exact string used to add
80 it in the first place.
83 assert flags
[0] != 'chrome'
86 if flag
in self
._current
_flags
:
87 self
._current
_flags
.remove(flag
)
88 self
._UpdateCommandLineFile
()
91 """Restores the flags to their original state."""
92 self
._current
_flags
= self
._TokenizeFlags
(self
._orig
_line
)
93 self
._UpdateCommandLineFile
()
95 def _UpdateCommandLineFile(self
):
96 """Writes out the command line to the file, or removes it if empty."""
97 logging
.info('Current flags: %s', self
._current
_flags
)
98 # Root is not required to write to /data/local/tmp/.
99 use_root
= '/data/local/tmp/' not in self
._cmdline
_file
100 if self
._current
_flags
:
101 # The first command line argument doesn't matter as we are not actually
102 # launching the chrome executable using this command line.
103 cmd_line
= ' '.join(['_'] + self
._current
_flags
)
104 self
._device
.WriteFile(
105 self
._cmdline
_file
, cmd_line
, as_root
=use_root
)
106 file_contents
= self
._device
.ReadFile(
107 self
._cmdline
_file
, as_root
=use_root
)
108 assert len(file_contents
) == 1 and file_contents
[0] == cmd_line
, (
109 'Failed to set the command line file at %s' % self
._cmdline
_file
)
111 self
._device
.RunShellCommand('rm ' + self
._cmdline
_file
,
113 assert not self
._device
.FileExists(self
._cmdline
_file
), (
114 'Failed to remove the command line file at %s' % self
._cmdline
_file
)
117 def _TokenizeFlags(line
):
118 """Changes the string containing the command line into a list of flags.
120 Follows similar logic to CommandLine.java::tokenizeQuotedArguments:
121 * Flags are split using whitespace, unless the whitespace is within a
122 pair of quotation marks.
123 * Unlike the Java version, we keep the quotation marks around switch
124 values since we need them to re-create the file when new flags are
128 line: A string containing the entire command line. The first token is
129 assumed to be the program name.
136 within_quotations
= False
138 # Move through the string character by character and build up each flag
140 for c
in line
.strip():
142 if len(current_flag
) > 0 and current_flag
[-1] == '\\':
143 # Last char was a backslash; pop it, and treat this " as a literal.
144 current_flag
= current_flag
[0:-1] + '"'
146 within_quotations
= not within_quotations
148 elif not within_quotations
and (c
is ' ' or c
is '\t'):
149 if current_flag
is not "":
150 tokenized_flags
.append(current_flag
)
155 # Tack on the last flag.
157 if within_quotations
:
158 logging
.warn('Unterminated quoted argument: ' + line
)
160 tokenized_flags
.append(current_flag
)
162 # Return everything but the program name.
163 return tokenized_flags
[1:]