1# Copyright 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. 4 5import copy 6import logging 7import optparse 8import os 9import shlex 10import socket 11import sys 12 13from py_utils import cloud_storage # pylint: disable=import-error 14 15from telemetry.core import platform 16from telemetry.core import util 17from telemetry.internal.browser import browser_finder 18from telemetry.internal.browser import browser_finder_exceptions 19from telemetry.internal.browser import profile_types 20from telemetry.internal.platform import device_finder 21from telemetry.internal.platform import remote_platform_options 22from telemetry.internal.platform.profiler import profiler_finder 23from telemetry.internal.util import binary_manager 24from telemetry.util import wpr_modes 25 26 27class BrowserFinderOptions(optparse.Values): 28 """Options to be used for discovering a browser.""" 29 30 def __init__(self, browser_type=None): 31 optparse.Values.__init__(self) 32 33 self.browser_type = browser_type 34 self.browser_executable = None 35 self.chrome_root = None # Path to src/ 36 self.chromium_output_dir = None # E.g.: out/Debug 37 self.device = None 38 self.cros_ssh_identity = None 39 40 self.cros_remote = None 41 42 self.profiler = None 43 self.verbosity = 0 44 45 self.browser_options = BrowserOptions() 46 self.output_file = None 47 48 self.remote_platform_options = None 49 50 self.no_performance_mode = False 51 52 def __repr__(self): 53 return str(sorted(self.__dict__.items())) 54 55 def Copy(self): 56 return copy.deepcopy(self) 57 58 def CreateParser(self, *args, **kwargs): 59 parser = optparse.OptionParser(*args, **kwargs) 60 61 # Selection group 62 group = optparse.OptionGroup(parser, 'Which browser to use') 63 group.add_option('--browser', 64 dest='browser_type', 65 default=None, 66 help='Browser type to run, ' 67 'in order of priority. Supported values: list,%s' % 68 ','.join(browser_finder.FindAllBrowserTypes(self))) 69 group.add_option('--browser-executable', 70 dest='browser_executable', 71 help='The exact browser to run.') 72 group.add_option('--chrome-root', 73 dest='chrome_root', 74 help='Where to look for chrome builds. ' 75 'Defaults to searching parent dirs by default.') 76 group.add_option('--chromium-output-directory', 77 dest='chromium_output_dir', 78 help='Where to look for build artifacts. ' 79 'Can also be specified by setting environment variable ' 80 'CHROMIUM_OUTPUT_DIR.') 81 group.add_option( 82 '--remote', 83 dest='cros_remote', 84 help='The hostname of a remote ChromeOS device to use.') 85 group.add_option( 86 '--remote-ssh-port', 87 type=int, 88 default=socket.getservbyname('ssh'), 89 dest='cros_remote_ssh_port', 90 help='The SSH port of the remote ChromeOS device (requires --remote).') 91 identity = None 92 testing_rsa = os.path.join( 93 util.GetTelemetryThirdPartyDir(), 'chromite', 'ssh_keys', 'testing_rsa') 94 if os.path.exists(testing_rsa): 95 identity = testing_rsa 96 group.add_option('--identity', 97 dest='cros_ssh_identity', 98 default=identity, 99 help='The identity file to use when ssh\'ing into the ChromeOS device') 100 parser.add_option_group(group) 101 102 # Debugging options 103 group = optparse.OptionGroup(parser, 'When things go wrong') 104 profiler_choices = profiler_finder.GetAllAvailableProfilers() 105 group.add_option( 106 '--profiler', default=None, type='choice', 107 choices=profiler_choices, 108 help='Record profiling data using this tool. Supported values: %s. ' 109 '(Notice: this flag cannot be used for Timeline Based Measurement ' 110 'benchmarks.)' % ', '.join(profiler_choices)) 111 group.add_option( 112 '-v', '--verbose', action='count', dest='verbosity', 113 help='Increase verbosity level (repeat as needed)') 114 group.add_option('--print-bootstrap-deps', 115 action='store_true', 116 help='Output bootstrap deps list.') 117 parser.add_option_group(group) 118 119 # Platform options 120 group = optparse.OptionGroup(parser, 'Platform options') 121 group.add_option('--no-performance-mode', action='store_true', 122 help='Some platforms run on "full performance mode" where the ' 123 'test is executed at maximum CPU speed in order to minimize noise ' 124 '(specially important for dashboards / continuous builds). ' 125 'This option prevents Telemetry from tweaking such platform settings.') 126 parser.add_option_group(group) 127 128 # Remote platform options 129 group = optparse.OptionGroup(parser, 'Remote platform options') 130 group.add_option('--android-blacklist-file', 131 help='Device blacklist JSON file.') 132 group.add_option('--device', 133 help='The device ID to use. ' 134 'If not specified, only 0 or 1 connected devices are supported. ' 135 'If specified as "android", all available Android devices are ' 136 'used.') 137 parser.add_option_group(group) 138 139 # Browser options. 140 self.browser_options.AddCommandLineArgs(parser) 141 142 real_parse = parser.parse_args 143 def ParseArgs(args=None): 144 defaults = parser.get_default_values() 145 for k, v in defaults.__dict__.items(): 146 if k in self.__dict__ and self.__dict__[k] != None: 147 continue 148 self.__dict__[k] = v 149 ret = real_parse(args, self) # pylint: disable=E1121 150 151 if self.verbosity >= 2: 152 logging.getLogger().setLevel(logging.DEBUG) 153 elif self.verbosity: 154 logging.getLogger().setLevel(logging.INFO) 155 else: 156 logging.getLogger().setLevel(logging.WARNING) 157 158 if self.chromium_output_dir: 159 os.environ['CHROMIUM_OUTPUT_DIR'] = self.chromium_output_dir 160 161 # Parse remote platform options. 162 self.BuildRemotePlatformOptions() 163 164 if self.remote_platform_options.device == 'list': 165 if binary_manager.NeedsInit(): 166 binary_manager.InitDependencyManager([]) 167 devices = device_finder.GetDevicesMatchingOptions(self) 168 print 'Available devices:' 169 for device in devices: 170 print ' ', device.name 171 sys.exit(0) 172 173 if self.browser_executable and not self.browser_type: 174 self.browser_type = 'exact' 175 if self.browser_type == 'list': 176 if binary_manager.NeedsInit(): 177 binary_manager.InitDependencyManager([]) 178 devices = device_finder.GetDevicesMatchingOptions(self) 179 if not devices: 180 sys.exit(0) 181 browser_types = {} 182 for device in devices: 183 try: 184 possible_browsers = browser_finder.GetAllAvailableBrowsers(self, 185 device) 186 browser_types[device.name] = sorted( 187 [browser.browser_type for browser in possible_browsers]) 188 except browser_finder_exceptions.BrowserFinderException as ex: 189 print >> sys.stderr, 'ERROR: ', ex 190 sys.exit(1) 191 print 'Available browsers:' 192 if len(browser_types) == 0: 193 print ' No devices were found.' 194 for device_name in sorted(browser_types.keys()): 195 print ' ', device_name 196 for browser_type in browser_types[device_name]: 197 print ' ', browser_type 198 sys.exit(0) 199 200 # Parse browser options. 201 self.browser_options.UpdateFromParseResults(self) 202 203 return ret 204 parser.parse_args = ParseArgs 205 return parser 206 207 # TODO(eakuefner): Factor this out into OptionBuilder pattern 208 def BuildRemotePlatformOptions(self): 209 if self.device or self.android_blacklist_file: 210 self.remote_platform_options = ( 211 remote_platform_options.AndroidPlatformOptions( 212 self.device, self.android_blacklist_file)) 213 214 # We delete these options because they should live solely in the 215 # AndroidPlatformOptions instance belonging to this class. 216 if self.device: 217 del self.device 218 if self.android_blacklist_file: 219 del self.android_blacklist_file 220 else: 221 self.remote_platform_options = ( 222 remote_platform_options.AndroidPlatformOptions()) 223 224 def AppendExtraBrowserArgs(self, args): 225 self.browser_options.AppendExtraBrowserArgs(args) 226 227 def MergeDefaultValues(self, defaults): 228 for k, v in defaults.__dict__.items(): 229 self.ensure_value(k, v) 230 231class BrowserOptions(object): 232 """Options to be used for launching a browser.""" 233 234 # Levels of browser logging. 235 NO_LOGGING = 'none' 236 NON_VERBOSE_LOGGING = 'non-verbose' 237 VERBOSE_LOGGING = 'verbose' 238 239 _LOGGING_LEVELS = (NO_LOGGING, NON_VERBOSE_LOGGING, VERBOSE_LOGGING) 240 _DEFAULT_LOGGING_LEVEL = NO_LOGGING 241 242 def __init__(self): 243 self.browser_type = None 244 self.show_stdout = False 245 246 self.extensions_to_load = [] 247 248 # If set, copy the generated profile to this path on exit. 249 self.output_profile_path = None 250 251 # When set to True, the browser will use the default profile. Telemetry 252 # will not provide an alternate profile directory. 253 self.dont_override_profile = False 254 self.profile_dir = None 255 self.profile_type = None 256 self._extra_browser_args = set() 257 self.extra_wpr_args = [] 258 self.wpr_mode = wpr_modes.WPR_OFF 259 self.full_performance_mode = True 260 261 # The amount of time Telemetry should wait for the browser to start. 262 # This property is not exposed as a command line option. 263 self._browser_startup_timeout = 60 264 265 self.disable_background_networking = True 266 self.browser_user_agent_type = None 267 268 self.clear_sytem_cache_for_browser_and_profile_on_start = False 269 self.startup_url = 'about:blank' 270 271 # Background pages of built-in component extensions can interfere with 272 # performance measurements. 273 self.disable_component_extensions_with_background_pages = True 274 # Disable default apps. 275 self.disable_default_apps = True 276 277 self.logging_verbosity = self._DEFAULT_LOGGING_LEVEL 278 279 # The cloud storage bucket & path for uploading logs data produced by the 280 # browser to. 281 # If logs_cloud_remote_path is None, a random remote path is generated every 282 # time the logs data is uploaded. 283 self.logs_cloud_bucket = cloud_storage.TELEMETRY_OUTPUT 284 self.logs_cloud_remote_path = None 285 286 # TODO(danduong): Find a way to store target_os here instead of 287 # finder_options. 288 self._finder_options = None 289 290 # Whether to take screen shot for failed page & put them in telemetry's 291 # profiling results. 292 self.take_screenshot_for_failed_page = False 293 294 def __repr__(self): 295 # This works around the infinite loop caused by the introduction of a 296 # circular reference with _finder_options. 297 obj = self.__dict__.copy() 298 del obj['_finder_options'] 299 return str(sorted(obj.items())) 300 301 def IsCrosBrowserOptions(self): 302 return False 303 304 @classmethod 305 def AddCommandLineArgs(cls, parser): 306 307 ############################################################################ 308 # Please do not add any more options here without first discussing with # 309 # a telemetry owner. This is not the right place for platform-specific # 310 # options. # 311 ############################################################################ 312 313 group = optparse.OptionGroup(parser, 'Browser options') 314 profile_choices = profile_types.GetProfileTypes() 315 group.add_option('--profile-type', 316 dest='profile_type', 317 type='choice', 318 default='clean', 319 choices=profile_choices, 320 help=('The user profile to use. A clean profile is used by default. ' 321 'Supported values: ' + ', '.join(profile_choices))) 322 group.add_option('--profile-dir', 323 dest='profile_dir', 324 help='Profile directory to launch the browser with. ' 325 'A clean profile is used by default') 326 group.add_option('--extra-browser-args', 327 dest='extra_browser_args_as_string', 328 help='Additional arguments to pass to the browser when it starts') 329 group.add_option('--extra-wpr-args', 330 dest='extra_wpr_args_as_string', 331 help=('Additional arguments to pass to Web Page Replay. ' 332 'See third_party/web-page-replay/replay.py for usage.')) 333 group.add_option('--show-stdout', 334 action='store_true', 335 help='When possible, will display the stdout of the process') 336 337 group.add_option('--browser-logging-verbosity', 338 dest='logging_verbosity', 339 type='choice', 340 choices=cls._LOGGING_LEVELS, 341 help=('Browser logging verbosity. The log file is saved in temp ' 342 "directory. Note that logging affects the browser's " 343 'performance. Supported values: %s. Defaults to %s.' % ( 344 ', '.join(cls._LOGGING_LEVELS), cls._DEFAULT_LOGGING_LEVEL))) 345 parser.add_option_group(group) 346 347 group = optparse.OptionGroup(parser, 'Compatibility options') 348 group.add_option('--gtest_output', 349 help='Ignored argument for compatibility with runtest.py harness') 350 parser.add_option_group(group) 351 352 def UpdateFromParseResults(self, finder_options): 353 """Copies our options from finder_options""" 354 browser_options_list = [ 355 'extra_browser_args_as_string', 356 'extra_wpr_args_as_string', 357 'profile_dir', 358 'profile_type', 359 'show_stdout', 360 ] 361 for o in browser_options_list: 362 a = getattr(finder_options, o, None) 363 if a is not None: 364 setattr(self, o, a) 365 delattr(finder_options, o) 366 367 self.browser_type = finder_options.browser_type 368 self._finder_options = finder_options 369 370 if hasattr(self, 'extra_browser_args_as_string'): 371 tmp = shlex.split( 372 self.extra_browser_args_as_string) 373 self.AppendExtraBrowserArgs(tmp) 374 delattr(self, 'extra_browser_args_as_string') 375 if hasattr(self, 'extra_wpr_args_as_string'): 376 tmp = shlex.split( 377 self.extra_wpr_args_as_string) 378 self.extra_wpr_args.extend(tmp) 379 delattr(self, 'extra_wpr_args_as_string') 380 if self.profile_type == 'default': 381 self.dont_override_profile = True 382 383 if self.profile_dir and self.profile_type != 'clean': 384 logging.critical( 385 "It's illegal to specify both --profile-type and --profile-dir.\n" 386 "For more information see: http://goo.gl/ngdGD5") 387 sys.exit(1) 388 389 if self.profile_dir and not os.path.isdir(self.profile_dir): 390 logging.critical( 391 "Directory specified by --profile-dir (%s) doesn't exist " 392 "or isn't a directory.\n" 393 "For more information see: http://goo.gl/ngdGD5" % self.profile_dir) 394 sys.exit(1) 395 396 if not self.profile_dir: 397 self.profile_dir = profile_types.GetProfileDir(self.profile_type) 398 399 if getattr(finder_options, 'logging_verbosity'): 400 self.logging_verbosity = finder_options.logging_verbosity 401 delattr(finder_options, 'logging_verbosity') 402 403 # This deferred import is necessary because browser_options is imported in 404 # telemetry/telemetry/__init__.py. 405 finder_options.browser_options = CreateChromeBrowserOptions(self) 406 407 @property 408 def finder_options(self): 409 return self._finder_options 410 411 @property 412 def extra_browser_args(self): 413 return self._extra_browser_args 414 415 @property 416 def browser_startup_timeout(self): 417 return self._browser_startup_timeout 418 419 @browser_startup_timeout.setter 420 def browser_startup_timeout(self, value): 421 self._browser_startup_timeout = value 422 423 def AppendExtraBrowserArgs(self, args): 424 if isinstance(args, list): 425 self._extra_browser_args.update(args) 426 else: 427 self._extra_browser_args.add(args) 428 429 430def CreateChromeBrowserOptions(br_options): 431 browser_type = br_options.browser_type 432 433 if (platform.GetHostPlatform().GetOSName() == 'chromeos' or 434 (browser_type and browser_type.startswith('cros'))): 435 return CrosBrowserOptions(br_options) 436 437 return br_options 438 439 440class ChromeBrowserOptions(BrowserOptions): 441 """Chrome-specific browser options.""" 442 443 def __init__(self, br_options): 444 super(ChromeBrowserOptions, self).__init__() 445 # Copy to self. 446 self.__dict__.update(br_options.__dict__) 447 448 449class CrosBrowserOptions(ChromeBrowserOptions): 450 """ChromeOS-specific browser options.""" 451 452 def __init__(self, br_options): 453 super(CrosBrowserOptions, self).__init__(br_options) 454 # Create a browser with oobe property. 455 self.create_browser_with_oobe = False 456 # Clear enterprise policy before logging in. 457 self.clear_enterprise_policy = True 458 # Disable GAIA/enterprise services. 459 self.disable_gaia_services = True 460 461 self.auto_login = True 462 self.gaia_login = False 463 self.username = 'test@test.test' 464 self.password = '' 465 self.gaia_id = '12345' 466 # For non-accelerated QEMU VMs. 467 self.browser_startup_timeout = 240 468 469 def IsCrosBrowserOptions(self): 470 return True 471