| 
									
										
										
										
											2020-06-11 07:51:18 +08:00
										 |  |  | import contextlib | 
					
						
							| 
									
										
										
										
											2021-09-14 17:31:45 -06:00
										 |  |  | import _imp | 
					
						
							| 
									
										
										
										
											2020-06-11 07:51:18 +08:00
										 |  |  | import importlib | 
					
						
							|  |  |  | import importlib.util | 
					
						
							|  |  |  | import os | 
					
						
							| 
									
										
										
										
											2022-02-07 17:09:07 -08:00
										 |  |  | import shutil | 
					
						
							| 
									
										
										
										
											2020-06-11 07:51:18 +08:00
										 |  |  | import sys | 
					
						
							|  |  |  | import unittest | 
					
						
							|  |  |  | import warnings | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-09-21 10:39:36 +03:00
										 |  |  | from .os_helper import unlink, temp_dir | 
					
						
							| 
									
										
										
										
											2020-06-11 07:51:18 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | @contextlib.contextmanager | 
					
						
							|  |  |  | def _ignore_deprecated_imports(ignore=True): | 
					
						
							|  |  |  |     """Context manager to suppress package and module deprecation
 | 
					
						
							|  |  |  |     warnings when importing them. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     If ignore is False, this context manager has no effect. | 
					
						
							|  |  |  |     """
 | 
					
						
							|  |  |  |     if ignore: | 
					
						
							|  |  |  |         with warnings.catch_warnings(): | 
					
						
							|  |  |  |             warnings.filterwarnings("ignore", ".+ (module|package)", | 
					
						
							|  |  |  |                                     DeprecationWarning) | 
					
						
							|  |  |  |             yield | 
					
						
							|  |  |  |     else: | 
					
						
							|  |  |  |         yield | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | def unload(name): | 
					
						
							|  |  |  |     try: | 
					
						
							|  |  |  |         del sys.modules[name] | 
					
						
							|  |  |  |     except KeyError: | 
					
						
							|  |  |  |         pass | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | def forget(modname): | 
					
						
							|  |  |  |     """'Forget' a module was ever imported.
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     This removes the module from sys.modules and deletes any PEP 3147/488 or | 
					
						
							|  |  |  |     legacy .pyc files. | 
					
						
							|  |  |  |     """
 | 
					
						
							|  |  |  |     unload(modname) | 
					
						
							|  |  |  |     for dirname in sys.path: | 
					
						
							|  |  |  |         source = os.path.join(dirname, modname + '.py') | 
					
						
							|  |  |  |         # It doesn't matter if they exist or not, unlink all possible | 
					
						
							|  |  |  |         # combinations of PEP 3147/488 and legacy pyc files. | 
					
						
							|  |  |  |         unlink(source + 'c') | 
					
						
							|  |  |  |         for opt in ('', 1, 2): | 
					
						
							|  |  |  |             unlink(importlib.util.cache_from_source(source, optimization=opt)) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | def make_legacy_pyc(source): | 
					
						
							|  |  |  |     """Move a PEP 3147/488 pyc file to its legacy pyc location.
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     :param source: The file system path to the source file.  The source file | 
					
						
							|  |  |  |         does not need to exist, however the PEP 3147/488 pyc file must exist. | 
					
						
							|  |  |  |     :return: The file system path to the legacy pyc file. | 
					
						
							|  |  |  |     """
 | 
					
						
							|  |  |  |     pyc_file = importlib.util.cache_from_source(source) | 
					
						
							|  |  |  |     up_one = os.path.dirname(os.path.abspath(source)) | 
					
						
							|  |  |  |     legacy_pyc = os.path.join(up_one, source + 'c') | 
					
						
							| 
									
										
										
										
											2022-02-07 17:09:07 -08:00
										 |  |  |     shutil.move(pyc_file, legacy_pyc) | 
					
						
							| 
									
										
										
										
											2020-06-11 07:51:18 +08:00
										 |  |  |     return legacy_pyc | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | def import_module(name, deprecated=False, *, required_on=()): | 
					
						
							|  |  |  |     """Import and return the module to be tested, raising SkipTest if
 | 
					
						
							|  |  |  |     it is not available. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     If deprecated is True, any module or package deprecation messages | 
					
						
							|  |  |  |     will be suppressed. If a module is required on a platform but optional for | 
					
						
							|  |  |  |     others, set required_on to an iterable of platform prefixes which will be | 
					
						
							|  |  |  |     compared against sys.platform. | 
					
						
							|  |  |  |     """
 | 
					
						
							|  |  |  |     with _ignore_deprecated_imports(deprecated): | 
					
						
							|  |  |  |         try: | 
					
						
							|  |  |  |             return importlib.import_module(name) | 
					
						
							|  |  |  |         except ImportError as msg: | 
					
						
							|  |  |  |             if sys.platform.startswith(tuple(required_on)): | 
					
						
							|  |  |  |                 raise | 
					
						
							|  |  |  |             raise unittest.SkipTest(str(msg)) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-09-30 19:20:39 +03:00
										 |  |  | def _save_and_remove_modules(names): | 
					
						
							|  |  |  |     orig_modules = {} | 
					
						
							|  |  |  |     prefixes = tuple(name + '.' for name in names) | 
					
						
							| 
									
										
										
										
											2020-06-11 07:51:18 +08:00
										 |  |  |     for modname in list(sys.modules): | 
					
						
							| 
									
										
										
										
											2021-09-30 19:20:39 +03:00
										 |  |  |         if modname in names or modname.startswith(prefixes): | 
					
						
							|  |  |  |             orig_modules[modname] = sys.modules.pop(modname) | 
					
						
							|  |  |  |     return orig_modules | 
					
						
							| 
									
										
										
										
											2020-06-11 07:51:18 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-09-14 17:31:45 -06:00
										 |  |  | @contextlib.contextmanager | 
					
						
							|  |  |  | def frozen_modules(enabled=True): | 
					
						
							|  |  |  |     """Force frozen modules to be used (or not).
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     This only applies to modules that haven't been imported yet. | 
					
						
							|  |  |  |     Also, some essential modules will always be imported frozen. | 
					
						
							|  |  |  |     """
 | 
					
						
							|  |  |  |     _imp._override_frozen_modules_for_tests(1 if enabled else -1) | 
					
						
							|  |  |  |     try: | 
					
						
							|  |  |  |         yield | 
					
						
							|  |  |  |     finally: | 
					
						
							|  |  |  |         _imp._override_frozen_modules_for_tests(0) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-02-15 18:16:00 -07:00
										 |  |  | @contextlib.contextmanager | 
					
						
							|  |  |  | def multi_interp_extensions_check(enabled=True): | 
					
						
							|  |  |  |     """Force legacy modules to be allowed in subinterpreters (or not).
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     ("legacy" == single-phase init) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     This only applies to modules that haven't been imported yet. | 
					
						
							|  |  |  |     It overrides the PyInterpreterConfig.check_multi_interp_extensions | 
					
						
							|  |  |  |     setting (see support.run_in_subinterp_with_config() and | 
					
						
							| 
									
										
										
										
											2024-04-24 10:18:24 -06:00
										 |  |  |     _interpreters.create()). | 
					
						
							| 
									
										
										
										
											2023-05-08 16:56:01 -06:00
										 |  |  | 
 | 
					
						
							|  |  |  |     Also see importlib.utils.allowing_all_extensions(). | 
					
						
							| 
									
										
										
										
											2023-02-15 18:16:00 -07:00
										 |  |  |     """
 | 
					
						
							|  |  |  |     old = _imp._override_multi_interp_extensions_check(1 if enabled else -1) | 
					
						
							|  |  |  |     try: | 
					
						
							|  |  |  |         yield | 
					
						
							|  |  |  |     finally: | 
					
						
							|  |  |  |         _imp._override_multi_interp_extensions_check(old) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-09-14 17:31:45 -06:00
										 |  |  | def import_fresh_module(name, fresh=(), blocked=(), *, | 
					
						
							|  |  |  |                         deprecated=False, | 
					
						
							|  |  |  |                         usefrozen=False, | 
					
						
							|  |  |  |                         ): | 
					
						
							| 
									
										
										
										
											2020-06-11 07:51:18 +08:00
										 |  |  |     """Import and return a module, deliberately bypassing sys.modules.
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     This function imports and returns a fresh copy of the named Python module | 
					
						
							|  |  |  |     by removing the named module from sys.modules before doing the import. | 
					
						
							|  |  |  |     Note that unlike reload, the original module is not affected by | 
					
						
							|  |  |  |     this operation. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     *fresh* is an iterable of additional module names that are also removed | 
					
						
							| 
									
										
										
										
											2021-09-30 19:20:39 +03:00
										 |  |  |     from the sys.modules cache before doing the import. If one of these | 
					
						
							|  |  |  |     modules can't be imported, None is returned. | 
					
						
							| 
									
										
										
										
											2020-06-11 07:51:18 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  |     *blocked* is an iterable of module names that are replaced with None | 
					
						
							|  |  |  |     in the module cache during the import to ensure that attempts to import | 
					
						
							|  |  |  |     them raise ImportError. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     The named module and any modules named in the *fresh* and *blocked* | 
					
						
							|  |  |  |     parameters are saved before starting the import and then reinserted into | 
					
						
							|  |  |  |     sys.modules when the fresh import is complete. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     Module and package deprecation messages are suppressed during this import | 
					
						
							|  |  |  |     if *deprecated* is True. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     This function will raise ImportError if the named module cannot be | 
					
						
							|  |  |  |     imported. | 
					
						
							| 
									
										
										
										
											2021-09-14 17:31:45 -06:00
										 |  |  | 
 | 
					
						
							|  |  |  |     If "usefrozen" is False (the default) then the frozen importer is | 
					
						
							|  |  |  |     disabled (except for essential modules like importlib._bootstrap). | 
					
						
							| 
									
										
										
										
											2020-06-11 07:51:18 +08:00
										 |  |  |     """
 | 
					
						
							|  |  |  |     # NOTE: test_heapq, test_json and test_warnings include extra sanity checks | 
					
						
							|  |  |  |     # to make sure that this utility function is working as expected | 
					
						
							|  |  |  |     with _ignore_deprecated_imports(deprecated): | 
					
						
							|  |  |  |         # Keep track of modules saved for later restoration as well | 
					
						
							|  |  |  |         # as those which just need a blocking entry removed | 
					
						
							| 
									
										
										
										
											2021-09-30 19:20:39 +03:00
										 |  |  |         fresh = list(fresh) | 
					
						
							|  |  |  |         blocked = list(blocked) | 
					
						
							|  |  |  |         names = {name, *fresh, *blocked} | 
					
						
							|  |  |  |         orig_modules = _save_and_remove_modules(names) | 
					
						
							|  |  |  |         for modname in blocked: | 
					
						
							|  |  |  |             sys.modules[modname] = None | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-11 07:51:18 +08:00
										 |  |  |         try: | 
					
						
							| 
									
										
										
										
											2021-09-14 17:31:45 -06:00
										 |  |  |             with frozen_modules(usefrozen): | 
					
						
							| 
									
										
										
										
											2021-09-30 19:20:39 +03:00
										 |  |  |                 # Return None when one of the "fresh" modules can not be imported. | 
					
						
							|  |  |  |                 try: | 
					
						
							|  |  |  |                     for modname in fresh: | 
					
						
							|  |  |  |                         __import__(modname) | 
					
						
							|  |  |  |                 except ImportError: | 
					
						
							|  |  |  |                     return None | 
					
						
							|  |  |  |                 return importlib.import_module(name) | 
					
						
							| 
									
										
										
										
											2020-06-11 07:51:18 +08:00
										 |  |  |         finally: | 
					
						
							| 
									
										
										
										
											2021-09-30 19:20:39 +03:00
										 |  |  |             _save_and_remove_modules(names) | 
					
						
							|  |  |  |             sys.modules.update(orig_modules) | 
					
						
							| 
									
										
										
										
											2020-06-11 07:51:18 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | class CleanImport(object): | 
					
						
							|  |  |  |     """Context manager to force import to return a new module reference.
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     This is useful for testing module-level behaviours, such as | 
					
						
							|  |  |  |     the emission of a DeprecationWarning on import. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     Use like this: | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         with CleanImport("foo"): | 
					
						
							|  |  |  |             importlib.import_module("foo") # new reference | 
					
						
							| 
									
										
										
										
											2021-09-14 17:31:45 -06:00
										 |  |  | 
 | 
					
						
							|  |  |  |     If "usefrozen" is False (the default) then the frozen importer is | 
					
						
							|  |  |  |     disabled (except for essential modules like importlib._bootstrap). | 
					
						
							| 
									
										
										
										
											2020-06-11 07:51:18 +08:00
										 |  |  |     """
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-09-14 17:31:45 -06:00
										 |  |  |     def __init__(self, *module_names, usefrozen=False): | 
					
						
							| 
									
										
										
										
											2020-06-11 07:51:18 +08:00
										 |  |  |         self.original_modules = sys.modules.copy() | 
					
						
							|  |  |  |         for module_name in module_names: | 
					
						
							|  |  |  |             if module_name in sys.modules: | 
					
						
							|  |  |  |                 module = sys.modules[module_name] | 
					
						
							|  |  |  |                 # It is possible that module_name is just an alias for | 
					
						
							|  |  |  |                 # another module (e.g. stub for modules renamed in 3.x). | 
					
						
							|  |  |  |                 # In that case, we also need delete the real module to clear | 
					
						
							|  |  |  |                 # the import cache. | 
					
						
							|  |  |  |                 if module.__name__ != module_name: | 
					
						
							|  |  |  |                     del sys.modules[module.__name__] | 
					
						
							|  |  |  |                 del sys.modules[module_name] | 
					
						
							| 
									
										
										
										
											2021-09-14 17:31:45 -06:00
										 |  |  |         self._frozen_modules = frozen_modules(usefrozen) | 
					
						
							| 
									
										
										
										
											2020-06-11 07:51:18 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def __enter__(self): | 
					
						
							| 
									
										
										
										
											2021-09-14 17:31:45 -06:00
										 |  |  |         self._frozen_modules.__enter__() | 
					
						
							| 
									
										
										
										
											2020-06-11 07:51:18 +08:00
										 |  |  |         return self | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def __exit__(self, *ignore_exc): | 
					
						
							|  |  |  |         sys.modules.update(self.original_modules) | 
					
						
							| 
									
										
										
										
											2021-09-14 17:31:45 -06:00
										 |  |  |         self._frozen_modules.__exit__(*ignore_exc) | 
					
						
							| 
									
										
										
										
											2020-06-11 07:51:18 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | class DirsOnSysPath(object): | 
					
						
							|  |  |  |     """Context manager to temporarily add directories to sys.path.
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     This makes a copy of sys.path, appends any directories given | 
					
						
							|  |  |  |     as positional arguments, then reverts sys.path to the copied | 
					
						
							|  |  |  |     settings when the context ends. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     Note that *all* sys.path modifications in the body of the | 
					
						
							|  |  |  |     context manager, including replacement of the object, | 
					
						
							|  |  |  |     will be reverted at the end of the block. | 
					
						
							|  |  |  |     """
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def __init__(self, *paths): | 
					
						
							|  |  |  |         self.original_value = sys.path[:] | 
					
						
							|  |  |  |         self.original_object = sys.path | 
					
						
							|  |  |  |         sys.path.extend(paths) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def __enter__(self): | 
					
						
							|  |  |  |         return self | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def __exit__(self, *ignore_exc): | 
					
						
							|  |  |  |         sys.path = self.original_object | 
					
						
							|  |  |  |         sys.path[:] = self.original_value | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | def modules_setup(): | 
					
						
							|  |  |  |     return sys.modules.copy(), | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | def modules_cleanup(oldmodules): | 
					
						
							|  |  |  |     # Encoders/decoders are registered permanently within the internal | 
					
						
							|  |  |  |     # codec cache. If we destroy the corresponding modules their | 
					
						
							|  |  |  |     # globals will be set to None which will trip up the cached functions. | 
					
						
							|  |  |  |     encodings = [(k, v) for k, v in sys.modules.items() | 
					
						
							|  |  |  |                  if k.startswith('encodings.')] | 
					
						
							|  |  |  |     sys.modules.clear() | 
					
						
							|  |  |  |     sys.modules.update(encodings) | 
					
						
							|  |  |  |     # XXX: This kind of problem can affect more than just encodings. | 
					
						
							|  |  |  |     # In particular extension modules (such as _ssl) don't cope | 
					
						
							|  |  |  |     # with reloading properly. Really, test modules should be cleaning | 
					
						
							|  |  |  |     # out the test specific modules they know they added (ala test_runpy) | 
					
						
							|  |  |  |     # rather than relying on this function (as test_importhooks and test_pkg | 
					
						
							|  |  |  |     # do currently). Implicitly imported *real* modules should be left alone | 
					
						
							|  |  |  |     # (see issue 10556). | 
					
						
							|  |  |  |     sys.modules.update(oldmodules) | 
					
						
							| 
									
										
										
										
											2022-10-12 17:26:21 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-03-12 17:36:21 -04:00
										 |  |  | @contextlib.contextmanager | 
					
						
							|  |  |  | def isolated_modules(): | 
					
						
							|  |  |  |     """
 | 
					
						
							|  |  |  |     Save modules on entry and cleanup on exit. | 
					
						
							|  |  |  |     """
 | 
					
						
							|  |  |  |     (saved,) = modules_setup() | 
					
						
							|  |  |  |     try: | 
					
						
							|  |  |  |         yield | 
					
						
							|  |  |  |     finally: | 
					
						
							|  |  |  |         modules_cleanup(saved) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-10-12 17:26:21 +02:00
										 |  |  | def mock_register_at_fork(func): | 
					
						
							|  |  |  |     # bpo-30599: Mock os.register_at_fork() when importing the random module, | 
					
						
							|  |  |  |     # since this function doesn't allow to unregister callbacks and would leak | 
					
						
							|  |  |  |     # memory. | 
					
						
							|  |  |  |     from unittest import mock | 
					
						
							|  |  |  |     return mock.patch('os.register_at_fork', create=True)(func) | 
					
						
							| 
									
										
										
										
											2023-09-21 10:39:36 +03:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | @contextlib.contextmanager | 
					
						
							|  |  |  | def ready_to_import(name=None, source=""): | 
					
						
							|  |  |  |     from test.support import script_helper | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     # 1. Sets up a temporary directory and removes it afterwards | 
					
						
							|  |  |  |     # 2. Creates the module file | 
					
						
							|  |  |  |     # 3. Temporarily clears the module from sys.modules (if any) | 
					
						
							|  |  |  |     # 4. Reverts or removes the module when cleaning up | 
					
						
							|  |  |  |     name = name or "spam" | 
					
						
							|  |  |  |     with temp_dir() as tempdir: | 
					
						
							|  |  |  |         path = script_helper.make_script(tempdir, name, source) | 
					
						
							|  |  |  |         old_module = sys.modules.pop(name, None) | 
					
						
							|  |  |  |         try: | 
					
						
							|  |  |  |             sys.path.insert(0, tempdir) | 
					
						
							|  |  |  |             yield name, path | 
					
						
							|  |  |  |             sys.path.remove(tempdir) | 
					
						
							|  |  |  |         finally: | 
					
						
							|  |  |  |             if old_module is not None: | 
					
						
							|  |  |  |                 sys.modules[name] = old_module | 
					
						
							|  |  |  |             else: | 
					
						
							|  |  |  |                 sys.modules.pop(name, None) |