| 
									
										
										
										
											2009-01-18 00:24:28 +00:00
										 |  |  | import importlib | 
					
						
							| 
									
										
										
										
											2009-02-01 02:05:11 +00:00
										 |  |  | from .. import abc | 
					
						
							| 
									
										
										
										
											2009-02-01 03:08:31 +00:00
										 |  |  | from . import util as source_util | 
					
						
							| 
									
										
										
										
											2009-01-18 00:24:28 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 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] | 
					
						
							| 
									
										
										
										
											2009-02-01 02:05:11 +00:00
										 |  |  |     def test_module(self): | 
					
						
							| 
									
										
										
										
											2009-02-01 03:08:31 +00:00
										 |  |  |         with source_util.create_modules('_temp') as mapping: | 
					
						
							| 
									
										
										
										
											2009-02-21 05:41:15 +00:00
										 |  |  |             loader = importlib.PyPycFileLoader('_temp', mapping['_temp'], False) | 
					
						
							| 
									
										
										
										
											2009-02-01 02:05:11 +00:00
										 |  |  |             module = loader.load_module('_temp') | 
					
						
							| 
									
										
										
										
											2009-01-18 00:24:28 +00:00
										 |  |  |             self.assert_('_temp' in sys.modules) | 
					
						
							| 
									
										
										
										
											2009-02-01 02:05:11 +00:00
										 |  |  |             check = {'__name__': '_temp', '__file__': mapping['_temp'], | 
					
						
							| 
									
										
										
										
											2009-02-07 01:52:25 +00:00
										 |  |  |                      '__package__': ''} | 
					
						
							| 
									
										
										
										
											2009-02-01 02:05:11 +00:00
										 |  |  |             for attr, value in check.items(): | 
					
						
							|  |  |  |                 self.assertEqual(getattr(module, attr), value) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def test_package(self): | 
					
						
							| 
									
										
										
										
											2009-02-01 03:08:31 +00:00
										 |  |  |         with source_util.create_modules('_pkg.__init__') as mapping: | 
					
						
							| 
									
										
										
										
											2009-02-21 05:41:15 +00:00
										 |  |  |             loader = importlib.PyPycFileLoader('_pkg', mapping['_pkg.__init__'], | 
					
						
							| 
									
										
										
										
											2009-02-01 02:05:11 +00:00
										 |  |  |                                              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): | 
					
						
							| 
									
										
										
										
											2009-02-01 03:08:31 +00:00
										 |  |  |         with source_util.create_modules('_pkg.__init__', '_pkg.mod')as mapping: | 
					
						
							| 
									
										
										
										
											2009-02-21 05:41:15 +00:00
										 |  |  |             loader = importlib.PyPycFileLoader('_pkg.mod', mapping['_pkg.mod'], | 
					
						
							| 
									
										
										
										
											2009-02-01 02:05:11 +00:00
										 |  |  |                                              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): | 
					
						
							| 
									
										
										
										
											2009-02-01 03:08:31 +00:00
										 |  |  |         with source_util.create_modules('_temp') as mapping: | 
					
						
							| 
									
										
										
										
											2009-02-21 05:41:15 +00:00
										 |  |  |             loader = importlib.PyPycFileLoader('_temp', mapping['_temp'], False) | 
					
						
							| 
									
										
										
										
											2009-02-01 02:05:11 +00:00
										 |  |  |             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' | 
					
						
							| 
									
										
										
										
											2009-02-01 03:08:31 +00:00
										 |  |  |         with source_util.create_modules(name) as mapping: | 
					
						
							| 
									
										
										
										
											2009-02-01 02:05:11 +00:00
										 |  |  |             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 +++') | 
					
						
							| 
									
										
										
										
											2009-02-21 05:41:15 +00:00
										 |  |  |             loader = importlib.PyPycFileLoader('_temp', mapping['_temp'], False) | 
					
						
							| 
									
										
										
										
											2009-02-01 02:05:11 +00:00
										 |  |  |             self.assertRaises(SyntaxError, loader.load_module, name) | 
					
						
							|  |  |  |             for attr in attributes: | 
					
						
							|  |  |  |                 self.assertEqual(getattr(orig_module, attr), value) | 
					
						
							| 
									
										
										
										
											2009-01-18 00:24:28 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     # [syntax error] | 
					
						
							|  |  |  |     def test_bad_syntax(self): | 
					
						
							| 
									
										
										
										
											2009-02-01 03:08:31 +00:00
										 |  |  |         with source_util.create_modules('_temp') as mapping: | 
					
						
							| 
									
										
										
										
											2009-01-18 00:24:28 +00:00
										 |  |  |             with open(mapping['_temp'], 'w') as file: | 
					
						
							|  |  |  |                 file.write('=') | 
					
						
							| 
									
										
										
										
											2009-02-21 05:41:15 +00:00
										 |  |  |             loader = importlib.PyPycFileLoader('_temp', mapping['_temp'], False) | 
					
						
							| 
									
										
										
										
											2009-01-18 00:24:28 +00:00
										 |  |  |             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 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-02-01 03:08:31 +00:00
										 |  |  |     @source_util.writes_bytecode | 
					
						
							| 
									
										
										
										
											2009-01-18 00:24:28 +00:00
										 |  |  |     def run_test(self, assertion): | 
					
						
							| 
									
										
										
										
											2009-02-01 03:08:31 +00:00
										 |  |  |         with source_util.create_modules('_temp') as mapping: | 
					
						
							| 
									
										
										
										
											2009-02-21 05:41:15 +00:00
										 |  |  |             loader = importlib.PyPycFileLoader('_temp', mapping['_temp'], False) | 
					
						
							| 
									
										
										
										
											2009-01-18 00:24:28 +00:00
										 |  |  |             loader.load_module('_temp') | 
					
						
							| 
									
										
										
										
											2009-02-01 03:08:31 +00:00
										 |  |  |             bytecode_path = source_util.bytecode_path(mapping['_temp']) | 
					
						
							| 
									
										
										
										
											2009-01-18 00:24:28 +00:00
										 |  |  |             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): | 
					
						
							| 
									
										
										
										
											2009-02-01 03:08:31 +00:00
										 |  |  |         with source_util.create_modules('_temp') as mapping: | 
					
						
							| 
									
										
										
										
											2009-01-18 00:24:28 +00:00
										 |  |  |             py_compile.compile(mapping['_temp']) | 
					
						
							|  |  |  |             os.unlink(mapping['_temp']) | 
					
						
							| 
									
										
										
										
											2009-02-01 03:08:31 +00:00
										 |  |  |             bytecode_path = source_util.bytecode_path(mapping['_temp']) | 
					
						
							| 
									
										
										
										
											2009-01-18 00:24:28 +00:00
										 |  |  |             with open(bytecode_path, 'r+b') as file: | 
					
						
							|  |  |  |                 file.seek(0) | 
					
						
							|  |  |  |                 file.write(b'\x00\x00\x00\x00') | 
					
						
							| 
									
										
										
										
											2009-02-21 05:41:15 +00:00
										 |  |  |             loader = importlib.PyPycFileLoader('_temp', mapping['_temp'], False) | 
					
						
							| 
									
										
										
										
											2009-01-18 00:24:28 +00:00
										 |  |  |             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): | 
					
						
							| 
									
										
										
										
											2009-02-21 05:41:15 +00:00
										 |  |  |         loader = importlib.PyPycFileLoader(module, file, pkg) | 
					
						
							| 
									
										
										
										
											2009-01-18 00:24:28 +00:00
										 |  |  |         return loader.load_module(module) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def run_test(self, test, *create, pkg=False): | 
					
						
							|  |  |  |         create += (test,) | 
					
						
							| 
									
										
										
										
											2009-02-01 03:08:31 +00:00
										 |  |  |         with source_util.create_modules(*create) as mapping: | 
					
						
							| 
									
										
										
										
											2009-01-18 00:24:28 +00:00
										 |  |  |             for name in create: | 
					
						
							|  |  |  |                 py_compile.compile(mapping[name]) | 
					
						
							|  |  |  |             if pkg: | 
					
						
							|  |  |  |                 import_name = test.rsplit('.', 1)[0] | 
					
						
							|  |  |  |             else: | 
					
						
							|  |  |  |                 import_name = test | 
					
						
							| 
									
										
										
										
											2009-02-21 05:41:15 +00:00
										 |  |  |             loader = importlib.PyPycFileLoader(import_name, mapping[test], pkg) | 
					
						
							| 
									
										
										
										
											2009-01-18 00:24:28 +00:00
										 |  |  |             # 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): | 
					
						
							| 
									
										
										
										
											2009-02-21 05:41:15 +00:00
										 |  |  |         loader = importlib.PyPycFileLoader(module_name, file, False) | 
					
						
							| 
									
										
										
										
											2009-01-18 00:24:28 +00:00
										 |  |  |         module = loader.load_module(module_name) | 
					
						
							|  |  |  |         self.assert_(module_name in sys.modules) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     # [bad magic] | 
					
						
							| 
									
										
										
										
											2009-02-01 03:08:31 +00:00
										 |  |  |     @source_util.writes_bytecode | 
					
						
							| 
									
										
										
										
											2009-01-18 00:24:28 +00:00
										 |  |  |     def test_bad_magic(self): | 
					
						
							| 
									
										
										
										
											2009-02-01 03:08:31 +00:00
										 |  |  |         with source_util.create_modules('_temp') as mapping: | 
					
						
							| 
									
										
										
										
											2009-01-18 00:24:28 +00:00
										 |  |  |             py_compile.compile(mapping['_temp']) | 
					
						
							| 
									
										
										
										
											2009-02-01 03:08:31 +00:00
										 |  |  |             bytecode_path = source_util.bytecode_path(mapping['_temp']) | 
					
						
							| 
									
										
										
										
											2009-01-18 00:24:28 +00:00
										 |  |  |             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] | 
					
						
							| 
									
										
										
										
											2009-02-01 03:08:31 +00:00
										 |  |  |     @source_util.writes_bytecode | 
					
						
							| 
									
										
										
										
											2009-01-18 00:24:28 +00:00
										 |  |  |     def test_bad_bytecode(self): | 
					
						
							|  |  |  |         zeros = b'\x00\x00\x00\x00' | 
					
						
							| 
									
										
										
										
											2009-02-01 03:08:31 +00:00
										 |  |  |         with source_util.create_modules('_temp') as mapping: | 
					
						
							| 
									
										
										
										
											2009-01-18 00:24:28 +00:00
										 |  |  |             py_compile.compile(mapping['_temp']) | 
					
						
							| 
									
										
										
										
											2009-02-01 03:08:31 +00:00
										 |  |  |             bytecode_path = source_util.bytecode_path(mapping['_temp']) | 
					
						
							| 
									
										
										
										
											2009-01-18 00:24:28 +00:00
										 |  |  |             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): | 
					
						
							| 
									
										
										
										
											2009-02-01 03:08:31 +00:00
										 |  |  |         with source_util.create_modules('_temp') as mapping: | 
					
						
							|  |  |  |             bytecode_path = source_util.bytecode_path(mapping['_temp']) | 
					
						
							| 
									
										
										
										
											2009-01-18 00:24:28 +00:00
										 |  |  |             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() |