mirror of
				https://github.com/python/cpython.git
				synced 2025-11-04 07:31:38 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			506 lines
		
	
	
	
		
			17 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
			
		
		
	
	
			506 lines
		
	
	
	
		
			17 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
import os
 | 
						|
import re
 | 
						|
import shlex
 | 
						|
import subprocess
 | 
						|
import sys
 | 
						|
import unittest
 | 
						|
import webbrowser
 | 
						|
from test import support
 | 
						|
from test.support import import_helper
 | 
						|
from test.support import is_apple_mobile
 | 
						|
from test.support import os_helper
 | 
						|
from test.support import requires_subprocess
 | 
						|
from test.support import threading_helper
 | 
						|
from unittest import mock
 | 
						|
 | 
						|
# The webbrowser module uses threading locks
 | 
						|
threading_helper.requires_working_threading(module=True)
 | 
						|
 | 
						|
URL = 'https://www.example.com'
 | 
						|
CMD_NAME = 'test'
 | 
						|
 | 
						|
 | 
						|
class PopenMock(mock.MagicMock):
 | 
						|
 | 
						|
    def poll(self):
 | 
						|
        return 0
 | 
						|
 | 
						|
    def wait(self, seconds=None):
 | 
						|
        return 0
 | 
						|
 | 
						|
 | 
						|
@requires_subprocess()
 | 
						|
class CommandTestMixin:
 | 
						|
 | 
						|
    def _test(self, meth, *, args=[URL], kw={}, options, arguments):
 | 
						|
        """Given a web browser instance method name along with arguments and
 | 
						|
        keywords for same (which defaults to the single argument URL), creates
 | 
						|
        a browser instance from the class pointed to by self.browser, calls the
 | 
						|
        indicated instance method with the indicated arguments, and compares
 | 
						|
        the resulting options and arguments passed to Popen by the browser
 | 
						|
        instance against the 'options' and 'args' lists.  Options are compared
 | 
						|
        in a position independent fashion, and the arguments are compared in
 | 
						|
        sequence order to whatever is left over after removing the options.
 | 
						|
 | 
						|
        """
 | 
						|
        popen = PopenMock()
 | 
						|
        support.patch(self, subprocess, 'Popen', popen)
 | 
						|
        browser = self.browser_class(name=CMD_NAME)
 | 
						|
        getattr(browser, meth)(*args, **kw)
 | 
						|
        popen_args = subprocess.Popen.call_args[0][0]
 | 
						|
        self.assertEqual(popen_args[0], CMD_NAME)
 | 
						|
        popen_args.pop(0)
 | 
						|
        for option in options:
 | 
						|
            self.assertIn(option, popen_args)
 | 
						|
            popen_args.pop(popen_args.index(option))
 | 
						|
        self.assertEqual(popen_args, arguments)
 | 
						|
 | 
						|
 | 
						|
class GenericBrowserCommandTest(CommandTestMixin, unittest.TestCase):
 | 
						|
 | 
						|
    browser_class = webbrowser.GenericBrowser
 | 
						|
 | 
						|
    def test_open(self):
 | 
						|
        self._test('open',
 | 
						|
                   options=[],
 | 
						|
                   arguments=[URL])
 | 
						|
 | 
						|
 | 
						|
class BackgroundBrowserCommandTest(CommandTestMixin, unittest.TestCase):
 | 
						|
 | 
						|
    browser_class = webbrowser.BackgroundBrowser
 | 
						|
 | 
						|
    def test_open(self):
 | 
						|
        self._test('open',
 | 
						|
                   options=[],
 | 
						|
                   arguments=[URL])
 | 
						|
 | 
						|
 | 
						|
class ChromeCommandTest(CommandTestMixin, unittest.TestCase):
 | 
						|
 | 
						|
    browser_class = webbrowser.Chrome
 | 
						|
 | 
						|
    def test_open(self):
 | 
						|
        self._test('open',
 | 
						|
                   options=[],
 | 
						|
                   arguments=[URL])
 | 
						|
 | 
						|
    def test_open_with_autoraise_false(self):
 | 
						|
        self._test('open', kw=dict(autoraise=False),
 | 
						|
                   options=[],
 | 
						|
                   arguments=[URL])
 | 
						|
 | 
						|
    def test_open_new(self):
 | 
						|
        self._test('open_new',
 | 
						|
                   options=['--new-window'],
 | 
						|
                   arguments=[URL])
 | 
						|
 | 
						|
    def test_open_new_tab(self):
 | 
						|
        self._test('open_new_tab',
 | 
						|
                   options=[],
 | 
						|
                   arguments=[URL])
 | 
						|
 | 
						|
    def test_open_bad_new_parameter(self):
 | 
						|
        with self.assertRaisesRegex(webbrowser.Error,
 | 
						|
                                    re.escape("Bad 'new' parameter to open(); "
 | 
						|
                                              "expected 0, 1, or 2, got 999")):
 | 
						|
            self._test('open',
 | 
						|
                       options=[],
 | 
						|
                       arguments=[URL],
 | 
						|
                       kw=dict(new=999))
 | 
						|
 | 
						|
 | 
						|
class EdgeCommandTest(CommandTestMixin, unittest.TestCase):
 | 
						|
 | 
						|
    browser_class = webbrowser.Edge
 | 
						|
 | 
						|
    def test_open(self):
 | 
						|
        self._test('open',
 | 
						|
                   options=[],
 | 
						|
                   arguments=[URL])
 | 
						|
 | 
						|
    def test_open_with_autoraise_false(self):
 | 
						|
        self._test('open', kw=dict(autoraise=False),
 | 
						|
                   options=[],
 | 
						|
                   arguments=[URL])
 | 
						|
 | 
						|
    def test_open_new(self):
 | 
						|
        self._test('open_new',
 | 
						|
                   options=['--new-window'],
 | 
						|
                   arguments=[URL])
 | 
						|
 | 
						|
    def test_open_new_tab(self):
 | 
						|
        self._test('open_new_tab',
 | 
						|
                   options=[],
 | 
						|
                   arguments=[URL])
 | 
						|
 | 
						|
 | 
						|
class MozillaCommandTest(CommandTestMixin, unittest.TestCase):
 | 
						|
 | 
						|
    browser_class = webbrowser.Mozilla
 | 
						|
 | 
						|
    def test_open(self):
 | 
						|
        self._test('open',
 | 
						|
                   options=[],
 | 
						|
                   arguments=[URL])
 | 
						|
 | 
						|
    def test_open_with_autoraise_false(self):
 | 
						|
        self._test('open', kw=dict(autoraise=False),
 | 
						|
                   options=[],
 | 
						|
                   arguments=[URL])
 | 
						|
 | 
						|
    def test_open_new(self):
 | 
						|
        self._test('open_new',
 | 
						|
                   options=[],
 | 
						|
                   arguments=['-new-window', URL])
 | 
						|
 | 
						|
    def test_open_new_tab(self):
 | 
						|
        self._test('open_new_tab',
 | 
						|
                   options=[],
 | 
						|
                   arguments=['-new-tab', URL])
 | 
						|
 | 
						|
 | 
						|
class EpiphanyCommandTest(CommandTestMixin, unittest.TestCase):
 | 
						|
 | 
						|
    browser_class = webbrowser.Epiphany
 | 
						|
 | 
						|
    def test_open(self):
 | 
						|
        self._test('open',
 | 
						|
                   options=['-n'],
 | 
						|
                   arguments=[URL])
 | 
						|
 | 
						|
    def test_open_with_autoraise_false(self):
 | 
						|
        self._test('open', kw=dict(autoraise=False),
 | 
						|
                   options=['-noraise', '-n'],
 | 
						|
                   arguments=[URL])
 | 
						|
 | 
						|
    def test_open_new(self):
 | 
						|
        self._test('open_new',
 | 
						|
                   options=['-w'],
 | 
						|
                   arguments=[URL])
 | 
						|
 | 
						|
    def test_open_new_tab(self):
 | 
						|
        self._test('open_new_tab',
 | 
						|
                   options=['-w'],
 | 
						|
                   arguments=[URL])
 | 
						|
 | 
						|
 | 
						|
class OperaCommandTest(CommandTestMixin, unittest.TestCase):
 | 
						|
 | 
						|
    browser_class = webbrowser.Opera
 | 
						|
 | 
						|
    def test_open(self):
 | 
						|
        self._test('open',
 | 
						|
                   options=[],
 | 
						|
                   arguments=[URL])
 | 
						|
 | 
						|
    def test_open_with_autoraise_false(self):
 | 
						|
        self._test('open', kw=dict(autoraise=False),
 | 
						|
                   options=[],
 | 
						|
                   arguments=[URL])
 | 
						|
 | 
						|
    def test_open_new(self):
 | 
						|
        self._test('open_new',
 | 
						|
                   options=['--new-window'],
 | 
						|
                   arguments=[URL])
 | 
						|
 | 
						|
    def test_open_new_tab(self):
 | 
						|
        self._test('open_new_tab',
 | 
						|
                   options=[],
 | 
						|
                   arguments=[URL])
 | 
						|
 | 
						|
 | 
						|
class ELinksCommandTest(CommandTestMixin, unittest.TestCase):
 | 
						|
 | 
						|
    browser_class = webbrowser.Elinks
 | 
						|
 | 
						|
    def test_open(self):
 | 
						|
        self._test('open', options=['-remote'],
 | 
						|
                   arguments=[f'openURL({URL})'])
 | 
						|
 | 
						|
    def test_open_with_autoraise_false(self):
 | 
						|
        self._test('open',
 | 
						|
                   options=['-remote'],
 | 
						|
                   arguments=[f'openURL({URL})'])
 | 
						|
 | 
						|
    def test_open_new(self):
 | 
						|
        self._test('open_new',
 | 
						|
                   options=['-remote'],
 | 
						|
                   arguments=[f'openURL({URL},new-window)'])
 | 
						|
 | 
						|
    def test_open_new_tab(self):
 | 
						|
        self._test('open_new_tab',
 | 
						|
                   options=['-remote'],
 | 
						|
                   arguments=[f'openURL({URL},new-tab)'])
 | 
						|
 | 
						|
 | 
						|
@unittest.skipUnless(sys.platform == "ios", "Test only applicable to iOS")
 | 
						|
class IOSBrowserTest(unittest.TestCase):
 | 
						|
    def _obj_ref(self, *args):
 | 
						|
        # Construct a string representation of the arguments that can be used
 | 
						|
        # as a proxy for object instance references
 | 
						|
        return "|".join(str(a) for a in args)
 | 
						|
 | 
						|
    @unittest.skipIf(getattr(webbrowser, "objc", None) is None,
 | 
						|
                     "iOS Webbrowser tests require ctypes")
 | 
						|
    def setUp(self):
 | 
						|
        # Intercept the the objc library. Wrap the calls to get the
 | 
						|
        # references to classes and selectors to return strings, and
 | 
						|
        # wrap msgSend to return stringified object references
 | 
						|
        self.orig_objc = webbrowser.objc
 | 
						|
 | 
						|
        webbrowser.objc = mock.Mock()
 | 
						|
        webbrowser.objc.objc_getClass = lambda cls: f"C#{cls.decode()}"
 | 
						|
        webbrowser.objc.sel_registerName = lambda sel: f"S#{sel.decode()}"
 | 
						|
        webbrowser.objc.objc_msgSend.side_effect = self._obj_ref
 | 
						|
 | 
						|
    def tearDown(self):
 | 
						|
        webbrowser.objc = self.orig_objc
 | 
						|
 | 
						|
    def _test(self, meth, **kwargs):
 | 
						|
        # The browser always gets focus, there's no concept of separate browser
 | 
						|
        # windows, and there's no API-level control over creating a new tab.
 | 
						|
        # Therefore, all calls to webbrowser are effectively the same.
 | 
						|
        getattr(webbrowser, meth)(URL, **kwargs)
 | 
						|
 | 
						|
        # The ObjC String version of the URL is created with UTF-8 encoding
 | 
						|
        url_string_args = [
 | 
						|
            "C#NSString",
 | 
						|
            "S#stringWithCString:encoding:",
 | 
						|
            b'https://www.example.com',
 | 
						|
            4,
 | 
						|
        ]
 | 
						|
        # The NSURL version of the URL is created from that string
 | 
						|
        url_obj_args = [
 | 
						|
            "C#NSURL",
 | 
						|
            "S#URLWithString:",
 | 
						|
            self._obj_ref(*url_string_args),
 | 
						|
        ]
 | 
						|
        # The openURL call is invoked on the shared application
 | 
						|
        shared_app_args = ["C#UIApplication", "S#sharedApplication"]
 | 
						|
 | 
						|
        # Verify that the last call is the one that opens the URL.
 | 
						|
        webbrowser.objc.objc_msgSend.assert_called_with(
 | 
						|
            self._obj_ref(*shared_app_args),
 | 
						|
            "S#openURL:options:completionHandler:",
 | 
						|
            self._obj_ref(*url_obj_args),
 | 
						|
            None,
 | 
						|
            None
 | 
						|
        )
 | 
						|
 | 
						|
    def test_open(self):
 | 
						|
        self._test('open')
 | 
						|
 | 
						|
    def test_open_with_autoraise_false(self):
 | 
						|
        self._test('open', autoraise=False)
 | 
						|
 | 
						|
    def test_open_new(self):
 | 
						|
        self._test('open_new')
 | 
						|
 | 
						|
    def test_open_new_tab(self):
 | 
						|
        self._test('open_new_tab')
 | 
						|
 | 
						|
 | 
						|
class BrowserRegistrationTest(unittest.TestCase):
 | 
						|
 | 
						|
    def setUp(self):
 | 
						|
        # Ensure we don't alter the real registered browser details
 | 
						|
        self._saved_tryorder = webbrowser._tryorder
 | 
						|
        webbrowser._tryorder = []
 | 
						|
        self._saved_browsers = webbrowser._browsers
 | 
						|
        webbrowser._browsers = {}
 | 
						|
 | 
						|
    def tearDown(self):
 | 
						|
        webbrowser._tryorder = self._saved_tryorder
 | 
						|
        webbrowser._browsers = self._saved_browsers
 | 
						|
 | 
						|
    def _check_registration(self, preferred):
 | 
						|
        class ExampleBrowser:
 | 
						|
            pass
 | 
						|
 | 
						|
        expected_tryorder = []
 | 
						|
        expected_browsers = {}
 | 
						|
 | 
						|
        self.assertEqual(webbrowser._tryorder, expected_tryorder)
 | 
						|
        self.assertEqual(webbrowser._browsers, expected_browsers)
 | 
						|
 | 
						|
        webbrowser.register('Example1', ExampleBrowser)
 | 
						|
        expected_tryorder = ['Example1']
 | 
						|
        expected_browsers['example1'] = [ExampleBrowser, None]
 | 
						|
        self.assertEqual(webbrowser._tryorder, expected_tryorder)
 | 
						|
        self.assertEqual(webbrowser._browsers, expected_browsers)
 | 
						|
 | 
						|
        instance = ExampleBrowser()
 | 
						|
        if preferred is not None:
 | 
						|
            webbrowser.register('example2', ExampleBrowser, instance,
 | 
						|
                                preferred=preferred)
 | 
						|
        else:
 | 
						|
            webbrowser.register('example2', ExampleBrowser, instance)
 | 
						|
        if preferred:
 | 
						|
            expected_tryorder = ['example2', 'Example1']
 | 
						|
        else:
 | 
						|
            expected_tryorder = ['Example1', 'example2']
 | 
						|
        expected_browsers['example2'] = [ExampleBrowser, instance]
 | 
						|
        self.assertEqual(webbrowser._tryorder, expected_tryorder)
 | 
						|
        self.assertEqual(webbrowser._browsers, expected_browsers)
 | 
						|
 | 
						|
    def test_register(self):
 | 
						|
        self._check_registration(preferred=False)
 | 
						|
 | 
						|
    def test_register_default(self):
 | 
						|
        self._check_registration(preferred=None)
 | 
						|
 | 
						|
    def test_register_preferred(self):
 | 
						|
        self._check_registration(preferred=True)
 | 
						|
 | 
						|
    @unittest.skipUnless(sys.platform == "darwin", "macOS specific test")
 | 
						|
    def test_no_xdg_settings_on_macOS(self):
 | 
						|
        # On macOS webbrowser should not use xdg-settings to
 | 
						|
        # look for X11 based browsers (for those users with
 | 
						|
        # XQuartz installed)
 | 
						|
        with mock.patch("subprocess.check_output") as ck_o:
 | 
						|
            webbrowser.register_standard_browsers()
 | 
						|
 | 
						|
        ck_o.assert_not_called()
 | 
						|
 | 
						|
 | 
						|
class ImportTest(unittest.TestCase):
 | 
						|
    def test_register(self):
 | 
						|
        webbrowser = import_helper.import_fresh_module('webbrowser')
 | 
						|
        self.assertIsNone(webbrowser._tryorder)
 | 
						|
        self.assertFalse(webbrowser._browsers)
 | 
						|
 | 
						|
        class ExampleBrowser:
 | 
						|
            pass
 | 
						|
        webbrowser.register('Example1', ExampleBrowser)
 | 
						|
        self.assertTrue(webbrowser._tryorder)
 | 
						|
        self.assertEqual(webbrowser._tryorder[-1], 'Example1')
 | 
						|
        self.assertTrue(webbrowser._browsers)
 | 
						|
        self.assertIn('example1', webbrowser._browsers)
 | 
						|
        self.assertEqual(webbrowser._browsers['example1'], [ExampleBrowser, None])
 | 
						|
 | 
						|
    def test_get(self):
 | 
						|
        webbrowser = import_helper.import_fresh_module('webbrowser')
 | 
						|
        self.assertIsNone(webbrowser._tryorder)
 | 
						|
        self.assertFalse(webbrowser._browsers)
 | 
						|
 | 
						|
        with self.assertRaises(webbrowser.Error):
 | 
						|
            webbrowser.get('fakebrowser')
 | 
						|
        self.assertIsNotNone(webbrowser._tryorder)
 | 
						|
 | 
						|
    @unittest.skipIf(" " in sys.executable, "test assumes no space in path (GH-114452)")
 | 
						|
    def test_synthesize(self):
 | 
						|
        webbrowser = import_helper.import_fresh_module('webbrowser')
 | 
						|
        name = os.path.basename(sys.executable).lower()
 | 
						|
        webbrowser.register(name, None, webbrowser.GenericBrowser(name))
 | 
						|
        webbrowser.get(sys.executable)
 | 
						|
 | 
						|
    @unittest.skipIf(
 | 
						|
        is_apple_mobile,
 | 
						|
        "Apple mobile doesn't allow modifying browser with environment"
 | 
						|
    )
 | 
						|
    def test_environment(self):
 | 
						|
        webbrowser = import_helper.import_fresh_module('webbrowser')
 | 
						|
        try:
 | 
						|
            browser = webbrowser.get().name
 | 
						|
        except webbrowser.Error as err:
 | 
						|
            self.skipTest(str(err))
 | 
						|
        with os_helper.EnvironmentVarGuard() as env:
 | 
						|
            env["BROWSER"] = browser
 | 
						|
            webbrowser = import_helper.import_fresh_module('webbrowser')
 | 
						|
            webbrowser.get()
 | 
						|
 | 
						|
    @unittest.skipIf(
 | 
						|
        is_apple_mobile,
 | 
						|
        "Apple mobile doesn't allow modifying browser with environment"
 | 
						|
    )
 | 
						|
    def test_environment_preferred(self):
 | 
						|
        webbrowser = import_helper.import_fresh_module('webbrowser')
 | 
						|
        try:
 | 
						|
            webbrowser.get()
 | 
						|
            least_preferred_browser = webbrowser.get(webbrowser._tryorder[-1]).name
 | 
						|
        except (webbrowser.Error, IndexError) as err:
 | 
						|
            self.skipTest(str(err))
 | 
						|
 | 
						|
        with os_helper.EnvironmentVarGuard() as env:
 | 
						|
            env["BROWSER"] = least_preferred_browser
 | 
						|
            webbrowser = import_helper.import_fresh_module('webbrowser')
 | 
						|
            self.assertEqual(webbrowser.get().name, least_preferred_browser)
 | 
						|
 | 
						|
        with os_helper.EnvironmentVarGuard() as env:
 | 
						|
            env["BROWSER"] = sys.executable
 | 
						|
            webbrowser = import_helper.import_fresh_module('webbrowser')
 | 
						|
            self.assertEqual(webbrowser.get().name, sys.executable)
 | 
						|
 | 
						|
 | 
						|
class CliTest(unittest.TestCase):
 | 
						|
    def test_parse_args(self):
 | 
						|
        for command, url, new_win in [
 | 
						|
            # No optional arguments
 | 
						|
            ("https://example.com", "https://example.com", 0),
 | 
						|
            # Each optional argument
 | 
						|
            ("https://example.com -n", "https://example.com", 1),
 | 
						|
            ("-n https://example.com", "https://example.com", 1),
 | 
						|
            ("https://example.com -t", "https://example.com", 2),
 | 
						|
            ("-t https://example.com", "https://example.com", 2),
 | 
						|
            # Long form
 | 
						|
            ("https://example.com --new-window", "https://example.com", 1),
 | 
						|
            ("--new-window https://example.com", "https://example.com", 1),
 | 
						|
            ("https://example.com --new-tab", "https://example.com", 2),
 | 
						|
            ("--new-tab https://example.com", "https://example.com", 2),
 | 
						|
        ]:
 | 
						|
            args = webbrowser.parse_args(shlex.split(command))
 | 
						|
 | 
						|
            self.assertEqual(args.url, url)
 | 
						|
            self.assertEqual(args.new_win, new_win)
 | 
						|
 | 
						|
    def test_parse_args_error(self):
 | 
						|
        for command in [
 | 
						|
            # Arguments must not both be given
 | 
						|
            "https://example.com -n -t",
 | 
						|
            "https://example.com --new-window --new-tab",
 | 
						|
            "https://example.com -n --new-tab",
 | 
						|
            "https://example.com --new-window -t",
 | 
						|
        ]:
 | 
						|
            with support.captured_stderr() as stderr:
 | 
						|
                with self.assertRaises(SystemExit):
 | 
						|
                    webbrowser.parse_args(shlex.split(command))
 | 
						|
                self.assertIn(
 | 
						|
                    'error: argument -t/--new-tab: not allowed with argument -n/--new-window',
 | 
						|
                    stderr.getvalue(),
 | 
						|
                )
 | 
						|
 | 
						|
        # Ensure ambiguous shortening fails
 | 
						|
        with support.captured_stderr() as stderr:
 | 
						|
            with self.assertRaises(SystemExit):
 | 
						|
                webbrowser.parse_args(shlex.split("https://example.com --new"))
 | 
						|
            self.assertIn(
 | 
						|
                'error: ambiguous option: --new could match --new-window, --new-tab',
 | 
						|
                stderr.getvalue()
 | 
						|
            )
 | 
						|
 | 
						|
    def test_main(self):
 | 
						|
        for command, expected_url, expected_new_win in [
 | 
						|
            # No optional arguments
 | 
						|
            ("https://example.com", "https://example.com", 0),
 | 
						|
            # Each optional argument
 | 
						|
            ("https://example.com -n", "https://example.com", 1),
 | 
						|
            ("-n https://example.com", "https://example.com", 1),
 | 
						|
            ("https://example.com -t", "https://example.com", 2),
 | 
						|
            ("-t https://example.com", "https://example.com", 2),
 | 
						|
            # Long form
 | 
						|
            ("https://example.com --new-window", "https://example.com", 1),
 | 
						|
            ("--new-window https://example.com", "https://example.com", 1),
 | 
						|
            ("https://example.com --new-tab", "https://example.com", 2),
 | 
						|
            ("--new-tab https://example.com", "https://example.com", 2),
 | 
						|
        ]:
 | 
						|
            with (
 | 
						|
                mock.patch("webbrowser.open", return_value=None) as mock_open,
 | 
						|
                mock.patch("builtins.print", return_value=None),
 | 
						|
            ):
 | 
						|
                webbrowser.main(shlex.split(command))
 | 
						|
                mock_open.assert_called_once_with(expected_url, expected_new_win)
 | 
						|
 | 
						|
 | 
						|
if __name__ == '__main__':
 | 
						|
    unittest.main()
 |