blob: 53456c2f7659e2e05c2bd393e37cb8ec3b3ccfcc [file] [log] [blame]
Eric Snow658af312014-04-19 00:13:23 -06001from test.support import run_unittest, unload, check_warnings, CleanImport
Christian Heimesdae2a892008-04-19 00:55:37 +00002import unittest
3import sys
Nick Coghlanc4e0d982013-04-14 22:30:42 +10004import importlib
Eric Snow37148b22014-01-04 15:09:53 -07005from importlib.util import spec_from_file_location
Christian Heimesdae2a892008-04-19 00:55:37 +00006import pkgutil
7import os
8import os.path
9import tempfile
10import shutil
11import zipfile
12
Nick Coghlan8ecf5042012-07-15 21:19:18 +100013# Note: pkgutil.walk_packages is currently tested in test_runpy. This is
14# a hack to get a major issue resolved for 3.3b2. Longer term, it should
15# be moved back here, perhaps by factoring out the helper code for
16# creating interesting package layouts to a separate module.
17# Issue #15348 declares this is indeed a dodgy hack ;)
Christian Heimesdae2a892008-04-19 00:55:37 +000018
19class PkgutilTests(unittest.TestCase):
20
21 def setUp(self):
22 self.dirname = tempfile.mkdtemp()
Ned Deily7010a072011-10-07 12:01:40 -070023 self.addCleanup(shutil.rmtree, self.dirname)
Christian Heimesdae2a892008-04-19 00:55:37 +000024 sys.path.insert(0, self.dirname)
25
26 def tearDown(self):
27 del sys.path[0]
Christian Heimesdae2a892008-04-19 00:55:37 +000028
29 def test_getdata_filesys(self):
30 pkg = 'test_getdata_filesys'
31
32 # Include a LF and a CRLF, to test that binary data is read back
33 RESOURCE_DATA = b'Hello, world!\nSecond line\r\nThird line'
34
35 # Make a package with some resources
36 package_dir = os.path.join(self.dirname, pkg)
37 os.mkdir(package_dir)
38 # Empty init.py
39 f = open(os.path.join(package_dir, '__init__.py'), "wb")
40 f.close()
41 # Resource files, res.txt, sub/res.txt
42 f = open(os.path.join(package_dir, 'res.txt'), "wb")
43 f.write(RESOURCE_DATA)
44 f.close()
45 os.mkdir(os.path.join(package_dir, 'sub'))
46 f = open(os.path.join(package_dir, 'sub', 'res.txt'), "wb")
47 f.write(RESOURCE_DATA)
48 f.close()
49
50 # Check we can read the resources
51 res1 = pkgutil.get_data(pkg, 'res.txt')
52 self.assertEqual(res1, RESOURCE_DATA)
53 res2 = pkgutil.get_data(pkg, 'sub/res.txt')
54 self.assertEqual(res2, RESOURCE_DATA)
55
56 del sys.modules[pkg]
57
58 def test_getdata_zipfile(self):
59 zip = 'test_getdata_zipfile.zip'
60 pkg = 'test_getdata_zipfile'
61
62 # Include a LF and a CRLF, to test that binary data is read back
63 RESOURCE_DATA = b'Hello, world!\nSecond line\r\nThird line'
64
65 # Make a package with some resources
66 zip_file = os.path.join(self.dirname, zip)
67 z = zipfile.ZipFile(zip_file, 'w')
68
69 # Empty init.py
70 z.writestr(pkg + '/__init__.py', "")
71 # Resource files, res.txt, sub/res.txt
72 z.writestr(pkg + '/res.txt', RESOURCE_DATA)
73 z.writestr(pkg + '/sub/res.txt', RESOURCE_DATA)
74 z.close()
75
76 # Check we can read the resources
77 sys.path.insert(0, zip_file)
78 res1 = pkgutil.get_data(pkg, 'res.txt')
79 self.assertEqual(res1, RESOURCE_DATA)
80 res2 = pkgutil.get_data(pkg, 'sub/res.txt')
81 self.assertEqual(res2, RESOURCE_DATA)
Alexandre Vassalotti515a74f2009-07-05 06:42:44 +000082
83 names = []
Eric Snowd5f92232016-09-07 18:37:17 -070084 for moduleinfo in pkgutil.iter_modules([zip_file]):
85 self.assertIsInstance(moduleinfo, pkgutil.ModuleInfo)
86 names.append(moduleinfo.name)
Alexandre Vassalotti515a74f2009-07-05 06:42:44 +000087 self.assertEqual(names, ['test_getdata_zipfile'])
88
Christian Heimesdae2a892008-04-19 00:55:37 +000089 del sys.path[0]
90
91 del sys.modules[pkg]
92
Ned Deilycaf5a222011-10-06 14:19:06 -070093 def test_unreadable_dir_on_syspath(self):
94 # issue7367 - walk_packages failed if unreadable dir on sys.path
95 package_name = "unreadable_package"
96 d = os.path.join(self.dirname, package_name)
97 # this does not appear to create an unreadable dir on Windows
98 # but the test should not fail anyway
99 os.mkdir(d, 0)
Ned Deily7010a072011-10-07 12:01:40 -0700100 self.addCleanup(os.rmdir, d)
Ned Deilycaf5a222011-10-06 14:19:06 -0700101 for t in pkgutil.walk_packages(path=[self.dirname]):
102 self.fail("unexpected package found")
Ned Deilycaf5a222011-10-06 14:19:06 -0700103
Łukasz Langa0d18c152016-06-11 18:02:46 -0700104 def test_walkpackages_filesys(self):
105 pkg1 = 'test_walkpackages_filesys'
106 pkg1_dir = os.path.join(self.dirname, pkg1)
107 os.mkdir(pkg1_dir)
108 f = open(os.path.join(pkg1_dir, '__init__.py'), "wb")
109 f.close()
110 os.mkdir(os.path.join(pkg1_dir, 'sub'))
111 f = open(os.path.join(pkg1_dir, 'sub', '__init__.py'), "wb")
112 f.close()
113 f = open(os.path.join(pkg1_dir, 'sub', 'mod.py'), "wb")
114 f.close()
115
116 # Now, to juice it up, let's add the opposite packages, too.
117 pkg2 = 'sub'
118 pkg2_dir = os.path.join(self.dirname, pkg2)
119 os.mkdir(pkg2_dir)
120 f = open(os.path.join(pkg2_dir, '__init__.py'), "wb")
121 f.close()
122 os.mkdir(os.path.join(pkg2_dir, 'test_walkpackages_filesys'))
123 f = open(os.path.join(pkg2_dir, 'test_walkpackages_filesys', '__init__.py'), "wb")
124 f.close()
125 f = open(os.path.join(pkg2_dir, 'test_walkpackages_filesys', 'mod.py'), "wb")
126 f.close()
127
128 expected = [
129 'sub',
130 'sub.test_walkpackages_filesys',
131 'sub.test_walkpackages_filesys.mod',
132 'test_walkpackages_filesys',
133 'test_walkpackages_filesys.sub',
134 'test_walkpackages_filesys.sub.mod',
135 ]
136 actual= [e[1] for e in pkgutil.walk_packages([self.dirname])]
137 self.assertEqual(actual, expected)
138
139 for pkg in expected:
140 if pkg.endswith('mod'):
141 continue
142 del sys.modules[pkg]
143
144 def test_walkpackages_zipfile(self):
145 """Tests the same as test_walkpackages_filesys, only with a zip file."""
146
147 zip = 'test_walkpackages_zipfile.zip'
148 pkg1 = 'test_walkpackages_zipfile'
149 pkg2 = 'sub'
150
151 zip_file = os.path.join(self.dirname, zip)
152 z = zipfile.ZipFile(zip_file, 'w')
153 z.writestr(pkg2 + '/__init__.py', "")
154 z.writestr(pkg2 + '/' + pkg1 + '/__init__.py', "")
155 z.writestr(pkg2 + '/' + pkg1 + '/mod.py', "")
156 z.writestr(pkg1 + '/__init__.py', "")
157 z.writestr(pkg1 + '/' + pkg2 + '/__init__.py', "")
158 z.writestr(pkg1 + '/' + pkg2 + '/mod.py', "")
159 z.close()
160
161 sys.path.insert(0, zip_file)
162 expected = [
163 'sub',
164 'sub.test_walkpackages_zipfile',
165 'sub.test_walkpackages_zipfile.mod',
166 'test_walkpackages_zipfile',
167 'test_walkpackages_zipfile.sub',
168 'test_walkpackages_zipfile.sub.mod',
169 ]
170 actual= [e[1] for e in pkgutil.walk_packages([zip_file])]
171 self.assertEqual(actual, expected)
172 del sys.path[0]
173
174 for pkg in expected:
175 if pkg.endswith('mod'):
176 continue
177 del sys.modules[pkg]
178
Sanyam Khuranab9c3da52017-06-13 22:41:14 +0530179 def test_walk_packages_raises_on_string_or_bytes_input(self):
180
181 str_input = 'test_dir'
182 with self.assertRaises((TypeError, ValueError)):
183 list(pkgutil.walk_packages(str_input))
184
185 bytes_input = b'test_dir'
186 with self.assertRaises((TypeError, ValueError)):
187 list(pkgutil.walk_packages(bytes_input))
Łukasz Langa0d18c152016-06-11 18:02:46 -0700188
Vinay Sajip1ed61612020-02-14 22:02:13 +0000189 def test_name_resolution(self):
190 import logging
191 import logging.handlers
192
193 success_cases = (
194 ('os', os),
195 ('os.path', os.path),
196 ('os.path:pathsep', os.path.pathsep),
197 ('logging', logging),
198 ('logging:', logging),
199 ('logging.handlers', logging.handlers),
200 ('logging.handlers:', logging.handlers),
201 ('logging.handlers:SysLogHandler', logging.handlers.SysLogHandler),
202 ('logging.handlers.SysLogHandler', logging.handlers.SysLogHandler),
203 ('logging.handlers:SysLogHandler.LOG_ALERT',
204 logging.handlers.SysLogHandler.LOG_ALERT),
205 ('logging.handlers.SysLogHandler.LOG_ALERT',
206 logging.handlers.SysLogHandler.LOG_ALERT),
207 ('builtins.int', int),
208 ('builtins:int', int),
209 ('builtins.int.from_bytes', int.from_bytes),
210 ('builtins:int.from_bytes', int.from_bytes),
211 ('builtins.ZeroDivisionError', ZeroDivisionError),
212 ('builtins:ZeroDivisionError', ZeroDivisionError),
213 ('os:path', os.path),
214 )
215
216 failure_cases = (
217 (None, TypeError),
218 (1, TypeError),
219 (2.0, TypeError),
220 (True, TypeError),
221 ('', ValueError),
222 ('?abc', ValueError),
223 ('abc/foo', ValueError),
224 ('foo', ImportError),
225 ('os.foo', AttributeError),
226 ('os.foo:', ImportError),
227 ('os.pth:pathsep', ImportError),
228 ('logging.handlers:NoSuchHandler', AttributeError),
229 ('logging.handlers:SysLogHandler.NO_SUCH_VALUE', AttributeError),
230 ('logging.handlers.SysLogHandler.NO_SUCH_VALUE', AttributeError),
231 ('ZeroDivisionError', ImportError),
Vinay Sajip4f17c5c2020-02-28 14:26:27 +0000232 ('os.path.9abc', ValueError),
233 ('9abc', ValueError),
Vinay Sajip1ed61612020-02-14 22:02:13 +0000234 )
235
Vinay Sajip4f17c5c2020-02-28 14:26:27 +0000236 # add some Unicode package names to the mix.
237
238 unicode_words = ('\u0935\u092e\u0938',
239 '\xe9', '\xc8',
240 '\uc548\ub155\ud558\uc138\uc694',
241 '\u3055\u3088\u306a\u3089',
242 '\u3042\u308a\u304c\u3068\u3046',
243 '\u0425\u043e\u0440\u043e\u0448\u043e',
244 '\u0441\u043f\u0430\u0441\u0438\u0431\u043e',
245 '\u73b0\u4ee3\u6c49\u8bed\u5e38\u7528\u5b57\u8868')
246
247 for uw in unicode_words:
248 d = os.path.join(self.dirname, uw)
249 os.makedirs(d, exist_ok=True)
250 # make an empty __init__.py file
251 f = os.path.join(d, '__init__.py')
252 with open(f, 'w') as f:
253 f.write('')
254 f.flush()
255 # now import the package we just created; clearing the caches is
256 # needed, otherwise the newly created package isn't found
257 importlib.invalidate_caches()
258 mod = importlib.import_module(uw)
259 success_cases += (uw, mod),
260 if len(uw) > 1:
261 failure_cases += (uw[:-1], ImportError),
262
263 # add an example with a Unicode digit at the start
264 failure_cases += ('\u0966\u0935\u092e\u0938', ValueError),
265
Vinay Sajip1ed61612020-02-14 22:02:13 +0000266 for s, expected in success_cases:
267 with self.subTest(s=s):
268 o = pkgutil.resolve_name(s)
269 self.assertEqual(o, expected)
270
271 for s, exc in failure_cases:
272 with self.subTest(s=s):
273 with self.assertRaises(exc):
274 pkgutil.resolve_name(s)
275
Łukasz Langa0d18c152016-06-11 18:02:46 -0700276
Christian Heimesdae2a892008-04-19 00:55:37 +0000277class PkgutilPEP302Tests(unittest.TestCase):
278
279 class MyTestLoader(object):
Brett Cannon02d84542015-01-09 11:39:21 -0500280 def create_module(self, spec):
281 return None
282
Eric Snow37148b22014-01-04 15:09:53 -0700283 def exec_module(self, mod):
Christian Heimesdae2a892008-04-19 00:55:37 +0000284 # Count how many times the module is reloaded
Eric Snow37148b22014-01-04 15:09:53 -0700285 mod.__dict__['loads'] = mod.__dict__.get('loads', 0) + 1
Christian Heimesdae2a892008-04-19 00:55:37 +0000286
287 def get_data(self, path):
288 return "Hello, world!"
289
290 class MyTestImporter(object):
Eric Snow37148b22014-01-04 15:09:53 -0700291 def find_spec(self, fullname, path=None, target=None):
292 loader = PkgutilPEP302Tests.MyTestLoader()
293 return spec_from_file_location(fullname,
294 '<%s>' % loader.__class__.__name__,
295 loader=loader,
296 submodule_search_locations=[])
Christian Heimesdae2a892008-04-19 00:55:37 +0000297
298 def setUp(self):
299 sys.meta_path.insert(0, self.MyTestImporter())
300
301 def tearDown(self):
302 del sys.meta_path[0]
303
304 def test_getdata_pep302(self):
Brett Cannonfdcdd9e2016-07-08 11:00:00 -0700305 # Use a dummy finder/loader
Christian Heimesdae2a892008-04-19 00:55:37 +0000306 self.assertEqual(pkgutil.get_data('foo', 'dummy'), "Hello, world!")
307 del sys.modules['foo']
308
309 def test_alreadyloaded(self):
310 # Ensure that get_data works without reloading - the "loads" module
311 # variable in the example loader should count how many times a reload
312 # occurs.
313 import foo
314 self.assertEqual(foo.loads, 1)
315 self.assertEqual(pkgutil.get_data('foo', 'dummy'), "Hello, world!")
316 self.assertEqual(foo.loads, 1)
317 del sys.modules['foo']
318
Eric V. Smitha790c9b2012-05-15 20:44:06 -0400319
Eric V. Smith984b11f2012-05-24 20:21:04 -0400320# These tests, especially the setup and cleanup, are hideous. They
321# need to be cleaned up once issue 14715 is addressed.
Eric V. Smitha790c9b2012-05-15 20:44:06 -0400322class ExtendPathTests(unittest.TestCase):
323 def create_init(self, pkgname):
324 dirname = tempfile.mkdtemp()
Eric V. Smitha790c9b2012-05-15 20:44:06 -0400325 sys.path.insert(0, dirname)
326
327 pkgdir = os.path.join(dirname, pkgname)
328 os.mkdir(pkgdir)
329 with open(os.path.join(pkgdir, '__init__.py'), 'w') as fl:
330 fl.write('from pkgutil import extend_path\n__path__ = extend_path(__path__, __name__)\n')
331
332 return dirname
333
334 def create_submodule(self, dirname, pkgname, submodule_name, value):
335 module_name = os.path.join(dirname, pkgname, submodule_name + '.py')
336 with open(module_name, 'w') as fl:
337 print('value={}'.format(value), file=fl)
338
Eric V. Smitha790c9b2012-05-15 20:44:06 -0400339 def test_simple(self):
Eric V. Smith984b11f2012-05-24 20:21:04 -0400340 pkgname = 'foo'
341 dirname_0 = self.create_init(pkgname)
342 dirname_1 = self.create_init(pkgname)
343 self.create_submodule(dirname_0, pkgname, 'bar', 0)
344 self.create_submodule(dirname_1, pkgname, 'baz', 1)
Eric V. Smitha790c9b2012-05-15 20:44:06 -0400345 import foo.bar
346 import foo.baz
347 # Ensure we read the expected values
348 self.assertEqual(foo.bar.value, 0)
349 self.assertEqual(foo.baz.value, 1)
350
351 # Ensure the path is set up correctly
352 self.assertEqual(sorted(foo.__path__),
Eric V. Smith984b11f2012-05-24 20:21:04 -0400353 sorted([os.path.join(dirname_0, pkgname),
354 os.path.join(dirname_1, pkgname)]))
355
356 # Cleanup
357 shutil.rmtree(dirname_0)
358 shutil.rmtree(dirname_1)
359 del sys.path[0]
360 del sys.path[0]
361 del sys.modules['foo']
362 del sys.modules['foo.bar']
363 del sys.modules['foo.baz']
364
Nick Coghlanc4e0d982013-04-14 22:30:42 +1000365
366 # Another awful testing hack to be cleaned up once the test_runpy
367 # helpers are factored out to a common location
368 def test_iter_importers(self):
369 iter_importers = pkgutil.iter_importers
370 get_importer = pkgutil.get_importer
371
372 pkgname = 'spam'
373 modname = 'eggs'
374 dirname = self.create_init(pkgname)
375 pathitem = os.path.join(dirname, pkgname)
376 fullname = '{}.{}'.format(pkgname, modname)
Eric Snow2ba66eb2013-11-22 13:55:23 -0700377 sys.modules.pop(fullname, None)
378 sys.modules.pop(pkgname, None)
Nick Coghlanc4e0d982013-04-14 22:30:42 +1000379 try:
380 self.create_submodule(dirname, pkgname, modname, 0)
381
382 importlib.import_module(fullname)
383
384 importers = list(iter_importers(fullname))
385 expected_importer = get_importer(pathitem)
386 for finder in importers:
Eric Snow37148b22014-01-04 15:09:53 -0700387 spec = pkgutil._get_spec(finder, fullname)
388 loader = spec.loader
Eric Snowb523f842013-11-22 09:05:39 -0700389 try:
390 loader = loader.loader
391 except AttributeError:
392 # For now we still allow raw loaders from
393 # find_module().
394 pass
Nick Coghlanc4e0d982013-04-14 22:30:42 +1000395 self.assertIsInstance(finder, importlib.machinery.FileFinder)
396 self.assertEqual(finder, expected_importer)
Eric Snowb523f842013-11-22 09:05:39 -0700397 self.assertIsInstance(loader,
Nick Coghlanc4e0d982013-04-14 22:30:42 +1000398 importlib.machinery.SourceFileLoader)
Eric Snow37148b22014-01-04 15:09:53 -0700399 self.assertIsNone(pkgutil._get_spec(finder, pkgname))
Nick Coghlanc4e0d982013-04-14 22:30:42 +1000400
401 with self.assertRaises(ImportError):
402 list(iter_importers('invalid.module'))
403
404 with self.assertRaises(ImportError):
405 list(iter_importers('.spam'))
406 finally:
407 shutil.rmtree(dirname)
408 del sys.path[0]
Eric Snowb523f842013-11-22 09:05:39 -0700409 try:
410 del sys.modules['spam']
411 del sys.modules['spam.eggs']
412 except KeyError:
413 pass
Nick Coghlanc4e0d982013-04-14 22:30:42 +1000414
415
Eric V. Smith984b11f2012-05-24 20:21:04 -0400416 def test_mixed_namespace(self):
417 pkgname = 'foo'
418 dirname_0 = self.create_init(pkgname)
419 dirname_1 = self.create_init(pkgname)
420 self.create_submodule(dirname_0, pkgname, 'bar', 0)
421 # Turn this into a PEP 420 namespace package
422 os.unlink(os.path.join(dirname_0, pkgname, '__init__.py'))
423 self.create_submodule(dirname_1, pkgname, 'baz', 1)
424 import foo.bar
425 import foo.baz
426 # Ensure we read the expected values
427 self.assertEqual(foo.bar.value, 0)
428 self.assertEqual(foo.baz.value, 1)
429
430 # Ensure the path is set up correctly
431 self.assertEqual(sorted(foo.__path__),
432 sorted([os.path.join(dirname_0, pkgname),
433 os.path.join(dirname_1, pkgname)]))
434
435 # Cleanup
436 shutil.rmtree(dirname_0)
437 shutil.rmtree(dirname_1)
438 del sys.path[0]
439 del sys.path[0]
440 del sys.modules['foo']
441 del sys.modules['foo.bar']
442 del sys.modules['foo.baz']
Eric V. Smitha790c9b2012-05-15 20:44:06 -0400443
444 # XXX: test .pkg files
445
446
Antoine Pitroub2dd8802012-07-09 21:23:58 +0200447class NestedNamespacePackageTest(unittest.TestCase):
448
449 def setUp(self):
450 self.basedir = tempfile.mkdtemp()
451 self.old_path = sys.path[:]
452
453 def tearDown(self):
454 sys.path[:] = self.old_path
455 shutil.rmtree(self.basedir)
456
457 def create_module(self, name, contents):
458 base, final = name.rsplit('.', 1)
459 base_path = os.path.join(self.basedir, base.replace('.', os.path.sep))
460 os.makedirs(base_path, exist_ok=True)
461 with open(os.path.join(base_path, final + ".py"), 'w') as f:
462 f.write(contents)
463
464 def test_nested(self):
465 pkgutil_boilerplate = (
466 'import pkgutil; '
467 '__path__ = pkgutil.extend_path(__path__, __name__)')
468 self.create_module('a.pkg.__init__', pkgutil_boilerplate)
469 self.create_module('b.pkg.__init__', pkgutil_boilerplate)
470 self.create_module('a.pkg.subpkg.__init__', pkgutil_boilerplate)
471 self.create_module('b.pkg.subpkg.__init__', pkgutil_boilerplate)
472 self.create_module('a.pkg.subpkg.c', 'c = 1')
473 self.create_module('b.pkg.subpkg.d', 'd = 2')
474 sys.path.insert(0, os.path.join(self.basedir, 'a'))
475 sys.path.insert(0, os.path.join(self.basedir, 'b'))
476 import pkg
477 self.addCleanup(unload, 'pkg')
478 self.assertEqual(len(pkg.__path__), 2)
479 import pkg.subpkg
480 self.addCleanup(unload, 'pkg.subpkg')
481 self.assertEqual(len(pkg.subpkg.__path__), 2)
482 from pkg.subpkg.c import c
483 from pkg.subpkg.d import d
484 self.assertEqual(c, 1)
485 self.assertEqual(d, 2)
486
487
Nick Coghlan85e729e2012-07-15 18:09:52 +1000488class ImportlibMigrationTests(unittest.TestCase):
489 # With full PEP 302 support in the standard import machinery, the
490 # PEP 302 emulation in this module is in the process of being
491 # deprecated in favour of importlib proper
492
493 def check_deprecated(self):
494 return check_warnings(
495 ("This emulation is deprecated, use 'importlib' instead",
496 DeprecationWarning))
497
498 def test_importer_deprecated(self):
499 with self.check_deprecated():
Łukasz Langa0d18c152016-06-11 18:02:46 -0700500 pkgutil.ImpImporter("")
Nick Coghlan85e729e2012-07-15 18:09:52 +1000501
502 def test_loader_deprecated(self):
503 with self.check_deprecated():
Łukasz Langa0d18c152016-06-11 18:02:46 -0700504 pkgutil.ImpLoader("", "", "", "")
Nick Coghlan85e729e2012-07-15 18:09:52 +1000505
506 def test_get_loader_avoids_emulation(self):
507 with check_warnings() as w:
508 self.assertIsNotNone(pkgutil.get_loader("sys"))
509 self.assertIsNotNone(pkgutil.get_loader("os"))
510 self.assertIsNotNone(pkgutil.get_loader("test.support"))
511 self.assertEqual(len(w.warnings), 0)
512
Brett Cannon4a2360d2016-08-12 10:53:53 -0700513 @unittest.skipIf(__name__ == '__main__', 'not compatible with __main__')
Nick Coghlandc855b72014-03-04 20:39:42 +1000514 def test_get_loader_handles_missing_loader_attribute(self):
515 global __loader__
516 this_loader = __loader__
517 del __loader__
518 try:
519 with check_warnings() as w:
520 self.assertIsNotNone(pkgutil.get_loader(__name__))
521 self.assertEqual(len(w.warnings), 0)
522 finally:
523 __loader__ = this_loader
524
Eric Snow658af312014-04-19 00:13:23 -0600525 def test_get_loader_handles_missing_spec_attribute(self):
526 name = 'spam'
527 mod = type(sys)(name)
528 del mod.__spec__
529 with CleanImport(name):
530 sys.modules[name] = mod
531 loader = pkgutil.get_loader(name)
532 self.assertIsNone(loader)
533
534 def test_get_loader_handles_spec_attribute_none(self):
535 name = 'spam'
536 mod = type(sys)(name)
537 mod.__spec__ = None
538 with CleanImport(name):
539 sys.modules[name] = mod
540 loader = pkgutil.get_loader(name)
541 self.assertIsNone(loader)
Nick Coghlandc855b72014-03-04 20:39:42 +1000542
Brett Cannon8447c702014-05-23 12:30:37 -0400543 def test_get_loader_None_in_sys_modules(self):
544 name = 'totally bogus'
545 sys.modules[name] = None
546 try:
547 loader = pkgutil.get_loader(name)
548 finally:
549 del sys.modules[name]
550 self.assertIsNone(loader)
551
552 def test_find_loader_missing_module(self):
553 name = 'totally bogus'
554 loader = pkgutil.find_loader(name)
555 self.assertIsNone(loader)
556
Nick Coghlandc855b72014-03-04 20:39:42 +1000557 def test_find_loader_avoids_emulation(self):
558 with check_warnings() as w:
559 self.assertIsNotNone(pkgutil.find_loader("sys"))
560 self.assertIsNotNone(pkgutil.find_loader("os"))
561 self.assertIsNotNone(pkgutil.find_loader("test.support"))
562 self.assertEqual(len(w.warnings), 0)
563
Nick Coghlan85e729e2012-07-15 18:09:52 +1000564 def test_get_importer_avoids_emulation(self):
Nick Coghlan94554922012-07-17 21:37:58 +1000565 # We use an illegal path so *none* of the path hooks should fire
Nick Coghlan85e729e2012-07-15 18:09:52 +1000566 with check_warnings() as w:
Nick Coghlan94554922012-07-17 21:37:58 +1000567 self.assertIsNone(pkgutil.get_importer("*??"))
Nick Coghlan85e729e2012-07-15 18:09:52 +1000568 self.assertEqual(len(w.warnings), 0)
569
570 def test_iter_importers_avoids_emulation(self):
571 with check_warnings() as w:
572 for importer in pkgutil.iter_importers(): pass
573 self.assertEqual(len(w.warnings), 0)
574
575
Christian Heimesdae2a892008-04-19 00:55:37 +0000576def test_main():
Antoine Pitroub2dd8802012-07-09 21:23:58 +0200577 run_unittest(PkgutilTests, PkgutilPEP302Tests, ExtendPathTests,
Nick Coghlan85e729e2012-07-15 18:09:52 +1000578 NestedNamespacePackageTest, ImportlibMigrationTests)
Benjamin Petersoncf626032014-02-16 14:52:01 -0500579 # this is necessary if test is run repeated (like when finding leaks)
580 import zipimport
581 import importlib
582 zipimport._zip_directory_cache.clear()
583 importlib.invalidate_caches()
Nick Coghlan85e729e2012-07-15 18:09:52 +1000584
Christian Heimesdae2a892008-04-19 00:55:37 +0000585
586if __name__ == '__main__':
587 test_main()