| 
									
										
										
										
											2009-01-30 00:22:35 +00:00
										 |  |  | from .. import abc | 
					
						
							| 
									
										
										
										
											2013-11-08 14:25:37 -05:00
										 |  |  | from .. import util | 
					
						
							| 
									
										
										
										
											2012-04-14 14:10:13 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-11-08 14:25:37 -05:00
										 |  |  | machinery = util.import_importlib('importlib.machinery') | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-04-17 00:19:56 +00:00
										 |  |  | import errno | 
					
						
							| 
									
										
										
										
											2012-04-14 14:10:13 -04:00
										 |  |  | import os | 
					
						
							| 
									
										
										
										
											2009-01-18 00:24:28 +00:00
										 |  |  | import py_compile | 
					
						
							| 
									
										
										
										
											2013-01-11 15:40:12 -05:00
										 |  |  | import stat | 
					
						
							|  |  |  | import sys | 
					
						
							|  |  |  | import tempfile | 
					
						
							| 
									
										
										
										
											2020-08-04 23:53:12 +08:00
										 |  |  | from test.support.import_helper import make_legacy_pyc | 
					
						
							| 
									
										
										
										
											2009-01-18 00:24:28 +00:00
										 |  |  | import unittest | 
					
						
							|  |  |  | import warnings | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-11-08 14:25:37 -05:00
										 |  |  | class FinderTests(abc.FinderTests): | 
					
						
							| 
									
										
										
										
											2009-01-18 00:24:28 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     """For a top-level module, it should just be found directly in the
 | 
					
						
							|  |  |  |     directory being searched. This is true for a directory with source | 
					
						
							|  |  |  |     [top-level source], bytecode [top-level bc], or both [top-level both]. | 
					
						
							|  |  |  |     There is also the possibility that it is a package [top-level package], in | 
					
						
							|  |  |  |     which case there will be a directory with the module name and an | 
					
						
							|  |  |  |     __init__.py file. If there is a directory without an __init__.py an | 
					
						
							|  |  |  |     ImportWarning is returned [empty dir]. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     For sub-modules and sub-packages, the same happens as above but only use | 
					
						
							|  |  |  |     the tail end of the name [sub module] [sub package] [sub empty]. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     When there is a conflict between a package and module having the same name | 
					
						
							|  |  |  |     in the same directory, the package wins out [package over module]. This is | 
					
						
							|  |  |  |     so that imports of modules within the package can occur rather than trigger | 
					
						
							|  |  |  |     an import error. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     When there is a package and module with the same name, always pick the | 
					
						
							|  |  |  |     package over the module [package over module]. This is so that imports from | 
					
						
							|  |  |  |     the package have the possibility of succeeding. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     """
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-08-20 13:18:15 +10:00
										 |  |  |     def get_finder(self, root): | 
					
						
							| 
									
										
										
										
											2013-11-08 14:25:37 -05:00
										 |  |  |         loader_details = [(self.machinery.SourceFileLoader, | 
					
						
							|  |  |  |                             self.machinery.SOURCE_SUFFIXES), | 
					
						
							|  |  |  |                           (self.machinery.SourcelessFileLoader, | 
					
						
							|  |  |  |                             self.machinery.BYTECODE_SUFFIXES)] | 
					
						
							|  |  |  |         return self.machinery.FileFinder(root, *loader_details) | 
					
						
							| 
									
										
										
										
											2012-08-20 13:18:15 +10:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-01-06 20:49:04 -07:00
										 |  |  |     def import_(self, root, module): | 
					
						
							|  |  |  |         finder = self.get_finder(root) | 
					
						
							|  |  |  |         return self._find(finder, module, loader_only=True) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-01-18 00:24:28 +00:00
										 |  |  |     def run_test(self, test, create=None, *, compile_=None, unlink=None): | 
					
						
							|  |  |  |         """Test the finding of 'test' with the creation of modules listed in
 | 
					
						
							|  |  |  |         'create'. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         Any names listed in 'compile_' are byte-compiled. Modules | 
					
						
							|  |  |  |         listed in 'unlink' have their source files deleted. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         """
 | 
					
						
							|  |  |  |         if create is None: | 
					
						
							|  |  |  |             create = {test} | 
					
						
							| 
									
										
										
										
											2014-05-09 14:32:57 -04:00
										 |  |  |         with util.create_modules(*create) as mapping: | 
					
						
							| 
									
										
										
										
											2009-01-18 00:24:28 +00:00
										 |  |  |             if compile_: | 
					
						
							|  |  |  |                 for name in compile_: | 
					
						
							|  |  |  |                     py_compile.compile(mapping[name]) | 
					
						
							|  |  |  |             if unlink: | 
					
						
							|  |  |  |                 for name in unlink: | 
					
						
							|  |  |  |                     os.unlink(mapping[name]) | 
					
						
							| 
									
										
										
										
											2010-04-17 00:19:56 +00:00
										 |  |  |                     try: | 
					
						
							|  |  |  |                         make_legacy_pyc(mapping[name]) | 
					
						
							|  |  |  |                     except OSError as error: | 
					
						
							|  |  |  |                         # Some tests do not set compile_=True so the source | 
					
						
							|  |  |  |                         # module will not get compiled and there will be no | 
					
						
							|  |  |  |                         # PEP 3147 pyc file to rename. | 
					
						
							|  |  |  |                         if error.errno != errno.ENOENT: | 
					
						
							|  |  |  |                             raise | 
					
						
							| 
									
										
										
										
											2009-01-18 00:24:28 +00:00
										 |  |  |             loader = self.import_(mapping['.root'], test) | 
					
						
							| 
									
										
										
										
											2009-06-30 23:06:06 +00:00
										 |  |  |             self.assertTrue(hasattr(loader, 'load_module')) | 
					
						
							| 
									
										
										
										
											2009-01-18 00:24:28 +00:00
										 |  |  |             return loader | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def test_module(self): | 
					
						
							|  |  |  |         # [top-level source] | 
					
						
							|  |  |  |         self.run_test('top_level') | 
					
						
							|  |  |  |         # [top-level bc] | 
					
						
							| 
									
										
										
										
											2010-04-17 00:19:56 +00:00
										 |  |  |         self.run_test('top_level', compile_={'top_level'}, | 
					
						
							|  |  |  |                       unlink={'top_level'}) | 
					
						
							| 
									
										
										
										
											2009-01-18 00:24:28 +00:00
										 |  |  |         # [top-level both] | 
					
						
							|  |  |  |         self.run_test('top_level', compile_={'top_level'}) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     # [top-level package] | 
					
						
							|  |  |  |     def test_package(self): | 
					
						
							|  |  |  |         # Source. | 
					
						
							|  |  |  |         self.run_test('pkg', {'pkg.__init__'}) | 
					
						
							|  |  |  |         # Bytecode. | 
					
						
							|  |  |  |         self.run_test('pkg', {'pkg.__init__'}, compile_={'pkg.__init__'}, | 
					
						
							|  |  |  |                 unlink={'pkg.__init__'}) | 
					
						
							|  |  |  |         # Both. | 
					
						
							|  |  |  |         self.run_test('pkg', {'pkg.__init__'}, compile_={'pkg.__init__'}) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     # [sub module] | 
					
						
							|  |  |  |     def test_module_in_package(self): | 
					
						
							| 
									
										
										
										
											2014-05-09 14:32:57 -04:00
										 |  |  |         with util.create_modules('pkg.__init__', 'pkg.sub') as mapping: | 
					
						
							| 
									
										
										
										
											2009-01-18 00:24:28 +00:00
										 |  |  |             pkg_dir = os.path.dirname(mapping['pkg.__init__']) | 
					
						
							|  |  |  |             loader = self.import_(pkg_dir, 'pkg.sub') | 
					
						
							| 
									
										
										
										
											2009-06-30 23:06:06 +00:00
										 |  |  |             self.assertTrue(hasattr(loader, 'load_module')) | 
					
						
							| 
									
										
										
										
											2009-01-18 00:24:28 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     # [sub package] | 
					
						
							|  |  |  |     def test_package_in_package(self): | 
					
						
							| 
									
										
										
										
											2014-05-09 14:32:57 -04:00
										 |  |  |         context = util.create_modules('pkg.__init__', 'pkg.sub.__init__') | 
					
						
							| 
									
										
										
										
											2009-01-18 00:24:28 +00:00
										 |  |  |         with context as mapping: | 
					
						
							|  |  |  |             pkg_dir = os.path.dirname(mapping['pkg.__init__']) | 
					
						
							|  |  |  |             loader = self.import_(pkg_dir, 'pkg.sub') | 
					
						
							| 
									
										
										
										
											2009-06-30 23:06:06 +00:00
										 |  |  |             self.assertTrue(hasattr(loader, 'load_module')) | 
					
						
							| 
									
										
										
										
											2009-01-18 00:24:28 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     # [package over modules] | 
					
						
							|  |  |  |     def test_package_over_module(self): | 
					
						
							|  |  |  |         name = '_temp' | 
					
						
							|  |  |  |         loader = self.run_test(name, {'{0}.__init__'.format(name), name}) | 
					
						
							| 
									
										
										
										
											2012-06-24 19:13:55 -04:00
										 |  |  |         self.assertIn('__init__', loader.get_filename(name)) | 
					
						
							| 
									
										
										
										
											2009-01-18 00:24:28 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def test_failure(self): | 
					
						
							| 
									
										
										
										
											2014-05-09 14:32:57 -04:00
										 |  |  |         with util.create_modules('blah') as mapping: | 
					
						
							| 
									
										
										
										
											2009-01-18 00:24:28 +00:00
										 |  |  |             nothing = self.import_(mapping['.root'], 'sdfsadsadf') | 
					
						
							| 
									
										
										
										
											2012-06-28 06:15:01 -04:00
										 |  |  |             self.assertIsNone(nothing) | 
					
						
							| 
									
										
										
										
											2009-01-18 00:24:28 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-07-03 22:18:47 +00:00
										 |  |  |     def test_empty_string_for_dir(self): | 
					
						
							|  |  |  |         # The empty string from sys.path means to search in the cwd. | 
					
						
							| 
									
										
										
										
											2013-11-08 14:25:37 -05:00
										 |  |  |         finder = self.machinery.FileFinder('', (self.machinery.SourceFileLoader, | 
					
						
							|  |  |  |             self.machinery.SOURCE_SUFFIXES)) | 
					
						
							| 
									
										
										
										
											2021-04-05 13:11:23 +09:00
										 |  |  |         with open('mod.py', 'w', encoding='utf-8') as file: | 
					
						
							| 
									
										
										
										
											2010-07-03 22:18:47 +00:00
										 |  |  |             file.write("# test file for importlib") | 
					
						
							|  |  |  |         try: | 
					
						
							| 
									
										
										
										
											2014-01-06 20:49:04 -07:00
										 |  |  |             loader = self._find(finder, 'mod', loader_only=True) | 
					
						
							| 
									
										
										
										
											2010-07-03 22:18:47 +00:00
										 |  |  |             self.assertTrue(hasattr(loader, 'load_module')) | 
					
						
							|  |  |  |         finally: | 
					
						
							|  |  |  |             os.unlink('mod.py') | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-02-27 18:15:42 -05:00
										 |  |  |     def test_invalidate_caches(self): | 
					
						
							|  |  |  |         # invalidate_caches() should reset the mtime. | 
					
						
							| 
									
										
										
										
											2013-11-08 14:25:37 -05:00
										 |  |  |         finder = self.machinery.FileFinder('', (self.machinery.SourceFileLoader, | 
					
						
							|  |  |  |             self.machinery.SOURCE_SUFFIXES)) | 
					
						
							| 
									
										
										
										
											2012-02-27 18:15:42 -05:00
										 |  |  |         finder._path_mtime = 42 | 
					
						
							|  |  |  |         finder.invalidate_caches() | 
					
						
							|  |  |  |         self.assertEqual(finder._path_mtime, -1) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-08-20 13:18:15 +10:00
										 |  |  |     # Regression test for http://bugs.python.org/issue14846 | 
					
						
							|  |  |  |     def test_dir_removal_handling(self): | 
					
						
							|  |  |  |         mod = 'mod' | 
					
						
							| 
									
										
										
										
											2014-05-09 14:32:57 -04:00
										 |  |  |         with util.create_modules(mod) as mapping: | 
					
						
							| 
									
										
										
										
											2012-08-20 13:18:15 +10:00
										 |  |  |             finder = self.get_finder(mapping['.root']) | 
					
						
							| 
									
										
										
										
											2014-01-06 20:49:04 -07:00
										 |  |  |             found = self._find(finder, 'mod', loader_only=True) | 
					
						
							|  |  |  |             self.assertIsNotNone(found) | 
					
						
							|  |  |  |         found = self._find(finder, 'mod', loader_only=True) | 
					
						
							|  |  |  |         self.assertIsNone(found) | 
					
						
							| 
									
										
										
										
											2009-01-18 00:24:28 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-01-11 15:40:12 -05:00
										 |  |  |     @unittest.skipUnless(sys.platform != 'win32', | 
					
						
							|  |  |  |             'os.chmod() does not support the needed arguments under Windows') | 
					
						
							|  |  |  |     def test_no_read_directory(self): | 
					
						
							|  |  |  |         # Issue #16730 | 
					
						
							|  |  |  |         tempdir = tempfile.TemporaryDirectory() | 
					
						
							|  |  |  |         original_mode = os.stat(tempdir.name).st_mode | 
					
						
							|  |  |  |         def cleanup(tempdir): | 
					
						
							|  |  |  |             """Cleanup function for the temporary directory.
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             Since we muck with the permissions, we want to set them back to | 
					
						
							|  |  |  |             their original values to make sure the directory can be properly | 
					
						
							|  |  |  |             cleaned up. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             """
 | 
					
						
							|  |  |  |             os.chmod(tempdir.name, original_mode) | 
					
						
							|  |  |  |             # If this is not explicitly called then the __del__ method is used, | 
					
						
							|  |  |  |             # but since already mucking around might as well explicitly clean | 
					
						
							|  |  |  |             # up. | 
					
						
							|  |  |  |             tempdir.__exit__(None, None, None) | 
					
						
							|  |  |  |         self.addCleanup(cleanup, tempdir) | 
					
						
							|  |  |  |         os.chmod(tempdir.name, stat.S_IWUSR | stat.S_IXUSR) | 
					
						
							|  |  |  |         finder = self.get_finder(tempdir.name) | 
					
						
							| 
									
										
										
										
											2014-01-06 20:49:04 -07:00
										 |  |  |         found = self._find(finder, 'doesnotexist') | 
					
						
							|  |  |  |         self.assertEqual(found, self.NOT_FOUND) | 
					
						
							| 
									
										
										
										
											2013-01-11 15:40:12 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def test_ignore_file(self): | 
					
						
							|  |  |  |         # If a directory got changed to a file from underneath us, then don't | 
					
						
							|  |  |  |         # worry about looking for submodules. | 
					
						
							|  |  |  |         with tempfile.NamedTemporaryFile() as file_obj: | 
					
						
							|  |  |  |             finder = self.get_finder(file_obj.name) | 
					
						
							| 
									
										
										
										
											2014-01-06 20:49:04 -07:00
										 |  |  |             found = self._find(finder, 'doesnotexist') | 
					
						
							|  |  |  |             self.assertEqual(found, self.NOT_FOUND) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-01-11 15:40:12 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-12-06 14:25:01 -05:00
										 |  |  | class FinderTestsPEP451(FinderTests): | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-01-06 20:49:04 -07:00
										 |  |  |     NOT_FOUND = None | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def _find(self, finder, name, loader_only=False): | 
					
						
							|  |  |  |         spec = finder.find_spec(name) | 
					
						
							|  |  |  |         return spec.loader if spec is not None else spec | 
					
						
							| 
									
										
										
										
											2013-12-06 14:25:01 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-05-16 11:40:40 -06:00
										 |  |  | 
 | 
					
						
							|  |  |  | (Frozen_FinderTestsPEP451, | 
					
						
							|  |  |  |  Source_FinderTestsPEP451 | 
					
						
							|  |  |  |  ) = util.test_both(FinderTestsPEP451, machinery=machinery) | 
					
						
							| 
									
										
										
										
											2013-12-06 14:25:01 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-01-06 20:49:04 -07:00
										 |  |  | class FinderTestsPEP420(FinderTests): | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     NOT_FOUND = (None, []) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def _find(self, finder, name, loader_only=False): | 
					
						
							|  |  |  |         with warnings.catch_warnings(): | 
					
						
							|  |  |  |             warnings.simplefilter("ignore", DeprecationWarning) | 
					
						
							|  |  |  |             loader_portions = finder.find_loader(name) | 
					
						
							|  |  |  |             return loader_portions[0] if loader_only else loader_portions | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-05-16 11:40:40 -06:00
										 |  |  | 
 | 
					
						
							|  |  |  | (Frozen_FinderTestsPEP420, | 
					
						
							|  |  |  |  Source_FinderTestsPEP420 | 
					
						
							|  |  |  |  ) = util.test_both(FinderTestsPEP420, machinery=machinery) | 
					
						
							| 
									
										
										
										
											2014-01-06 20:49:04 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-12-06 14:25:01 -05:00
										 |  |  | class FinderTestsPEP302(FinderTests): | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-01-06 20:49:04 -07:00
										 |  |  |     NOT_FOUND = None | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def _find(self, finder, name, loader_only=False): | 
					
						
							|  |  |  |         with warnings.catch_warnings(): | 
					
						
							|  |  |  |             warnings.simplefilter("ignore", DeprecationWarning) | 
					
						
							|  |  |  |             return finder.find_module(name) | 
					
						
							| 
									
										
										
										
											2013-12-06 14:25:01 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-01-11 15:40:12 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-05-16 11:40:40 -06:00
										 |  |  | (Frozen_FinderTestsPEP302, | 
					
						
							|  |  |  |  Source_FinderTestsPEP302 | 
					
						
							|  |  |  |  ) = util.test_both(FinderTestsPEP302, machinery=machinery) | 
					
						
							| 
									
										
										
										
											2009-01-18 00:24:28 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | if __name__ == '__main__': | 
					
						
							| 
									
										
										
										
											2013-11-08 14:25:37 -05:00
										 |  |  |     unittest.main() |