mirror of
				https://github.com/python/cpython.git
				synced 2025-10-25 15:58:57 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			704 lines
		
	
	
	
		
			26 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
			
		
		
	
	
			704 lines
		
	
	
	
		
			26 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
| from pathlib import Path
 | |
| from test.support.import_helper import unload, CleanImport
 | |
| from test.support.warnings_helper import check_warnings, ignore_warnings
 | |
| import unittest
 | |
| import sys
 | |
| import importlib
 | |
| from importlib.util import spec_from_file_location
 | |
| import pkgutil
 | |
| import os
 | |
| import os.path
 | |
| import tempfile
 | |
| import shutil
 | |
| import zipfile
 | |
| 
 | |
| from test.support.import_helper import DirsOnSysPath
 | |
| from test.support.os_helper import FakePath
 | |
| from test.test_importlib.util import uncache
 | |
| 
 | |
| # Note: pkgutil.walk_packages is currently tested in test_runpy. This is
 | |
| # a hack to get a major issue resolved for 3.3b2. Longer term, it should
 | |
| # be moved back here, perhaps by factoring out the helper code for
 | |
| # creating interesting package layouts to a separate module.
 | |
| # Issue #15348 declares this is indeed a dodgy hack ;)
 | |
| 
 | |
| class PkgutilTests(unittest.TestCase):
 | |
| 
 | |
|     def setUp(self):
 | |
|         self.dirname = tempfile.mkdtemp()
 | |
|         self.addCleanup(shutil.rmtree, self.dirname)
 | |
|         sys.path.insert(0, self.dirname)
 | |
| 
 | |
|     def tearDown(self):
 | |
|         del sys.path[0]
 | |
| 
 | |
|     def test_getdata_filesys(self):
 | |
|         pkg = 'test_getdata_filesys'
 | |
| 
 | |
|         # Include a LF and a CRLF, to test that binary data is read back
 | |
|         RESOURCE_DATA = b'Hello, world!\nSecond line\r\nThird line'
 | |
| 
 | |
|         # Make a package with some resources
 | |
|         package_dir = os.path.join(self.dirname, pkg)
 | |
|         os.mkdir(package_dir)
 | |
|         # Empty init.py
 | |
|         f = open(os.path.join(package_dir, '__init__.py'), "wb")
 | |
|         f.close()
 | |
|         # Resource files, res.txt, sub/res.txt
 | |
|         f = open(os.path.join(package_dir, 'res.txt'), "wb")
 | |
|         f.write(RESOURCE_DATA)
 | |
|         f.close()
 | |
|         os.mkdir(os.path.join(package_dir, 'sub'))
 | |
|         f = open(os.path.join(package_dir, 'sub', 'res.txt'), "wb")
 | |
|         f.write(RESOURCE_DATA)
 | |
|         f.close()
 | |
| 
 | |
|         # Check we can read the resources
 | |
|         res1 = pkgutil.get_data(pkg, 'res.txt')
 | |
|         self.assertEqual(res1, RESOURCE_DATA)
 | |
|         res2 = pkgutil.get_data(pkg, 'sub/res.txt')
 | |
|         self.assertEqual(res2, RESOURCE_DATA)
 | |
| 
 | |
|         del sys.modules[pkg]
 | |
| 
 | |
|     def test_getdata_zipfile(self):
 | |
|         zip = 'test_getdata_zipfile.zip'
 | |
|         pkg = 'test_getdata_zipfile'
 | |
| 
 | |
|         # Include a LF and a CRLF, to test that binary data is read back
 | |
|         RESOURCE_DATA = b'Hello, world!\nSecond line\r\nThird line'
 | |
| 
 | |
|         # Make a package with some resources
 | |
|         zip_file = os.path.join(self.dirname, zip)
 | |
|         z = zipfile.ZipFile(zip_file, 'w')
 | |
| 
 | |
|         # Empty init.py
 | |
|         z.writestr(pkg + '/__init__.py', "")
 | |
|         # Resource files, res.txt, sub/res.txt
 | |
|         z.writestr(pkg + '/res.txt', RESOURCE_DATA)
 | |
|         z.writestr(pkg + '/sub/res.txt', RESOURCE_DATA)
 | |
|         z.close()
 | |
| 
 | |
|         # Check we can read the resources
 | |
|         sys.path.insert(0, zip_file)
 | |
|         res1 = pkgutil.get_data(pkg, 'res.txt')
 | |
|         self.assertEqual(res1, RESOURCE_DATA)
 | |
|         res2 = pkgutil.get_data(pkg, 'sub/res.txt')
 | |
|         self.assertEqual(res2, RESOURCE_DATA)
 | |
| 
 | |
|         names = []
 | |
|         for moduleinfo in pkgutil.iter_modules([zip_file]):
 | |
|             self.assertIsInstance(moduleinfo, pkgutil.ModuleInfo)
 | |
|             names.append(moduleinfo.name)
 | |
|         self.assertEqual(names, ['test_getdata_zipfile'])
 | |
| 
 | |
|         del sys.path[0]
 | |
| 
 | |
|         del sys.modules[pkg]
 | |
| 
 | |
|     def test_issue44061_iter_modules(self):
 | |
|         #see: issue44061
 | |
|         zip = 'test_getdata_zipfile.zip'
 | |
|         pkg = 'test_getdata_zipfile'
 | |
| 
 | |
|         # Include a LF and a CRLF, to test that binary data is read back
 | |
|         RESOURCE_DATA = b'Hello, world!\nSecond line\r\nThird line'
 | |
| 
 | |
|         # Make a package with some resources
 | |
|         zip_file = os.path.join(self.dirname, zip)
 | |
|         z = zipfile.ZipFile(zip_file, 'w')
 | |
| 
 | |
|         # Empty init.py
 | |
|         z.writestr(pkg + '/__init__.py', "")
 | |
|         # Resource files, res.txt
 | |
|         z.writestr(pkg + '/res.txt', RESOURCE_DATA)
 | |
|         z.close()
 | |
| 
 | |
|         # Check we can read the resources
 | |
|         sys.path.insert(0, zip_file)
 | |
|         try:
 | |
|             res = pkgutil.get_data(pkg, 'res.txt')
 | |
|             self.assertEqual(res, RESOURCE_DATA)
 | |
| 
 | |
|             # make sure iter_modules accepts Path objects
 | |
|             names = []
 | |
|             for moduleinfo in pkgutil.iter_modules([FakePath(zip_file)]):
 | |
|                 self.assertIsInstance(moduleinfo, pkgutil.ModuleInfo)
 | |
|                 names.append(moduleinfo.name)
 | |
|             self.assertEqual(names, [pkg])
 | |
|         finally:
 | |
|             del sys.path[0]
 | |
|             sys.modules.pop(pkg, None)
 | |
| 
 | |
|         # assert path must be None or list of paths
 | |
|         expected_msg = "path must be None or list of paths to look for modules in"
 | |
|         with self.assertRaisesRegex(ValueError, expected_msg):
 | |
|             list(pkgutil.iter_modules("invalid_path"))
 | |
| 
 | |
|     def test_unreadable_dir_on_syspath(self):
 | |
|         # issue7367 - walk_packages failed if unreadable dir on sys.path
 | |
|         package_name = "unreadable_package"
 | |
|         d = os.path.join(self.dirname, package_name)
 | |
|         # this does not appear to create an unreadable dir on Windows
 | |
|         #   but the test should not fail anyway
 | |
|         os.mkdir(d, 0)
 | |
|         self.addCleanup(os.rmdir, d)
 | |
|         for t in pkgutil.walk_packages(path=[self.dirname]):
 | |
|             self.fail("unexpected package found")
 | |
| 
 | |
|     def test_walkpackages_filesys(self):
 | |
|         pkg1 = 'test_walkpackages_filesys'
 | |
|         pkg1_dir = os.path.join(self.dirname, pkg1)
 | |
|         os.mkdir(pkg1_dir)
 | |
|         f = open(os.path.join(pkg1_dir, '__init__.py'), "wb")
 | |
|         f.close()
 | |
|         os.mkdir(os.path.join(pkg1_dir, 'sub'))
 | |
|         f = open(os.path.join(pkg1_dir, 'sub', '__init__.py'), "wb")
 | |
|         f.close()
 | |
|         f = open(os.path.join(pkg1_dir, 'sub', 'mod.py'), "wb")
 | |
|         f.close()
 | |
| 
 | |
|         # Now, to juice it up, let's add the opposite packages, too.
 | |
|         pkg2 = 'sub'
 | |
|         pkg2_dir = os.path.join(self.dirname, pkg2)
 | |
|         os.mkdir(pkg2_dir)
 | |
|         f = open(os.path.join(pkg2_dir, '__init__.py'), "wb")
 | |
|         f.close()
 | |
|         os.mkdir(os.path.join(pkg2_dir, 'test_walkpackages_filesys'))
 | |
|         f = open(os.path.join(pkg2_dir, 'test_walkpackages_filesys', '__init__.py'), "wb")
 | |
|         f.close()
 | |
|         f = open(os.path.join(pkg2_dir, 'test_walkpackages_filesys', 'mod.py'), "wb")
 | |
|         f.close()
 | |
| 
 | |
|         expected = [
 | |
|             'sub',
 | |
|             'sub.test_walkpackages_filesys',
 | |
|             'sub.test_walkpackages_filesys.mod',
 | |
|             'test_walkpackages_filesys',
 | |
|             'test_walkpackages_filesys.sub',
 | |
|             'test_walkpackages_filesys.sub.mod',
 | |
|         ]
 | |
|         actual= [e[1] for e in pkgutil.walk_packages([self.dirname])]
 | |
|         self.assertEqual(actual, expected)
 | |
| 
 | |
|         for pkg in expected:
 | |
|             if pkg.endswith('mod'):
 | |
|                 continue
 | |
|             del sys.modules[pkg]
 | |
| 
 | |
|     def test_walkpackages_zipfile(self):
 | |
|         """Tests the same as test_walkpackages_filesys, only with a zip file."""
 | |
| 
 | |
|         zip = 'test_walkpackages_zipfile.zip'
 | |
|         pkg1 = 'test_walkpackages_zipfile'
 | |
|         pkg2 = 'sub'
 | |
| 
 | |
|         zip_file = os.path.join(self.dirname, zip)
 | |
|         z = zipfile.ZipFile(zip_file, 'w')
 | |
|         z.writestr(pkg2 + '/__init__.py', "")
 | |
|         z.writestr(pkg2 + '/' + pkg1 + '/__init__.py', "")
 | |
|         z.writestr(pkg2 + '/' + pkg1 + '/mod.py', "")
 | |
|         z.writestr(pkg1 + '/__init__.py', "")
 | |
|         z.writestr(pkg1 + '/' + pkg2 + '/__init__.py', "")
 | |
|         z.writestr(pkg1 + '/' + pkg2 + '/mod.py', "")
 | |
|         z.close()
 | |
| 
 | |
|         sys.path.insert(0, zip_file)
 | |
|         expected = [
 | |
|             'sub',
 | |
|             'sub.test_walkpackages_zipfile',
 | |
|             'sub.test_walkpackages_zipfile.mod',
 | |
|             'test_walkpackages_zipfile',
 | |
|             'test_walkpackages_zipfile.sub',
 | |
|             'test_walkpackages_zipfile.sub.mod',
 | |
|         ]
 | |
|         actual= [e[1] for e in pkgutil.walk_packages([zip_file])]
 | |
|         self.assertEqual(actual, expected)
 | |
|         del sys.path[0]
 | |
| 
 | |
|         for pkg in expected:
 | |
|             if pkg.endswith('mod'):
 | |
|                 continue
 | |
|             del sys.modules[pkg]
 | |
| 
 | |
|     def test_walk_packages_raises_on_string_or_bytes_input(self):
 | |
| 
 | |
|         str_input = 'test_dir'
 | |
|         with self.assertRaises((TypeError, ValueError)):
 | |
|             list(pkgutil.walk_packages(str_input))
 | |
| 
 | |
|         bytes_input = b'test_dir'
 | |
|         with self.assertRaises((TypeError, ValueError)):
 | |
|             list(pkgutil.walk_packages(bytes_input))
 | |
| 
 | |
|     def test_name_resolution(self):
 | |
|         import logging
 | |
|         import logging.handlers
 | |
| 
 | |
|         success_cases = (
 | |
|             ('os', os),
 | |
|             ('os.path', os.path),
 | |
|             ('os.path:pathsep', os.path.pathsep),
 | |
|             ('logging', logging),
 | |
|             ('logging:', logging),
 | |
|             ('logging.handlers', logging.handlers),
 | |
|             ('logging.handlers:', logging.handlers),
 | |
|             ('logging.handlers:SysLogHandler', logging.handlers.SysLogHandler),
 | |
|             ('logging.handlers.SysLogHandler', logging.handlers.SysLogHandler),
 | |
|             ('logging.handlers:SysLogHandler.LOG_ALERT',
 | |
|                 logging.handlers.SysLogHandler.LOG_ALERT),
 | |
|             ('logging.handlers.SysLogHandler.LOG_ALERT',
 | |
|                 logging.handlers.SysLogHandler.LOG_ALERT),
 | |
|             ('builtins.int', int),
 | |
|             ('builtins:int', int),
 | |
|             ('builtins.int.from_bytes', int.from_bytes),
 | |
|             ('builtins:int.from_bytes', int.from_bytes),
 | |
|             ('builtins.ZeroDivisionError', ZeroDivisionError),
 | |
|             ('builtins:ZeroDivisionError', ZeroDivisionError),
 | |
|             ('os:path', os.path),
 | |
|         )
 | |
| 
 | |
|         failure_cases = (
 | |
|             (None, TypeError),
 | |
|             (1, TypeError),
 | |
|             (2.0, TypeError),
 | |
|             (True, TypeError),
 | |
|             ('', ValueError),
 | |
|             ('?abc', ValueError),
 | |
|             ('abc/foo', ValueError),
 | |
|             ('foo', ImportError),
 | |
|             ('os.foo', AttributeError),
 | |
|             ('os.foo:', ImportError),
 | |
|             ('os.pth:pathsep', ImportError),
 | |
|             ('logging.handlers:NoSuchHandler', AttributeError),
 | |
|             ('logging.handlers:SysLogHandler.NO_SUCH_VALUE', AttributeError),
 | |
|             ('logging.handlers.SysLogHandler.NO_SUCH_VALUE', AttributeError),
 | |
|             ('ZeroDivisionError', ImportError),
 | |
|             ('os.path.9abc', ValueError),
 | |
|             ('9abc', ValueError),
 | |
|         )
 | |
| 
 | |
|         # add some Unicode package names to the mix.
 | |
| 
 | |
|         unicode_words = ('\u0935\u092e\u0938',
 | |
|                          '\xe9', '\xc8',
 | |
|                          '\uc548\ub155\ud558\uc138\uc694',
 | |
|                          '\u3055\u3088\u306a\u3089',
 | |
|                          '\u3042\u308a\u304c\u3068\u3046',
 | |
|                          '\u0425\u043e\u0440\u043e\u0448\u043e',
 | |
|                          '\u0441\u043f\u0430\u0441\u0438\u0431\u043e',
 | |
|                          '\u73b0\u4ee3\u6c49\u8bed\u5e38\u7528\u5b57\u8868')
 | |
| 
 | |
|         for uw in unicode_words:
 | |
|             d = os.path.join(self.dirname, uw)
 | |
|             try:
 | |
|                 os.makedirs(d, exist_ok=True)
 | |
|             except  UnicodeEncodeError:
 | |
|                 # When filesystem encoding cannot encode uw: skip this test
 | |
|                 continue
 | |
|             # make an empty __init__.py file
 | |
|             f = os.path.join(d, '__init__.py')
 | |
|             with open(f, 'w') as f:
 | |
|                 f.write('')
 | |
|                 f.flush()
 | |
|             # now import the package we just created; clearing the caches is
 | |
|             # needed, otherwise the newly created package isn't found
 | |
|             importlib.invalidate_caches()
 | |
|             mod = importlib.import_module(uw)
 | |
|             success_cases += (uw, mod),
 | |
|             if len(uw) > 1:
 | |
|                 failure_cases += (uw[:-1], ImportError),
 | |
| 
 | |
|         # add an example with a Unicode digit at the start
 | |
|         failure_cases += ('\u0966\u0935\u092e\u0938', ValueError),
 | |
| 
 | |
|         for s, expected in success_cases:
 | |
|             with self.subTest(s=s):
 | |
|                 o = pkgutil.resolve_name(s)
 | |
|                 self.assertEqual(o, expected)
 | |
| 
 | |
|         for s, exc in failure_cases:
 | |
|             with self.subTest(s=s):
 | |
|                 with self.assertRaises(exc):
 | |
|                     pkgutil.resolve_name(s)
 | |
| 
 | |
|     def test_name_resolution_import_rebinding(self):
 | |
|         # The same data is also used for testing import in test_import and
 | |
|         # mock.patch in test_unittest.
 | |
|         path = os.path.join(os.path.dirname(__file__), 'test_import', 'data')
 | |
|         with uncache('package3', 'package3.submodule'), DirsOnSysPath(path):
 | |
|             self.assertEqual(pkgutil.resolve_name('package3.submodule.attr'), 'submodule')
 | |
|         with uncache('package3', 'package3.submodule'), DirsOnSysPath(path):
 | |
|             self.assertEqual(pkgutil.resolve_name('package3.submodule:attr'), 'submodule')
 | |
|         with uncache('package3', 'package3.submodule'), DirsOnSysPath(path):
 | |
|             self.assertEqual(pkgutil.resolve_name('package3:submodule.attr'), 'rebound')
 | |
|             self.assertEqual(pkgutil.resolve_name('package3.submodule.attr'), 'submodule')
 | |
|             self.assertEqual(pkgutil.resolve_name('package3:submodule.attr'), 'rebound')
 | |
|         with uncache('package3', 'package3.submodule'), DirsOnSysPath(path):
 | |
|             self.assertEqual(pkgutil.resolve_name('package3:submodule.attr'), 'rebound')
 | |
|             self.assertEqual(pkgutil.resolve_name('package3.submodule:attr'), 'submodule')
 | |
|             self.assertEqual(pkgutil.resolve_name('package3:submodule.attr'), 'rebound')
 | |
| 
 | |
|     def test_name_resolution_import_rebinding2(self):
 | |
|         path = os.path.join(os.path.dirname(__file__), 'test_import', 'data')
 | |
|         with uncache('package4', 'package4.submodule'), DirsOnSysPath(path):
 | |
|             self.assertEqual(pkgutil.resolve_name('package4.submodule.attr'), 'submodule')
 | |
|         with uncache('package4', 'package4.submodule'), DirsOnSysPath(path):
 | |
|             self.assertEqual(pkgutil.resolve_name('package4.submodule:attr'), 'submodule')
 | |
|         with uncache('package4', 'package4.submodule'), DirsOnSysPath(path):
 | |
|             self.assertEqual(pkgutil.resolve_name('package4:submodule.attr'), 'origin')
 | |
|             self.assertEqual(pkgutil.resolve_name('package4.submodule.attr'), 'submodule')
 | |
|             self.assertEqual(pkgutil.resolve_name('package4:submodule.attr'), 'submodule')
 | |
|         with uncache('package4', 'package4.submodule'), DirsOnSysPath(path):
 | |
|             self.assertEqual(pkgutil.resolve_name('package4:submodule.attr'), 'origin')
 | |
|             self.assertEqual(pkgutil.resolve_name('package4.submodule:attr'), 'submodule')
 | |
|             self.assertEqual(pkgutil.resolve_name('package4:submodule.attr'), 'submodule')
 | |
| 
 | |
| 
 | |
| class PkgutilPEP302Tests(unittest.TestCase):
 | |
| 
 | |
|     class MyTestLoader(object):
 | |
|         def create_module(self, spec):
 | |
|             return None
 | |
| 
 | |
|         def exec_module(self, mod):
 | |
|             # Count how many times the module is reloaded
 | |
|             mod.__dict__['loads'] = mod.__dict__.get('loads', 0) + 1
 | |
| 
 | |
|         def get_data(self, path):
 | |
|             return "Hello, world!"
 | |
| 
 | |
|     class MyTestImporter(object):
 | |
|         def find_spec(self, fullname, path=None, target=None):
 | |
|             loader = PkgutilPEP302Tests.MyTestLoader()
 | |
|             return spec_from_file_location(fullname,
 | |
|                                            '<%s>' % loader.__class__.__name__,
 | |
|                                            loader=loader,
 | |
|                                            submodule_search_locations=[])
 | |
| 
 | |
|     def setUp(self):
 | |
|         sys.meta_path.insert(0, self.MyTestImporter())
 | |
| 
 | |
|     def tearDown(self):
 | |
|         del sys.meta_path[0]
 | |
| 
 | |
|     def test_getdata_pep302(self):
 | |
|         # Use a dummy finder/loader
 | |
|         self.assertEqual(pkgutil.get_data('foo', 'dummy'), "Hello, world!")
 | |
|         del sys.modules['foo']
 | |
| 
 | |
|     def test_alreadyloaded(self):
 | |
|         # Ensure that get_data works without reloading - the "loads" module
 | |
|         # variable in the example loader should count how many times a reload
 | |
|         # occurs.
 | |
|         import foo
 | |
|         self.assertEqual(foo.loads, 1)
 | |
|         self.assertEqual(pkgutil.get_data('foo', 'dummy'), "Hello, world!")
 | |
|         self.assertEqual(foo.loads, 1)
 | |
|         del sys.modules['foo']
 | |
| 
 | |
| 
 | |
| # These tests, especially the setup and cleanup, are hideous. They
 | |
| # need to be cleaned up once issue 14715 is addressed.
 | |
| class ExtendPathTests(unittest.TestCase):
 | |
|     def create_init(self, pkgname):
 | |
|         dirname = tempfile.mkdtemp()
 | |
|         sys.path.insert(0, dirname)
 | |
| 
 | |
|         pkgdir = os.path.join(dirname, pkgname)
 | |
|         os.mkdir(pkgdir)
 | |
|         with open(os.path.join(pkgdir, '__init__.py'), 'w') as fl:
 | |
|             fl.write('from pkgutil import extend_path\n__path__ = extend_path(__path__, __name__)\n')
 | |
| 
 | |
|         return dirname
 | |
| 
 | |
|     def create_submodule(self, dirname, pkgname, submodule_name, value):
 | |
|         module_name = os.path.join(dirname, pkgname, submodule_name + '.py')
 | |
|         with open(module_name, 'w') as fl:
 | |
|             print('value={}'.format(value), file=fl)
 | |
| 
 | |
|     def test_simple(self):
 | |
|         pkgname = 'foo'
 | |
|         dirname_0 = self.create_init(pkgname)
 | |
|         dirname_1 = self.create_init(pkgname)
 | |
|         self.create_submodule(dirname_0, pkgname, 'bar', 0)
 | |
|         self.create_submodule(dirname_1, pkgname, 'baz', 1)
 | |
|         import foo.bar
 | |
|         import foo.baz
 | |
|         # Ensure we read the expected values
 | |
|         self.assertEqual(foo.bar.value, 0)
 | |
|         self.assertEqual(foo.baz.value, 1)
 | |
| 
 | |
|         # Ensure the path is set up correctly
 | |
|         self.assertEqual(sorted(foo.__path__),
 | |
|                          sorted([os.path.join(dirname_0, pkgname),
 | |
|                                  os.path.join(dirname_1, pkgname)]))
 | |
| 
 | |
|         # Cleanup
 | |
|         shutil.rmtree(dirname_0)
 | |
|         shutil.rmtree(dirname_1)
 | |
|         del sys.path[0]
 | |
|         del sys.path[0]
 | |
|         del sys.modules['foo']
 | |
|         del sys.modules['foo.bar']
 | |
|         del sys.modules['foo.baz']
 | |
| 
 | |
| 
 | |
|     # Another awful testing hack to be cleaned up once the test_runpy
 | |
|     # helpers are factored out to a common location
 | |
|     def test_iter_importers(self):
 | |
|         iter_importers = pkgutil.iter_importers
 | |
|         get_importer = pkgutil.get_importer
 | |
| 
 | |
|         pkgname = 'spam'
 | |
|         modname = 'eggs'
 | |
|         dirname = self.create_init(pkgname)
 | |
|         pathitem = os.path.join(dirname, pkgname)
 | |
|         fullname = '{}.{}'.format(pkgname, modname)
 | |
|         sys.modules.pop(fullname, None)
 | |
|         sys.modules.pop(pkgname, None)
 | |
|         try:
 | |
|             self.create_submodule(dirname, pkgname, modname, 0)
 | |
| 
 | |
|             importlib.import_module(fullname)
 | |
| 
 | |
|             importers = list(iter_importers(fullname))
 | |
|             expected_importer = get_importer(pathitem)
 | |
|             for finder in importers:
 | |
|                 spec = finder.find_spec(fullname)
 | |
|                 loader = spec.loader
 | |
|                 try:
 | |
|                     loader = loader.loader
 | |
|                 except AttributeError:
 | |
|                     # For now we still allow raw loaders from
 | |
|                     # find_module().
 | |
|                     pass
 | |
|                 self.assertIsInstance(finder, importlib.machinery.FileFinder)
 | |
|                 self.assertEqual(finder, expected_importer)
 | |
|                 self.assertIsInstance(loader,
 | |
|                                       importlib.machinery.SourceFileLoader)
 | |
|                 self.assertIsNone(finder.find_spec(pkgname))
 | |
| 
 | |
|             with self.assertRaises(ImportError):
 | |
|                 list(iter_importers('invalid.module'))
 | |
| 
 | |
|             with self.assertRaises(ImportError):
 | |
|                 list(iter_importers('.spam'))
 | |
|         finally:
 | |
|             shutil.rmtree(dirname)
 | |
|             del sys.path[0]
 | |
|             try:
 | |
|                 del sys.modules['spam']
 | |
|                 del sys.modules['spam.eggs']
 | |
|             except KeyError:
 | |
|                 pass
 | |
| 
 | |
| 
 | |
|     def test_mixed_namespace(self):
 | |
|         pkgname = 'foo'
 | |
|         dirname_0 = self.create_init(pkgname)
 | |
|         dirname_1 = self.create_init(pkgname)
 | |
|         self.create_submodule(dirname_0, pkgname, 'bar', 0)
 | |
|         # Turn this into a PEP 420 namespace package
 | |
|         os.unlink(os.path.join(dirname_0, pkgname, '__init__.py'))
 | |
|         self.create_submodule(dirname_1, pkgname, 'baz', 1)
 | |
|         import foo.bar
 | |
|         import foo.baz
 | |
|         # Ensure we read the expected values
 | |
|         self.assertEqual(foo.bar.value, 0)
 | |
|         self.assertEqual(foo.baz.value, 1)
 | |
| 
 | |
|         # Ensure the path is set up correctly
 | |
|         self.assertEqual(sorted(foo.__path__),
 | |
|                          sorted([os.path.join(dirname_0, pkgname),
 | |
|                                  os.path.join(dirname_1, pkgname)]))
 | |
| 
 | |
|         # Cleanup
 | |
|         shutil.rmtree(dirname_0)
 | |
|         shutil.rmtree(dirname_1)
 | |
|         del sys.path[0]
 | |
|         del sys.path[0]
 | |
|         del sys.modules['foo']
 | |
|         del sys.modules['foo.bar']
 | |
|         del sys.modules['foo.baz']
 | |
| 
 | |
| 
 | |
|     def test_extend_path_argument_types(self):
 | |
|         pkgname = 'foo'
 | |
|         dirname_0 = self.create_init(pkgname)
 | |
| 
 | |
|         # If the input path is not a list it is returned unchanged
 | |
|         self.assertEqual('notalist', pkgutil.extend_path('notalist', 'foo'))
 | |
|         self.assertEqual(('not', 'a', 'list'), pkgutil.extend_path(('not', 'a', 'list'), 'foo'))
 | |
|         self.assertEqual(123, pkgutil.extend_path(123, 'foo'))
 | |
|         self.assertEqual(None, pkgutil.extend_path(None, 'foo'))
 | |
| 
 | |
|         # Cleanup
 | |
|         shutil.rmtree(dirname_0)
 | |
|         del sys.path[0]
 | |
| 
 | |
| 
 | |
|     def test_extend_path_pkg_files(self):
 | |
|         pkgname = 'foo'
 | |
|         dirname_0 = self.create_init(pkgname)
 | |
| 
 | |
|         with open(os.path.join(dirname_0, 'bar.pkg'), 'w') as pkg_file:
 | |
|             pkg_file.write('\n'.join([
 | |
|                 'baz',
 | |
|                 '/foo/bar/baz',
 | |
|                 '',
 | |
|                 '#comment'
 | |
|             ]))
 | |
| 
 | |
|         extended_paths = pkgutil.extend_path(sys.path, 'bar')
 | |
| 
 | |
|         self.assertEqual(extended_paths[:-2], sys.path)
 | |
|         self.assertEqual(extended_paths[-2], 'baz')
 | |
|         self.assertEqual(extended_paths[-1], '/foo/bar/baz')
 | |
| 
 | |
|         # Cleanup
 | |
|         shutil.rmtree(dirname_0)
 | |
|         del sys.path[0]
 | |
| 
 | |
| 
 | |
| class NestedNamespacePackageTest(unittest.TestCase):
 | |
| 
 | |
|     def setUp(self):
 | |
|         self.basedir = tempfile.mkdtemp()
 | |
|         self.old_path = sys.path[:]
 | |
| 
 | |
|     def tearDown(self):
 | |
|         sys.path[:] = self.old_path
 | |
|         shutil.rmtree(self.basedir)
 | |
| 
 | |
|     def create_module(self, name, contents):
 | |
|         base, final = name.rsplit('.', 1)
 | |
|         base_path = os.path.join(self.basedir, base.replace('.', os.path.sep))
 | |
|         os.makedirs(base_path, exist_ok=True)
 | |
|         with open(os.path.join(base_path, final + ".py"), 'w') as f:
 | |
|             f.write(contents)
 | |
| 
 | |
|     def test_nested(self):
 | |
|         pkgutil_boilerplate = (
 | |
|             'import pkgutil; '
 | |
|             '__path__ = pkgutil.extend_path(__path__, __name__)')
 | |
|         self.create_module('a.pkg.__init__', pkgutil_boilerplate)
 | |
|         self.create_module('b.pkg.__init__', pkgutil_boilerplate)
 | |
|         self.create_module('a.pkg.subpkg.__init__', pkgutil_boilerplate)
 | |
|         self.create_module('b.pkg.subpkg.__init__', pkgutil_boilerplate)
 | |
|         self.create_module('a.pkg.subpkg.c', 'c = 1')
 | |
|         self.create_module('b.pkg.subpkg.d', 'd = 2')
 | |
|         sys.path.insert(0, os.path.join(self.basedir, 'a'))
 | |
|         sys.path.insert(0, os.path.join(self.basedir, 'b'))
 | |
|         import pkg
 | |
|         self.addCleanup(unload, 'pkg')
 | |
|         self.assertEqual(len(pkg.__path__), 2)
 | |
|         import pkg.subpkg
 | |
|         self.addCleanup(unload, 'pkg.subpkg')
 | |
|         self.assertEqual(len(pkg.subpkg.__path__), 2)
 | |
|         from pkg.subpkg.c import c
 | |
|         from pkg.subpkg.d import d
 | |
|         self.assertEqual(c, 1)
 | |
|         self.assertEqual(d, 2)
 | |
| 
 | |
| 
 | |
| class ImportlibMigrationTests(unittest.TestCase):
 | |
|     # With full PEP 302 support in the standard import machinery, the
 | |
|     # PEP 302 emulation in this module is in the process of being
 | |
|     # deprecated in favour of importlib proper
 | |
| 
 | |
|     @unittest.skipIf(__name__ == '__main__', 'not compatible with __main__')
 | |
|     @ignore_warnings(category=DeprecationWarning)
 | |
|     def test_get_loader_handles_missing_loader_attribute(self):
 | |
|         global __loader__
 | |
|         this_loader = __loader__
 | |
|         del __loader__
 | |
|         try:
 | |
|             self.assertIsNotNone(pkgutil.get_loader(__name__))
 | |
|         finally:
 | |
|             __loader__ = this_loader
 | |
| 
 | |
|     @ignore_warnings(category=DeprecationWarning)
 | |
|     def test_get_loader_handles_missing_spec_attribute(self):
 | |
|         name = 'spam'
 | |
|         mod = type(sys)(name)
 | |
|         del mod.__spec__
 | |
|         with CleanImport(name):
 | |
|             try:
 | |
|                 sys.modules[name] = mod
 | |
|                 loader = pkgutil.get_loader(name)
 | |
|             finally:
 | |
|                 sys.modules.pop(name, None)
 | |
|         self.assertIsNone(loader)
 | |
| 
 | |
|     @ignore_warnings(category=DeprecationWarning)
 | |
|     def test_get_loader_handles_spec_attribute_none(self):
 | |
|         name = 'spam'
 | |
|         mod = type(sys)(name)
 | |
|         mod.__spec__ = None
 | |
|         with CleanImport(name):
 | |
|             try:
 | |
|                 sys.modules[name] = mod
 | |
|                 loader = pkgutil.get_loader(name)
 | |
|             finally:
 | |
|                 sys.modules.pop(name, None)
 | |
|         self.assertIsNone(loader)
 | |
| 
 | |
|     @ignore_warnings(category=DeprecationWarning)
 | |
|     def test_get_loader_None_in_sys_modules(self):
 | |
|         name = 'totally bogus'
 | |
|         sys.modules[name] = None
 | |
|         try:
 | |
|             loader = pkgutil.get_loader(name)
 | |
|         finally:
 | |
|             del sys.modules[name]
 | |
|         self.assertIsNone(loader)
 | |
| 
 | |
|     def test_get_loader_is_deprecated(self):
 | |
|         with check_warnings(
 | |
|             (r".*\bpkgutil.get_loader\b.*", DeprecationWarning),
 | |
|         ):
 | |
|             res = pkgutil.get_loader("sys")
 | |
|         self.assertIsNotNone(res)
 | |
| 
 | |
|     def test_find_loader_is_deprecated(self):
 | |
|         with check_warnings(
 | |
|             (r".*\bpkgutil.find_loader\b.*", DeprecationWarning),
 | |
|         ):
 | |
|             res = pkgutil.find_loader("sys")
 | |
|         self.assertIsNotNone(res)
 | |
| 
 | |
|     @ignore_warnings(category=DeprecationWarning)
 | |
|     def test_find_loader_missing_module(self):
 | |
|         name = 'totally bogus'
 | |
|         loader = pkgutil.find_loader(name)
 | |
|         self.assertIsNone(loader)
 | |
| 
 | |
|     def test_get_importer_avoids_emulation(self):
 | |
|         # We use an illegal path so *none* of the path hooks should fire
 | |
|         with check_warnings() as w:
 | |
|             self.assertIsNone(pkgutil.get_importer("*??"))
 | |
|             self.assertEqual(len(w.warnings), 0)
 | |
| 
 | |
|     def test_issue44061(self):
 | |
|         try:
 | |
|             pkgutil.get_importer(Path("/home"))
 | |
|         except AttributeError:
 | |
|             self.fail("Unexpected AttributeError when calling get_importer")
 | |
| 
 | |
|     def test_iter_importers_avoids_emulation(self):
 | |
|         with check_warnings() as w:
 | |
|             for importer in pkgutil.iter_importers(): pass
 | |
|             self.assertEqual(len(w.warnings), 0)
 | |
| 
 | |
| 
 | |
| def tearDownModule():
 | |
|     # this is necessary if test is run repeated (like when finding leaks)
 | |
|     import zipimport
 | |
|     import importlib
 | |
|     zipimport._zip_directory_cache.clear()
 | |
|     importlib.invalidate_caches()
 | |
| 
 | |
| 
 | |
| if __name__ == '__main__':
 | |
|     unittest.main()
 | 
