1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 168 169 170 171 172 173 174 175 176 177 178 179 180 181 182 183 184 185 186 187 188 189 190 191 192 193 194 195 196 197 198 199 200 201 202 203 204 205 206 207 208 209 210 211 212 213 214 215 216 217 218 219 220 221 222 223 224 225 226 227 228 229 230 231 232 233 234 235 236 237 238 239 240 241 242 243 244 245 246 247 248 249 250 251 252 253 254 255 256 257 258 259 260 261 262 263 264 265 266 267 268 269 270 271 272 273 274 275 276 277 278 279 280 281 282 283 284 285 286 287 288 289 290 291 292 293 294 295 296 297 298 299 300 301 302 303 304 305 306 307 308 309 310 311 312 313 314 315 316 317 318 319 320 321 322 323 324 325 326 327 328 329 330 331 332 333 334 335 336 337 338 339 340 341 342 343 344 345 346 347 348 349 350 351 352 353 354 355 356 357 358 359 360 361 362 363 364 365 366 367 368 369 370 371 372 373 374 375 376 377 378 379 380 381 382 383 384 385 386 387 388 389 390 391 392 393 394 395 396 397 398 399 400 401 402 403 404 405 406 407 408 409 410 411 412 413 414 415 416 417 418 419 420 421 422 423 424 425 426 427 428 429 430 431 432 433 434 435 436 437 438 439 440 441 442 443 444 445 446 447 448 449 450 451 452 453 454 455 456 457 458 459 460 461 462 463 464 465 466 467 468 469 470 471 472 473 474 475 476 477 478 479 480 481 482 483 484 485 486 487 488 489 490 491 492 493 494 495 496 497 498 499 500 501 502 503 504 505 506 507 508 509 510 511 512 513 514 515 516 517 518 519 520 521 522 523 524 525 526 527 528 529 530 531 532 533 534 535 536 537 538 539 540 541 542 543 544 545 546 547 548 549 550 551 552 553 554 555 556 557 558 559 560 561 562 563 564 565 566 567 568 569 570 571 572 573 574 575 576 577 578 579 580 581 582 583 584 585 586 587 588 589 590 591 592 593 594 595 596 597 598 599 600 601 602 603 604 605 606 607 608 609 610 611 612 613 614 615 616 617 618 619 620 621 622 623 624 625 626 627 628 629 630 631 632 633 634 635 636 637 638 639 640 641 642 643 644 645 646 647 648 649 650 651 652 653 654 655 656 657 658 659 660 661 662 663 664 665 666 667 668 669 670 671 672 673 674 675 676 677 678 679 680 681 682 683 684 685 686 687 688 689 690 691 692 693 694 695 696 697 698 699 700 701 702 703 704 705 706 707 708 709 710 711 712 713 714 715 716 717 718 719 720 721 722 723 724 725 726 727 728 729 730 731 732 733 734 735 736 737
|
# This Source Code Form is subject to the terms of the Mozilla Public
# License, v. 2.0. If a copy of the MPL was not distributed with this
# file, You can obtain one at http://mozilla.org/MPL/2.0/
# ALL CHANGES TO THIS FILE MUST HAVE REVIEW FROM A MARIONETTE PEER!
#
# Please refer to INSTRUCTIONS TO ADD A NEW PREFERENCE in
# remote/shared/RecommendedPreferences.sys.mjs
#
# The Marionette Python client is used out-of-tree with various builds of
# Firefox. Removing a preference from this file will cause regressions,
# so please be careful and get review from a Testing :: Marionette peer
# before you make any changes to this file.
import codecs
import io
import json
import os
import sys
import tempfile
import time
import traceback
from copy import deepcopy
import mozversion
from mozprofile import Profile
from mozrunner import FennecEmulatorRunner, Runner
from . import errors
if sys.platform.startswith("darwin"):
# Marionette's own processhandler is only used on MacOS for now
from .processhandler import UNKNOWN_RETURNCODE, ProcessHandler
class GeckoInstance:
required_prefs = {
# Make sure Shield doesn't hit the network.
"app.normandy.api_url": "",
# Increase the APZ content response timeout in tests to 1 minute.
# This is to accommodate the fact that test environments tends to be slower
# than production environments (with the b2g emulator being the slowest of them
# all), resulting in the production timeout value sometimes being exceeded
# and causing false-positive test failures. See bug 1176798, bug 1177018,
# bug 1210465.
"apz.content_response_timeout": 60000,
# Disable extension discovery
"browser.discovery.enabled": False,
# Make sure error page is not shown for blank pages with 4xx or 5xx response code
"browser.http.blank_page_with_error_response.enabled": True,
# Disable CFR features for automated tests.
"browser.newtabpage.activity-stream.asrouter.userprefs.cfr.features": False,
# Don't pull weather data from the network
"browser.newtabpage.activity-stream.discoverystream.region-weather-config": "",
# Don't pull wallpaper content from the network
"browser.newtabpage.activity-stream.newtabWallpapers.enabled": False,
# Remove once Firefox 140 is no longer supported (see bug 1902921)
"browser.newtabpage.activity-stream.newtabWallpapers.v2.enabled": False,
# Don't pull sponsored Top Sites content from the network
"browser.newtabpage.activity-stream.showSponsoredTopSites": False,
# Disable geolocation ping (#1)
"browser.region.network.url": "",
# Don't pull Top Sites content from the network
"browser.topsites.contile.enabled": False,
# Disable translations
"browser.translations.enable": False,
# Disable UI tour
"browser.uitour.enabled": False,
# Disable captive portal
"captivedetect.canonicalURL": "",
# Defensively disable data reporting systems
"datareporting.healthreport.documentServerURI": "http://%(server)s/dummy/healthreport/",
"datareporting.healthreport.logging.consoleEnabled": False,
"datareporting.healthreport.service.enabled": False,
"datareporting.healthreport.service.firstRun": False,
"datareporting.healthreport.uploadEnabled": False,
"datareporting.usage.uploadEnabled": False,
"telemetry.fog.test.localhost_port": -1,
# Do not show datareporting policy notifications which can interfere with tests
"datareporting.policy.dataSubmissionEnabled": False,
"datareporting.policy.dataSubmissionPolicyBypassNotification": True,
# Disable popup-blocker
"dom.disable_open_during_load": False,
# Enabling the support for File object creation in the content process.
"dom.file.createInChild": True,
# Disable delayed user input event handling
"dom.input_events.security.minNumTicks": 0,
# Disable delayed user input event handling
"dom.input_events.security.minTimeElapsedInMS": 0,
# Disable the ProcessHangMonitor
"dom.ipc.reportProcessHangs": False,
# No slow script dialogs
"dom.max_chrome_script_run_time": 0,
"dom.max_script_run_time": 0,
# Disable navigation change rate limitation
"dom.navigation.navigationRateLimit.count": 0,
# DOM Push
"dom.push.connection.enabled": False,
# Screen Orientation API
"dom.screenorientation.allow-lock": True,
# Disable dialog abuse if alerts are triggered too quickly
"dom.successive_dialog_time_limit": 0,
# Only load extensions from the application and user profile
# AddonManager.SCOPE_PROFILE + AddonManager.SCOPE_APPLICATION
"extensions.autoDisableScopes": 0,
"extensions.enabledScopes": 5,
# Disable form autofill for extensions and credit cards
"extensions.formautofill.addresses.enabled": False,
"extensions.formautofill.creditCards.enabled": False,
# Disable metadata caching for installed add-ons by default
"extensions.getAddons.cache.enabled": False,
# Disable intalling any distribution add-ons
"extensions.installDistroAddons": False,
# Turn off extension updates so they don't bother tests
"extensions.update.enabled": False,
"extensions.update.notifyUser": False,
# Redirect various extension update URLs
"extensions.blocklist.detailsURL": (
"http://%(server)s/extensions-dummy/blocklistDetailsURL"
),
"extensions.blocklist.itemURL": "http://%(server)s/extensions-dummy/blocklistItemURL",
"extensions.hotfix.url": "http://%(server)s/extensions-dummy/hotfixURL",
"extensions.systemAddon.update.enabled": False,
"extensions.update.background.url": (
"http://%(server)s/extensions-dummy/updateBackgroundURL"
),
"extensions.update.url": "http://%(server)s/extensions-dummy/updateURL",
# Make sure opening about:addons won"t hit the network
"extensions.getAddons.discovery.api_url": "data:, ",
"extensions.getAddons.get.url": "http://%(server)s/extensions-dummy/repositoryGetURL",
"extensions.getAddons.search.browseURL": (
"http://%(server)s/extensions-dummy/repositoryBrowseURL"
),
# Allow the application to have focus even it runs in the background
"focusmanager.testmode": True,
# Disable useragent updates
"general.useragent.updates.enabled": False,
# Disable geolocation ping (#2)
"geo.provider.network.url": "",
# Always use network provider for geolocation tests
# so we bypass the OSX dialog raised by the corelocation provider
"geo.provider.testing": True,
# Do not scan Wifi
"geo.wifi.scan": False,
# Ensure webrender is on, no need for environment variables
"gfx.webrender.all": True,
# Disable idle-daily notifications to avoid expensive operations
# that may cause unexpected test timeouts.
"idle.lastDailyNotification": -1,
# Disable Firefox accounts ping
"identity.fxaccounts.auth.uri": "https://{server}/dummy/fxa",
# Disable download and usage of OpenH264, and Widevine plugins
"media.gmp-manager.updateEnabled": False,
# Disable the GFX sanity window
"media.sanity-test.disabled": True,
"media.volume_scale": "0.01",
# Disable connectivity service pings
"network.connectivity-service.enabled": False,
# Do not prompt for temporary redirects
"network.http.prompt-temp-redirect": False,
# Do not automatically switch between offline and online
"network.manage-offline-status": False,
# Make sure SNTP requests don't hit the network
"network.sntp.pools": "%(server)s",
# Disabled for causing marionette crashes on OSX. See bug 1882856
"network.dns.native_https_query": False,
# Privacy and Tracking Protection
"privacy.trackingprotection.enabled": False,
"privacy.trackingprotection.pbmode.enabled": False,
# Disable recommended automation prefs in CI
"remote.prefs.recommended": False,
# Don't do network connections for mitm priming
"security.certerrors.mitm.priming.enabled": False,
# Tests don't wait for the notification button security delay
"security.notification_enable_delay": 0,
# Do not download intermediate certificates
"security.remote_settings.intermediates.enabled": False,
# Disable logging for remote settings
"services.settings.loglevel": "off",
# Ensure blocklist updates don't hit the network
"services.settings.server": "data:,#remote-settings-dummy/v1",
# Disable password capture, so that tests that include forms aren"t
# influenced by the presence of the persistent doorhanger notification
"signon.rememberSignons": False,
# Disable alerts for credential issues
"signon.management.page.breach-alerts.enabled": False,
"signon.management.page.vulnerable-passwords.enabled": False,
# Prevent starting into safe mode after application crashes
# Do not show TOU new user modal which can interfere with tests
"termsofuse.bypassNotification": True,
"toolkit.startup.max_resumed_crashes": -1,
# Disable most telemetry pings
"toolkit.telemetry.server": "https://%(server)s/telemetry-dummy/",
# Disable window occlusion on Windows, see Bug 1802473.
"widget.windows.window_occlusion_tracking.enabled": False,
}
def __init__(
self,
host=None,
port=None,
bin=None,
profile=None,
addons=None,
app_args=None,
symbols_path=None,
gecko_log=None,
prefs=None,
workspace=None,
verbose=0,
headless=False,
):
self.runner_class = Runner
self.app_args = app_args or []
self.runner = None
self.symbols_path = symbols_path
self.binary = bin
self.marionette_host = host
self.marionette_port = port
self.addons = addons
self.prefs = prefs
self.required_prefs = deepcopy(self.required_prefs)
if prefs:
self.required_prefs.update(prefs)
self._gecko_log_option = gecko_log
self._gecko_log = None
self.verbose = verbose
self.headless = headless
# keep track of errors to decide whether instance is unresponsive
self.unresponsive_count = 0
# Alternative to default temporary directory
self.workspace = workspace
# Don't use the 'profile' property here, because sub-classes could add
# further preferences and data, which would not be included in the new
# profile
self._profile = profile
@property
def gecko_log(self):
if self._gecko_log:
return self._gecko_log
path = self._gecko_log_option
if path != "-":
if path is None:
path = "gecko.log"
elif os.path.isdir(path):
fname = f"gecko-{time.time()}.log"
path = os.path.join(path, fname)
path = os.path.realpath(path)
if os.access(path, os.F_OK):
os.remove(path)
self._gecko_log = path
return self._gecko_log
@property
def profile(self):
return self._profile
@profile.setter
def profile(self, value):
self._update_profile(value)
def _update_profile(self, profile=None, profile_name=None):
"""Check if the profile has to be created, or replaced.
:param profile: A Profile instance to be used.
:param name: Profile name to be used in the path.
"""
if self.runner and self.runner.is_running():
raise errors.MarionetteException(
"The current profile can only be updated "
"when the instance is not running"
)
if isinstance(profile, Profile):
# Only replace the profile if it is not the current one
if hasattr(self, "_profile") and profile is self._profile:
return
else:
profile_args = self.profile_args
profile_path = profile
# If a path to a profile is given then clone it
if isinstance(profile_path, str):
profile_args["path_from"] = profile_path
profile_args["path_to"] = tempfile.mkdtemp(
suffix=f".{profile_name or os.path.basename(profile_path)}",
dir=self.workspace,
)
# The target must not exist yet
os.rmdir(profile_args["path_to"])
profile = Profile.clone(**profile_args)
# Otherwise create a new profile
else:
profile_args["profile"] = tempfile.mkdtemp(
suffix=".{}".format(profile_name or "mozrunner"),
dir=self.workspace,
)
profile = Profile(**profile_args)
profile.create_new = True
if isinstance(self.profile, Profile):
self.profile.cleanup()
self._profile = profile
def switch_profile(self, profile_name=None, clone_from=None):
"""Switch the profile by using the given name, and optionally clone it.
Compared to :attr:`profile` this method allows to switch the profile
by giving control over the profile name as used for the new profile. It
also always creates a new blank profile, or as clone of an existent one.
:param profile_name: Optional, name of the profile, which will be used
as part of the profile path (folder name containing the profile).
:clone_from: Optional, if specified the new profile will be cloned
based on the given profile. This argument can be an instance of
``mozprofile.Profile``, or the path of the profile.
"""
if isinstance(clone_from, Profile):
clone_from = clone_from.profile
self._update_profile(clone_from, profile_name=profile_name)
@property
def profile_args(self):
args = {"preferences": deepcopy(self.required_prefs)}
args["preferences"]["marionette.port"] = self.marionette_port
args["preferences"]["marionette.defaultPrefs.port"] = self.marionette_port
if self.prefs:
args["preferences"].update(self.prefs)
if self.verbose:
level = "Trace" if self.verbose >= 2 else "Debug"
args["preferences"]["remote.log.level"] = level
if "-jsdebugger" in self.app_args:
args["preferences"].update(
{
"devtools.browsertoolbox.panel": "jsdebugger",
"devtools.chrome.enabled": True,
"devtools.debugger.prompt-connection": False,
"devtools.debugger.remote-enabled": True,
"devtools.testing": True,
}
)
if self.addons:
args["addons"] = self.addons
return args
@classmethod
def create(cls, app=None, *args, **kwargs):
try:
if not app and kwargs["bin"] is not None:
app_id = mozversion.get_version(binary=kwargs["bin"])["application_id"]
app = app_ids[app_id]
instance_class = apps[app]
except (OSError, KeyError):
exc, val, tb = sys.exc_info()
msg = f'Application "{app}" unknown (should be one of {list(apps.keys())})'
raise NotImplementedError(msg).with_traceback(tb)
return instance_class(*args, **kwargs)
def start(self):
self._update_profile(self.profile)
self.runner = self.runner_class(**self._get_runner_args())
self.runner.start()
def _get_runner_args(self):
process_args = {
"processOutputLine": [NullOutput()],
"universal_newlines": True,
}
if self.gecko_log == "-":
if getattr(sys.stdout, "encoding") == "utf-8":
process_args["stream"] = sys.stdout
elif hasattr(sys.stdout, "buffer"):
process_args["stream"] = codecs.getwriter("utf-8")(sys.stdout.buffer)
elif isinstance(sys.stdout, io.TextIOBase):
# If sys.stdout expects unicode strings, we can't wrap it because the
# wrapper will write byte strings. This can happen when e.g. tests
# replace sys.stdout with a io.StringIO().
process_args["stream"] = sys.stdout
else:
process_args["stream"] = codecs.getwriter("utf-8")(sys.stdout)
else:
process_args["logfile"] = self.gecko_log
env = os.environ.copy()
# Store all required preferences for tests which need to create clean profiles.
required_prefs_keys = list(self.required_prefs.keys())
env["MOZ_MARIONETTE_REQUIRED_PREFS"] = json.dumps(required_prefs_keys)
if self.headless:
env["MOZ_HEADLESS"] = "1"
env["DISPLAY"] = "77" # Set a fake display.
# environment variables needed for crashreporting
# https://developer.mozilla.org/docs/Environment_variables_affecting_crash_reporting
env.update(
{
"MOZ_CRASHREPORTER": "1",
"MOZ_CRASHREPORTER_NO_REPORT": "1",
"MOZ_CRASHREPORTER_SHUTDOWN": "1",
}
)
extra_args = ["-marionette", "-remote-allow-system-access"]
args = {
"binary": self.binary,
"profile": self.profile,
"cmdargs": extra_args + self.app_args,
"env": env,
"symbols_path": self.symbols_path,
"process_args": process_args,
}
if sys.platform.startswith("darwin"):
# Bug 1887666: The custom process handler class for Marionette is
# only supported on MacOS at the moment.
args["process_class"] = ProcessHandler
return args
def close(self, clean=False):
"""
Close the managed Gecko process.
Depending on self.runner_class, setting `clean` to True may also kill
the emulator process in which this instance is running.
:param clean: If True, also perform runner cleanup.
"""
if self.runner:
self.runner.stop()
if clean:
self.runner.cleanup()
if clean:
if isinstance(self.profile, Profile):
self.profile.cleanup()
self.profile = None
def restart(self, prefs=None, clean=True):
"""
Close then start the managed Gecko process.
:param prefs: Dictionary of preference names and values.
:param clean: If True, reset the profile before starting.
"""
if prefs:
self.prefs = prefs
else:
self.prefs = None
self.close(clean=clean)
self.start()
def update_process(self, pid, timeout=None):
"""Update the process to track when the application re-launched itself"""
if sys.platform.startswith("darwin"):
# The new process handler is only supported on MacOS yet
returncode = self.runner.process_handler.update_process(pid, timeout)
if returncode not in [0, UNKNOWN_RETURNCODE]:
raise OSError(
f"Old process inappropriately quit with exit code: {returncode}"
)
else:
returncode = self.runner.process_handler.check_for_detached(pid)
class FennecInstance(GeckoInstance):
fennec_prefs = {
# Enable output for dump() and chrome console API
"browser.dom.window.dump.enabled": True,
"devtools.console.stdout.chrome": True,
# Disable safe browsing / tracking protection updates
"browser.safebrowsing.update.enabled": False,
# Do not restore the last open set of tabs if the browser has crashed
"browser.sessionstore.resume_from_crash": False,
}
def __init__(
self,
emulator_binary=None,
avd_home=None,
avd=None,
adb_path=None,
serial=None,
connect_to_running_emulator=False,
package_name=None,
env=None,
*args,
**kwargs,
):
required_prefs = deepcopy(FennecInstance.fennec_prefs)
required_prefs.update(kwargs.get("prefs", {}))
super(FennecInstance, self).__init__(*args, **kwargs)
self.required_prefs.update(required_prefs)
self.runner_class = FennecEmulatorRunner
# runner args
self._package_name = package_name
self.emulator_binary = emulator_binary
self.avd_home = avd_home
self.adb_path = adb_path
self.avd = avd
self.env = env
self.serial = serial
self.connect_to_running_emulator = connect_to_running_emulator
@property
def package_name(self):
"""
Name of app to run on emulator.
Note that FennecInstance does not use self.binary
"""
if self._package_name is None:
self._package_name = "org.mozilla.fennec"
user = os.getenv("USER")
if user:
self._package_name += "_" + user
return self._package_name
def start(self):
self._update_profile(self.profile)
self.runner = self.runner_class(**self._get_runner_args())
try:
if self.connect_to_running_emulator:
self.runner.device.connect()
self.runner.start()
except Exception:
exc_cls, exc, tb = sys.exc_info()
raise exc_cls(
f"Error possibly due to runner or device args: {exc}"
).with_traceback(tb)
# forward marionette port
self.runner.device.device.forward(
local=f"tcp:{self.marionette_port}",
remote=f"tcp:{self.marionette_port}",
)
def _get_runner_args(self):
process_args = {
"processOutputLine": [NullOutput()],
"universal_newlines": True,
}
env = {} if self.env is None else self.env.copy()
runner_args = {
"app": self.package_name,
"avd_home": self.avd_home,
"adb_path": self.adb_path,
"binary": self.emulator_binary,
"env": env,
"profile": self.profile,
"cmdargs": ["-marionette"] + self.app_args,
"symbols_path": self.symbols_path,
"process_args": process_args,
"logdir": self.workspace or os.getcwd(),
"serial": self.serial,
}
if self.avd:
runner_args["avd"] = self.avd
return runner_args
def close(self, clean=False):
"""
Close the managed Gecko process.
If `clean` is True and the Fennec instance is running in an
emulator managed by mozrunner, this will stop the emulator.
:param clean: If True, also perform runner cleanup.
"""
super(FennecInstance, self).close(clean)
if clean and self.runner and self.runner.device.connected:
try:
self.runner.device.device.remove_forwards(f"tcp:{self.marionette_port}")
self.unresponsive_count = 0
except Exception:
self.unresponsive_count += 1
traceback.print_exception(*sys.exc_info())
class DesktopInstance(GeckoInstance):
desktop_prefs = {
# Disable Firefox old build background check
"app.update.checkInstallTime": False,
# Disable automatically upgrading Firefox
#
# Note: Possible update tests could reset or flip the value to allow
# updates to be downloaded and applied.
"app.update.disabledForTesting": True,
# !!! For backward compatibility up to Firefox 64. Only remove
# when this Firefox version is no longer supported by the client !!!
"app.update.auto": False,
# Disable the profile backup service.
"browser.backup.enabled": False,
# Don't show the content blocking introduction panel
# We use a larger number than the default 22 to have some buffer
# This can be removed once Firefox 69 and 68 ESR and are no longer supported.
"browser.contentblocking.introCount": 99,
# Enable output for dump() and chrome console API
"browser.dom.window.dump.enabled": True,
"devtools.console.stdout.chrome": True,
# Indicate that the download panel has been shown once so that whichever
# download test runs first doesn"t show the popup inconsistently
"browser.download.panel.shown": True,
# Do not show the EULA notification which can interfer with tests
"browser.EULA.override": True,
# Disable Activity Stream telemetry pings
"browser.newtabpage.activity-stream.telemetry": False,
# Always display a blank page
"browser.newtabpage.enabled": False,
# Background thumbnails in particular cause grief, and disabling thumbnails
# in general can"t hurt - we re-enable them when tests need them
"browser.pagethumbnails.capturing_disabled": True,
# Disable safe browsing / tracking protection updates
"browser.safebrowsing.update.enabled": False,
# Disable updates to search engines
"browser.search.update": False,
# Do not restore the last open set of tabs if the browser has crashed
"browser.sessionstore.resume_from_crash": False,
# Don't check for the default web browser during startup
"browser.shell.checkDefaultBrowser": False,
# Disable session restore infobar
"browser.startup.couldRestoreSession.count": -1,
# Needed for branded builds to prevent opening a second tab on startup
"browser.startup.homepage_override.mstone": "ignore",
# Start with a blank page by default
"browser.startup.page": 0,
# Unload the previously selected tab immediately
"browser.tabs.remote.unloadDelayMs": 0,
# Don't unload tabs when available memory is running low
"browser.tabs.unloadOnLowMemory": False,
# Do not warn when closing all open tabs
"browser.tabs.warnOnClose": False,
# Do not warn when closing all other open tabs
"browser.tabs.warnOnCloseOtherTabs": False,
# Do not warn when multiple tabs will be opened
"browser.tabs.warnOnOpen": False,
# Don't show the Bookmarks Toolbar on any tab (the above pref that
# disables the New Tab Page ends up showing the toolbar on about:blank).
"browser.toolbars.bookmarks.visibility": "never",
# Disable the UI tour
"browser.uitour.enabled": False,
# Turn off Merino suggestions in the location bar so as not to trigger network
# connections.
"browser.urlbar.merino.endpointURL": "",
# Turn off search suggestions in the location bar so as not to trigger network
# connections.
"browser.urlbar.suggest.searches": False,
# Don't warn when exiting the browser
"browser.warnOnQuit": False,
# Disable the QoS manager on MacOS and the priority manager on all other
# platforms to not cause stalled processes in background tabs when the
# overall CPU load on the machine is high.
#
# TODO: Should be considered to get removed once bug 1960741 is fixed.
"threads.lower_mainthread_priority_in_background.enabled": False,
"dom.ipc.processPriorityManager.enabled": False,
# Turn off semantic history search as it triggers network connections to
# download ML models.
"places.semanticHistory.featureGate": False,
# Disable first-run welcome page
"startup.homepage_welcome_url": "about:blank",
"startup.homepage_welcome_url.additional": "",
}
def __init__(self, *args, **kwargs):
required_prefs = deepcopy(DesktopInstance.desktop_prefs)
required_prefs.update(kwargs.get("prefs", {}))
super(DesktopInstance, self).__init__(*args, **kwargs)
self.required_prefs.update(required_prefs)
class ThunderbirdInstance(GeckoInstance):
def __init__(self, *args, **kwargs):
super(ThunderbirdInstance, self).__init__(*args, **kwargs)
try:
# Copied alongside in the test archive
from .thunderbirdinstance import thunderbird_prefs
except ImportError:
try:
# Directly from the source tree
here = os.path.dirname(__file__)
sys.path.append(
os.path.join(here, "../../../../comm/testing/marionette")
)
from thunderbirdinstance import thunderbird_prefs
except ImportError:
thunderbird_prefs = {}
self.required_prefs.update(thunderbird_prefs)
class NullOutput:
def __call__(self, line):
pass
apps = {
"fennec": FennecInstance,
"fxdesktop": DesktopInstance,
"thunderbird": ThunderbirdInstance,
}
app_ids = {
"{aa3c5121-dab2-40e2-81ca-7ea25febc110}": "fennec",
"{ec8030f7-c20a-464f-9b0e-13a3a9e97384}": "fxdesktop",
"{3550f703-e582-4d05-9a08-453d09bdfdc6}": "thunderbird",
}
|