# 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/.
import sys
import unittest
from urllib.parse import quote
from marionette_driver import errors
from marionette_driver.by import By
from marionette_harness import MarionetteTestCase
def inline(doc):
return "data:text/html;charset=utf-8,{}".format(quote(doc))
class TestServerQuitApplication(MarionetteTestCase):
def tearDown(self):
if self.marionette.session is None:
self.marionette.start_session()
def quit(self, flags=None, safe_mode=False):
body = {}
if flags is not None:
body["flags"] = list(
flags,
)
if safe_mode:
body["safeMode"] = safe_mode
resp = self.marionette._send_message("Marionette:Quit", body)
self.marionette.session_id = None
self.marionette.session = None
self.marionette.process_id = None
self.marionette.profile = None
self.marionette.window = None
self.assertIn("cause", resp)
self.marionette.client.close()
self.marionette.instance.runner.wait()
return resp["cause"]
def test_types(self):
for typ in [42, True, "foo", []]:
print("testing type {}".format(type(typ)))
with self.assertRaises(errors.InvalidArgumentException):
self.marionette._send_message("Marionette:Quit", typ)
with self.assertRaises(errors.InvalidArgumentException):
self.quit("foo")
def test_undefined_default(self):
cause = self.quit()
self.assertEqual("shutdown", cause)
def test_empty_default(self):
cause = self.quit(())
self.assertEqual("shutdown", cause)
def test_incompatible_quit_flags(self):
with self.assertRaises(errors.InvalidArgumentException):
self.quit(("eAttemptQuit", "eForceQuit"))
def test_attempt_quit(self):
cause = self.quit(("eAttemptQuit",))
self.assertEqual("shutdown", cause)
def test_force_quit(self):
cause = self.quit(("eForceQuit",))
self.assertEqual("shutdown", cause)
def test_safe_mode_requires_restart(self):
with self.assertRaises(errors.InvalidArgumentException):
self.quit(("eAttemptQuit",), True)
@unittest.skipUnless(sys.platform.startswith("darwin"), "Only supported on MacOS")
def test_silent_quit_missing_windowless_capability(self):
with self.assertRaises(errors.UnsupportedOperationException):
self.quit(("eSilently",))
class TestQuitRestart(MarionetteTestCase):
def setUp(self):
MarionetteTestCase.setUp(self)
self.pid = self.marionette.process_id
self.profile = self.marionette.profile
self.session_id = self.marionette.session_id
# Use a preference to check that the restart was successful. If its
# value has not been forced, a restart will cause a reset of it.
self.assertNotEqual(
self.marionette.get_pref("startup.homepage_welcome_url"), "about:about"
)
self.marionette.set_pref("startup.homepage_welcome_url", "about:about")
def tearDown(self):
# Ensure to restart a session if none exist for clean-up
if self.marionette.session is None:
self.marionette.start_session()
self.marionette.clear_pref("startup.homepage_welcome_url")
MarionetteTestCase.tearDown(self)
@property
def is_safe_mode(self):
with self.marionette.using_context("chrome"):
return self.marionette.execute_script(
"""
return Services.appinfo.inSafeMode;
"""
)
def shutdown(self, restart=False):
self.marionette.set_context("chrome")
self.marionette.execute_script(
"""
let flags = Ci.nsIAppStartup.eAttemptQuit;
if (arguments[0]) {
flags |= Ci.nsIAppStartup.eRestart;
}
Services.startup.quit(flags);
""",
script_args=(restart,),
)
def test_force_restart(self):
self.marionette.restart(in_app=False)
self.assertEqual(self.marionette.profile, self.profile)
self.assertNotEqual(self.marionette.session_id, self.session_id)
# A forced restart will cause a new process id
self.assertNotEqual(self.marionette.process_id, self.pid)
self.assertNotEqual(
self.marionette.get_pref("startup.homepage_welcome_url"), "about:about"
)
def test_force_clean_restart(self):
self.marionette.restart(in_app=False, clean=True)
self.assertNotEqual(self.marionette.profile, self.profile)
self.assertNotEqual(self.marionette.session_id, self.session_id)
# A forced restart will cause a new process id
self.assertNotEqual(self.marionette.process_id, self.pid)
self.assertNotEqual(
self.marionette.get_pref("startup.homepage_welcome_url"), "about:about"
)
def test_force_quit(self):
self.marionette.quit(in_app=False)
self.assertEqual(self.marionette.session, None)
with self.assertRaisesRegexp(
errors.InvalidSessionIdException, "Please start a session"
):
self.marionette.get_url()
def test_force_clean_quit(self):
self.marionette.quit(in_app=False, clean=True)
self.assertEqual(self.marionette.session, None)
with self.assertRaisesRegexp(
errors.InvalidSessionIdException, "Please start a session"
):
self.marionette.get_url()
self.marionette.start_session()
self.assertNotEqual(self.marionette.profile, self.profile)
self.assertNotEqual(self.marionette.session_id, self.session_id)
self.assertNotEqual(
self.marionette.get_pref("startup.homepage_welcome_url"), "about:about"
)
def test_quit_no_in_app_and_clean(self):
# Test that in_app and clean cannot be used in combination
with self.assertRaisesRegexp(
ValueError, "cannot be triggered with the clean flag set"
):
self.marionette.quit(in_app=True, clean=True)
def test_restart_no_in_app_and_clean(self):
# Test that in_app and clean cannot be used in combination
with self.assertRaisesRegexp(
ValueError, "cannot be triggered with the clean flag set"
):
self.marionette.restart(in_app=True, clean=True)
def test_restart_preserves_requested_capabilities(self):
self.marionette.delete_session()
self.marionette.start_session(capabilities={"moz:fooBar": True})
self.marionette.restart(in_app=False)
self.assertEqual(self.marionette.session.get("moz:fooBar"), True)
def test_restart_safe_mode(self):
try:
self.assertFalse(self.is_safe_mode, "Safe Mode is unexpectedly enabled")
self.marionette.restart(safe_mode=True)
self.assertTrue(self.is_safe_mode, "Safe Mode is not enabled")
finally:
self.marionette.quit(in_app=False, clean=True)
def test_restart_safe_mode_requires_in_app(self):
self.assertFalse(self.is_safe_mode, "Safe Mode is unexpectedly enabled")
with self.assertRaisesRegexp(ValueError, "in_app restart is required"):
self.marionette.restart(in_app=False, safe_mode=True)
self.assertFalse(self.is_safe_mode, "Safe Mode is unexpectedly enabled")
self.marionette.quit(in_app=False, clean=True)
def test_in_app_restart(self):
details = self.marionette.restart()
self.assertTrue(details["in_app"], "Expected in_app restart")
self.assertFalse(details["forced"], "Expected non-forced shutdown")
self.assertEqual(self.marionette.profile, self.profile)
self.assertNotEqual(self.marionette.session_id, self.session_id)
self.assertNotEqual(self.marionette.process_id, self.pid)
self.assertNotEqual(
self.marionette.get_pref("startup.homepage_welcome_url"), "about:about"
)
def test_in_app_restart_component_prevents_shutdown(self):
with self.marionette.using_context("chrome"):
self.marionette.execute_script(
"""
Services.obs.addObserver(subject => {
let cancelQuit = subject.QueryInterface(Ci.nsISupportsPRBool);
cancelQuit.data = true;
}, "quit-application-requested");
"""
)
details = self.marionette.restart()
self.assertTrue(details["in_app"], "Expected in_app restart")
self.assertTrue(details["forced"], "Expected forced shutdown")
self.assertEqual(self.marionette.profile, self.profile)
self.assertNotEqual(self.marionette.session_id, self.session_id)
self.assertNotEqual(self.marionette.process_id, self.pid)
self.assertNotEqual(
self.marionette.get_pref("startup.homepage_welcome_url"), "about:about"
)
def test_in_app_restart_with_callback(self):
details = self.marionette.restart(callback=lambda: self.shutdown(restart=True))
self.assertTrue(details["in_app"], "Expected in_app restart")
self.assertEqual(self.marionette.profile, self.profile)
self.assertNotEqual(self.marionette.session_id, self.session_id)
self.assertNotEqual(self.marionette.process_id, self.pid)
self.assertNotEqual(
self.marionette.get_pref("startup.homepage_welcome_url"), "about:about"
)
def test_in_app_restart_with_non_callable_callback(self):
with self.assertRaisesRegexp(ValueError, "is not callable"):
self.marionette.restart(callback=4)
self.assertEqual(self.marionette.instance.runner.returncode, None)
self.assertEqual(self.marionette.is_shutting_down, False)
@unittest.skipIf(sys.platform.startswith("win"), "Bug 1493796")
def test_in_app_restart_with_callback_but_process_quits_instead(self):
try:
timeout_shutdown = self.marionette.shutdown_timeout
timeout_startup = self.marionette.startup_timeout
self.marionette.shutdown_timeout = 5
self.marionette.startup_timeout = 0
with self.assertRaisesRegexp(
IOError, "Process unexpectedly quit without restarting"
):
self.marionette.restart(callback=lambda: self.shutdown(restart=False))
finally:
self.marionette.shutdown_timeout = timeout_shutdown
self.marionette.startup_timeout = timeout_startup
@unittest.skipIf(sys.platform.startswith("win"), "Bug 1493796")
def test_in_app_restart_with_callback_missing_shutdown(self):
try:
timeout_shutdown = self.marionette.shutdown_timeout
timeout_startup = self.marionette.startup_timeout
self.marionette.shutdown_timeout = 5
self.marionette.startup_timeout = 0
with self.assertRaisesRegexp(
IOError, "the connection to Marionette server is lost"
):
self.marionette.restart(in_app=True, callback=lambda: False)
finally:
self.marionette.shutdown_timeout = timeout_shutdown
self.marionette.startup_timeout = timeout_startup
def test_in_app_restart_preserves_requested_capabilities(self):
self.marionette.delete_session()
self.marionette.start_session(capabilities={"moz:fooBar": True})
details = self.marionette.restart()
self.assertTrue(details["in_app"], "Expected in_app restart")
self.assertEqual(self.marionette.session.get("moz:fooBar"), True)
@unittest.skipUnless(sys.platform.startswith("darwin"), "Only supported on MacOS")
def test_in_app_silent_restart_fails_without_windowless_flag_on_mac_os(self):
self.marionette.delete_session()
self.marionette.start_session()
with self.assertRaises(errors.UnsupportedOperationException):
self.marionette.restart(silent=True)
@unittest.skipUnless(sys.platform.startswith("darwin"), "Only supported on MacOS")
def test_in_app_silent_restart_windowless_flag_on_mac_os(self):
self.marionette.delete_session()
self.marionette.start_session(capabilities={"moz:windowless": True})
self.marionette.restart(silent=True)
self.assertTrue(self.marionette.session_capabilities["moz:windowless"])
self.marionette.restart()
self.assertTrue(self.marionette.session_capabilities["moz:windowless"])
self.marionette.delete_session()
@unittest.skipUnless(sys.platform.startswith("darwin"), "Only supported on MacOS")
def test_in_app_silent_restart_requires_in_app(self):
self.marionette.delete_session()
self.marionette.start_session(capabilities={"moz:windowless": True})
with self.assertRaisesRegexp(ValueError, "in_app restart is required"):
self.marionette.restart(in_app=False, silent=True)
self.marionette.delete_session()
@unittest.skipIf(
sys.platform.startswith("darwin"), "Not supported on other platforms than MacOS"
)
def test_in_app_silent_restart_windowless_flag_unsupported_platforms(self):
self.marionette.delete_session()
with self.assertRaises(errors.SessionNotCreatedException):
self.marionette.start_session(capabilities={"moz:windowless": True})
def test_in_app_quit(self):
details = self.marionette.quit()
self.assertTrue(details["in_app"], "Expected in_app shutdown")
self.assertFalse(details["forced"], "Expected non-forced shutdown")
self.assertEqual(self.marionette.instance.runner.returncode, 0)
self.assertEqual(self.marionette.session, None)
with self.assertRaisesRegexp(
errors.InvalidSessionIdException, "Please start a session"
):
self.marionette.get_url()
self.marionette.start_session()
self.assertEqual(self.marionette.profile, self.profile)
self.assertNotEqual(self.marionette.session_id, self.session_id)
self.assertNotEqual(
self.marionette.get_pref("startup.homepage_welcome_url"), "about:about"
)
def test_in_app_quit_forced_because_component_prevents_shutdown(self):
with self.marionette.using_context("chrome"):
self.marionette.execute_script(
"""
Services.obs.addObserver(subject => {
let cancelQuit = subject.QueryInterface(Ci.nsISupportsPRBool);
cancelQuit.data = true;
}, "quit-application-requested");
"""
)
details = self.marionette.quit()
self.assertTrue(details["in_app"], "Expected in_app shutdown")
self.assertTrue(details["forced"], "Expected forced shutdown")
self.assertEqual(self.marionette.instance.runner.returncode, 0)
self.assertEqual(self.marionette.session, None)
with self.assertRaisesRegexp(
errors.InvalidSessionIdException, "Please start a session"
):
self.marionette.get_url()
self.marionette.start_session()
self.assertEqual(self.marionette.profile, self.profile)
self.assertNotEqual(self.marionette.session_id, self.session_id)
self.assertNotEqual(
self.marionette.get_pref("startup.homepage_welcome_url"), "about:about"
)
def test_in_app_quit_with_callback(self):
details = self.marionette.quit(callback=self.shutdown)
self.assertTrue(details["in_app"], "Expected in_app shutdown")
self.assertFalse(details["forced"], "Expected non-forced shutdown")
self.assertEqual(self.marionette.instance.runner.returncode, 0)
self.assertEqual(self.marionette.is_shutting_down, False)
self.assertEqual(self.marionette.session, None)
with self.assertRaisesRegexp(
errors.InvalidSessionIdException, "Please start a session"
):
self.marionette.get_url()
self.marionette.start_session()
self.assertEqual(self.marionette.profile, self.profile)
self.assertNotEqual(self.marionette.session_id, self.session_id)
self.assertNotEqual(
self.marionette.get_pref("startup.homepage_welcome_url"), "about:about"
)
def test_in_app_quit_with_non_callable_callback(self):
with self.assertRaisesRegexp(ValueError, "is not callable"):
self.marionette.quit(callback=4)
self.assertEqual(self.marionette.instance.runner.returncode, None)
self.assertEqual(self.marionette.is_shutting_down, False)
def test_in_app_quit_forced_because_callback_does_not_shutdown(self):
try:
timeout = self.marionette.shutdown_timeout
self.marionette.shutdown_timeout = 5
with self.assertRaisesRegexp(IOError, "Process still running"):
self.marionette.quit(in_app=True, callback=lambda: False)
self.assertNotEqual(self.marionette.instance.runner.returncode, None)
self.assertEqual(self.marionette.is_shutting_down, False)
finally:
self.marionette.shutdown_timeout = timeout
self.marionette.start_session()
def test_in_app_quit_with_callback_that_raises_an_exception(self):
def errorneous_callback():
raise Exception("foo")
with self.assertRaisesRegexp(Exception, "foo"):
self.marionette.quit(in_app=True, callback=errorneous_callback)
self.assertEqual(self.marionette.instance.runner.returncode, None)
self.assertEqual(self.marionette.is_shutting_down, False)
self.assertIsNotNone(self.marionette.session)
self.marionette.current_window_handle
def test_in_app_quit_with_dismissed_beforeunload_prompt(self):
self.marionette.navigate(
inline(
"""
"""
)
)
self.marionette.find_element(By.TAG_NAME, "input").send_keys("foo")
self.marionette.quit()
self.assertNotEqual(self.marionette.instance.runner.returncode, None)
self.marionette.start_session()
def test_reset_context_after_quit_by_set_context(self):
# Check that we are in content context which is used by default in
# Marionette
self.assertNotIn(
"chrome://",
self.marionette.get_url(),
"Context does not default to content",
)
self.marionette.set_context("chrome")
self.marionette.quit()
self.assertEqual(self.marionette.session, None)
self.marionette.start_session()
self.assertNotIn(
"chrome://",
self.marionette.get_url(),
"Not in content context after quit with using_context",
)
def test_reset_context_after_quit_by_using_context(self):
# Check that we are in content context which is used by default in
# Marionette
self.assertNotIn(
"chrome://",
self.marionette.get_url(),
"Context does not default to content",
)
with self.marionette.using_context("chrome"):
self.marionette.quit()
self.assertEqual(self.marionette.session, None)
self.marionette.start_session()
self.assertNotIn(
"chrome://",
self.marionette.get_url(),
"Not in content context after quit with using_context",
)
def test_keep_context_after_restart_by_set_context(self):
# Check that we are in content context which is used by default in
# Marionette
self.assertNotIn(
"chrome://", self.marionette.get_url(), "Context doesn't default to content"
)
# restart while we are in chrome context
self.marionette.set_context("chrome")
self.marionette.restart()
self.assertNotEqual(self.marionette.process_id, self.pid)
self.assertIn(
"chrome://",
self.marionette.get_url(),
"Not in chrome context after a restart with set_context",
)
def test_keep_context_after_restart_by_using_context(self):
# Check that we are in content context which is used by default in
# Marionette
self.assertNotIn(
"chrome://",
self.marionette.get_url(),
"Context does not default to content",
)
# restart while we are in chrome context
with self.marionette.using_context("chrome"):
self.marionette.restart()
self.assertNotEqual(self.marionette.process_id, self.pid)
self.assertIn(
"chrome://",
self.marionette.get_url(),
"Not in chrome context after a restart with using_context",
)