mirror of
				https://github.com/python/cpython.git
				synced 2025-10-31 13:41:24 +00:00 
			
		
		
		
	Introduce importlib.abc. The module contains various ABCs related to imports
(mostly stuff specified by PEP 302). There are two ABCs, PyLoader and PyPycLoader, which help with implementing source and source/bytecode loaders by implementing load_module in terms of other methods. This removes a lot of gritty details loaders typically have to worry about.
This commit is contained in:
		
							parent
							
								
									aa1c8d8899
								
							
						
					
					
						commit
						2a922ed6ad
					
				
					 9 changed files with 739 additions and 171 deletions
				
			
		|  | @ -1,271 +0,0 @@ | |||
| import importlib | ||||
| from .. import abc | ||||
| from . import util as source_util | ||||
| 
 | ||||
| import imp | ||||
| import os | ||||
| import py_compile | ||||
| import sys | ||||
| import unittest | ||||
| 
 | ||||
| 
 | ||||
| class SimpleTest(unittest.TestCase): | ||||
| 
 | ||||
|     """Should have no issue importing a source module [basic]. And if there is | ||||
|     a syntax error, it should raise a SyntaxError [syntax error]. | ||||
| 
 | ||||
|     """ | ||||
| 
 | ||||
|     # [basic] | ||||
|     def test_module(self): | ||||
|         with source_util.create_modules('_temp') as mapping: | ||||
|             loader = importlib.PyPycFileLoader('_temp', mapping['_temp'], False) | ||||
|             module = loader.load_module('_temp') | ||||
|             self.assert_('_temp' in sys.modules) | ||||
|             check = {'__name__': '_temp', '__file__': mapping['_temp'], | ||||
|                      '__package__': ''} | ||||
|             for attr, value in check.items(): | ||||
|                 self.assertEqual(getattr(module, attr), value) | ||||
| 
 | ||||
|     def test_package(self): | ||||
|         with source_util.create_modules('_pkg.__init__') as mapping: | ||||
|             loader = importlib.PyPycFileLoader('_pkg', mapping['_pkg.__init__'], | ||||
|                                              True) | ||||
|             module = loader.load_module('_pkg') | ||||
|             self.assert_('_pkg' in sys.modules) | ||||
|             check = {'__name__': '_pkg', '__file__': mapping['_pkg.__init__'], | ||||
|                      '__path__': [os.path.dirname(mapping['_pkg.__init__'])], | ||||
|                      '__package__': '_pkg'} | ||||
|             for attr, value in check.items(): | ||||
|                 self.assertEqual(getattr(module, attr), value) | ||||
| 
 | ||||
| 
 | ||||
|     def test_lacking_parent(self): | ||||
|         with source_util.create_modules('_pkg.__init__', '_pkg.mod')as mapping: | ||||
|             loader = importlib.PyPycFileLoader('_pkg.mod', mapping['_pkg.mod'], | ||||
|                                              False) | ||||
|             module = loader.load_module('_pkg.mod') | ||||
|             self.assert_('_pkg.mod' in sys.modules) | ||||
|             check = {'__name__': '_pkg.mod', '__file__': mapping['_pkg.mod'], | ||||
|                      '__package__': '_pkg'} | ||||
|             for attr, value in check.items(): | ||||
|                 self.assertEqual(getattr(module, attr), value) | ||||
| 
 | ||||
|     def fake_mtime(self, fxn): | ||||
|         """Fake mtime to always be higher than expected.""" | ||||
|         return lambda name: fxn(name) + 1 | ||||
| 
 | ||||
|     def test_module_reuse(self): | ||||
|         with source_util.create_modules('_temp') as mapping: | ||||
|             loader = importlib.PyPycFileLoader('_temp', mapping['_temp'], False) | ||||
|             module = loader.load_module('_temp') | ||||
|             module_id = id(module) | ||||
|             module_dict_id = id(module.__dict__) | ||||
|             with open(mapping['_temp'], 'w') as file: | ||||
|                 file.write("testing_var = 42\n") | ||||
|             # For filesystems where the mtime is only to a second granularity, | ||||
|             # everything that has happened above can be too fast; | ||||
|             # force an mtime on the source that is guaranteed to be different | ||||
|             # than the original mtime. | ||||
|             loader.source_mtime = self.fake_mtime(loader.source_mtime) | ||||
|             module = loader.load_module('_temp') | ||||
|             self.assert_('testing_var' in module.__dict__, | ||||
|                          "'testing_var' not in " | ||||
|                             "{0}".format(list(module.__dict__.keys()))) | ||||
|             self.assertEqual(module, sys.modules['_temp']) | ||||
|             self.assertEqual(id(module), module_id) | ||||
|             self.assertEqual(id(module.__dict__), module_dict_id) | ||||
| 
 | ||||
|     def test_state_after_failure(self): | ||||
|         # A failed reload should leave the original module intact. | ||||
|         attributes = ('__file__', '__path__', '__package__') | ||||
|         value = '<test>' | ||||
|         name = '_temp' | ||||
|         with source_util.create_modules(name) as mapping: | ||||
|             orig_module = imp.new_module(name) | ||||
|             for attr in attributes: | ||||
|                 setattr(orig_module, attr, value) | ||||
|             with open(mapping[name], 'w') as file: | ||||
|                 file.write('+++ bad syntax +++') | ||||
|             loader = importlib.PyPycFileLoader('_temp', mapping['_temp'], False) | ||||
|             self.assertRaises(SyntaxError, loader.load_module, name) | ||||
|             for attr in attributes: | ||||
|                 self.assertEqual(getattr(orig_module, attr), value) | ||||
| 
 | ||||
|     # [syntax error] | ||||
|     def test_bad_syntax(self): | ||||
|         with source_util.create_modules('_temp') as mapping: | ||||
|             with open(mapping['_temp'], 'w') as file: | ||||
|                 file.write('=') | ||||
|             loader = importlib.PyPycFileLoader('_temp', mapping['_temp'], False) | ||||
|             self.assertRaises(SyntaxError, loader.load_module, '_temp') | ||||
|             self.assert_('_temp' not in sys.modules) | ||||
| 
 | ||||
| 
 | ||||
| class DontWriteBytecodeTest(unittest.TestCase): | ||||
| 
 | ||||
|     """If sys.dont_write_bytcode is true then no bytecode should be created.""" | ||||
| 
 | ||||
|     def tearDown(self): | ||||
|         sys.dont_write_bytecode = False | ||||
| 
 | ||||
|     @source_util.writes_bytecode | ||||
|     def run_test(self, assertion): | ||||
|         with source_util.create_modules('_temp') as mapping: | ||||
|             loader = importlib.PyPycFileLoader('_temp', mapping['_temp'], False) | ||||
|             loader.load_module('_temp') | ||||
|             bytecode_path = source_util.bytecode_path(mapping['_temp']) | ||||
|             assertion(bytecode_path) | ||||
| 
 | ||||
|     def test_bytecode_written(self): | ||||
|         fxn = lambda bc_path: self.assert_(os.path.exists(bc_path)) | ||||
|         self.run_test(fxn) | ||||
| 
 | ||||
|     def test_bytecode_not_written(self): | ||||
|         sys.dont_write_bytecode = True | ||||
|         fxn = lambda bc_path: self.assert_(not os.path.exists(bc_path)) | ||||
|         self.run_test(fxn) | ||||
| 
 | ||||
| 
 | ||||
| class BadDataTest(unittest.TestCase): | ||||
| 
 | ||||
|     """If the bytecode has a magic number that does not match the | ||||
|     interpreters', ImportError is raised [bad magic]. The timestamp can have | ||||
|     any value. And bad marshal data raises ValueError. | ||||
| 
 | ||||
|     """ | ||||
| 
 | ||||
|     # [bad magic] | ||||
|     def test_bad_magic(self): | ||||
|         with source_util.create_modules('_temp') as mapping: | ||||
|             py_compile.compile(mapping['_temp']) | ||||
|             os.unlink(mapping['_temp']) | ||||
|             bytecode_path = source_util.bytecode_path(mapping['_temp']) | ||||
|             with open(bytecode_path, 'r+b') as file: | ||||
|                 file.seek(0) | ||||
|                 file.write(b'\x00\x00\x00\x00') | ||||
|             loader = importlib.PyPycFileLoader('_temp', mapping['_temp'], False) | ||||
|             self.assertRaises(ImportError, loader.load_module, '_temp') | ||||
|             self.assert_('_temp' not in sys.modules) | ||||
| 
 | ||||
| 
 | ||||
| class SourceBytecodeInteraction(unittest.TestCase): | ||||
| 
 | ||||
|     """When both source and bytecode are present, certain rules dictate which | ||||
|     version of the code takes precedent. All things being equal, the bytecode | ||||
|     is used with the value of __file__ set to the source [basic top-level], | ||||
|     [basic package], [basic sub-module], [basic sub-package]. | ||||
| 
 | ||||
|     """ | ||||
| 
 | ||||
|     def import_(self, file, module, *, pkg=False): | ||||
|         loader = importlib.PyPycFileLoader(module, file, pkg) | ||||
|         return loader.load_module(module) | ||||
| 
 | ||||
|     def run_test(self, test, *create, pkg=False): | ||||
|         create += (test,) | ||||
|         with source_util.create_modules(*create) as mapping: | ||||
|             for name in create: | ||||
|                 py_compile.compile(mapping[name]) | ||||
|             if pkg: | ||||
|                 import_name = test.rsplit('.', 1)[0] | ||||
|             else: | ||||
|                 import_name = test | ||||
|             loader = importlib.PyPycFileLoader(import_name, mapping[test], pkg) | ||||
|             # Because some platforms only have a granularity to the second for | ||||
|             # atime you can't check the physical files. Instead just make it an | ||||
|             # exception trigger if source was read. | ||||
|             loader.get_source = lambda self, x: 42 | ||||
|             module = loader.load_module(import_name) | ||||
|             self.assertEqual(module.__file__, mapping[name]) | ||||
|             self.assert_(import_name in sys.modules) | ||||
|             self.assertEqual(id(module), id(sys.modules[import_name])) | ||||
| 
 | ||||
|     # [basic top-level] | ||||
|     def test_basic_top_level(self): | ||||
|         self.run_test('top_level') | ||||
| 
 | ||||
|     # [basic package] | ||||
|     def test_basic_package(self): | ||||
|         self.run_test('pkg.__init__', pkg=True) | ||||
| 
 | ||||
|     # [basic sub-module] | ||||
|     def test_basic_sub_module(self): | ||||
|         self.run_test('pkg.sub', 'pkg.__init__') | ||||
| 
 | ||||
|     # [basic sub-package] | ||||
|     def test_basic_sub_package(self): | ||||
|         self.run_test('pkg.sub.__init__', 'pkg.__init__', pkg=True) | ||||
| 
 | ||||
| 
 | ||||
| class BadBytecodeTest(unittest.TestCase): | ||||
| 
 | ||||
|     """But there are several things about the bytecode which might lead to the | ||||
|     source being preferred. If the magic number differs from what the | ||||
|     interpreter uses, then the source is used with the bytecode regenerated. | ||||
|     If the timestamp is older than the modification time for the source then | ||||
|     the bytecode is not used [bad timestamp]. | ||||
| 
 | ||||
|     But if the marshal data is bad, even if the magic number and timestamp | ||||
|     work, a ValueError is raised and the source is not used [bad marshal]. | ||||
| 
 | ||||
|     """ | ||||
| 
 | ||||
|     def import_(self, file, module_name): | ||||
|         loader = importlib.PyPycFileLoader(module_name, file, False) | ||||
|         module = loader.load_module(module_name) | ||||
|         self.assert_(module_name in sys.modules) | ||||
| 
 | ||||
|     # [bad magic] | ||||
|     @source_util.writes_bytecode | ||||
|     def test_bad_magic(self): | ||||
|         with source_util.create_modules('_temp') as mapping: | ||||
|             py_compile.compile(mapping['_temp']) | ||||
|             bytecode_path = source_util.bytecode_path(mapping['_temp']) | ||||
|             with open(bytecode_path, 'r+b') as bytecode_file: | ||||
|                 bytecode_file.seek(0) | ||||
|                 bytecode_file.write(b'\x00\x00\x00\x00') | ||||
|             self.import_(mapping['_temp'], '_temp') | ||||
|             with open(bytecode_path, 'rb') as bytecode_file: | ||||
|                 self.assertEqual(bytecode_file.read(4), imp.get_magic()) | ||||
| 
 | ||||
|     # [bad timestamp] | ||||
|     @source_util.writes_bytecode | ||||
|     def test_bad_bytecode(self): | ||||
|         zeros = b'\x00\x00\x00\x00' | ||||
|         with source_util.create_modules('_temp') as mapping: | ||||
|             py_compile.compile(mapping['_temp']) | ||||
|             bytecode_path = source_util.bytecode_path(mapping['_temp']) | ||||
|             with open(bytecode_path, 'r+b') as bytecode_file: | ||||
|                 bytecode_file.seek(4) | ||||
|                 bytecode_file.write(zeros) | ||||
|             self.import_(mapping['_temp'], '_temp') | ||||
|             source_mtime = os.path.getmtime(mapping['_temp']) | ||||
|             source_timestamp = importlib._w_long(source_mtime) | ||||
|             with open(bytecode_path, 'rb') as bytecode_file: | ||||
|                 bytecode_file.seek(4) | ||||
|                 self.assertEqual(bytecode_file.read(4), source_timestamp) | ||||
| 
 | ||||
|     # [bad marshal] | ||||
|     def test_bad_marshal(self): | ||||
|         with source_util.create_modules('_temp') as mapping: | ||||
|             bytecode_path = source_util.bytecode_path(mapping['_temp']) | ||||
|             source_mtime = os.path.getmtime(mapping['_temp']) | ||||
|             source_timestamp = importlib._w_long(source_mtime) | ||||
|             with open(bytecode_path, 'wb') as bytecode_file: | ||||
|                 bytecode_file.write(imp.get_magic()) | ||||
|                 bytecode_file.write(source_timestamp) | ||||
|                 bytecode_file.write(b'AAAA') | ||||
|             self.assertRaises(ValueError, self.import_, mapping['_temp'], | ||||
|                                 '_temp') | ||||
|             self.assert_('_temp' not in sys.modules) | ||||
| 
 | ||||
| 
 | ||||
| def test_main(): | ||||
|     from test.support import run_unittest | ||||
|     run_unittest(SimpleTest, DontWriteBytecodeTest, BadDataTest, | ||||
|                  SourceBytecodeInteraction, BadBytecodeTest) | ||||
| 
 | ||||
| 
 | ||||
| if __name__ == '__main__': | ||||
|     test_main() | ||||
		Loading…
	
	Add table
		Add a link
		
	
		Reference in a new issue
	
	 Brett Cannon
						Brett Cannon