| 
									
										
										
										
											2009-01-18 00:24:28 +00:00
										 |  |  | import importlib | 
					
						
							| 
									
										
										
										
											2009-03-12 22:47:53 +00:00
										 |  |  | from importlib import _bootstrap | 
					
						
							| 
									
										
										
										
											2009-02-01 02:05:11 +00:00
										 |  |  | from .. import abc | 
					
						
							| 
									
										
										
										
											2010-07-03 22:18:47 +00:00
										 |  |  | from .. import util | 
					
						
							| 
									
										
										
										
											2009-02-01 03:08:31 +00:00
										 |  |  | from . import util as source_util | 
					
						
							| 
									
										
										
										
											2009-01-18 00:24:28 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | import imp | 
					
						
							| 
									
										
										
										
											2010-07-03 21:48:25 +00:00
										 |  |  | import marshal | 
					
						
							| 
									
										
										
										
											2009-01-18 00:24:28 +00:00
										 |  |  | import os | 
					
						
							|  |  |  | import py_compile | 
					
						
							| 
									
										
										
										
											2009-11-07 23:55:05 +00:00
										 |  |  | import stat | 
					
						
							| 
									
										
										
										
											2009-01-18 00:24:28 +00:00
										 |  |  | import sys | 
					
						
							|  |  |  | import unittest | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-05-18 14:15:20 +00:00
										 |  |  | from test.support import make_legacy_pyc | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-01-18 00:24:28 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 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: | 
					
						
							| 
									
										
										
										
											2010-07-03 21:48:25 +00:00
										 |  |  |             loader = _bootstrap._SourceFileLoader('_temp', mapping['_temp']) | 
					
						
							| 
									
										
										
										
											2009-02-01 02:05:11 +00:00
										 |  |  |             module = loader.load_module('_temp') | 
					
						
							| 
									
										
										
										
											2009-06-30 23:06:06 +00:00
										 |  |  |             self.assertTrue('_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: | 
					
						
							| 
									
										
										
										
											2010-07-03 21:48:25 +00:00
										 |  |  |             loader = _bootstrap._SourceFileLoader('_pkg', | 
					
						
							|  |  |  |                                                  mapping['_pkg.__init__']) | 
					
						
							| 
									
										
										
										
											2009-02-01 02:05:11 +00:00
										 |  |  |             module = loader.load_module('_pkg') | 
					
						
							| 
									
										
										
										
											2009-06-30 23:06:06 +00:00
										 |  |  |             self.assertTrue('_pkg' in sys.modules) | 
					
						
							| 
									
										
										
										
											2009-02-01 02:05:11 +00:00
										 |  |  |             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: | 
					
						
							| 
									
										
										
										
											2010-07-03 21:48:25 +00:00
										 |  |  |             loader = _bootstrap._SourceFileLoader('_pkg.mod', | 
					
						
							|  |  |  |                                                     mapping['_pkg.mod']) | 
					
						
							| 
									
										
										
										
											2009-02-01 02:05:11 +00:00
										 |  |  |             module = loader.load_module('_pkg.mod') | 
					
						
							| 
									
										
										
										
											2009-06-30 23:06:06 +00:00
										 |  |  |             self.assertTrue('_pkg.mod' in sys.modules) | 
					
						
							| 
									
										
										
										
											2009-02-01 02:05:11 +00:00
										 |  |  |             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: | 
					
						
							| 
									
										
										
										
											2010-07-03 21:48:25 +00:00
										 |  |  |             loader = _bootstrap._SourceFileLoader('_temp', mapping['_temp']) | 
					
						
							| 
									
										
										
										
											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. | 
					
						
							| 
									
										
										
										
											2010-07-03 21:48:25 +00:00
										 |  |  |             loader.path_mtime = self.fake_mtime(loader.path_mtime) | 
					
						
							| 
									
										
										
										
											2009-02-01 02:05:11 +00:00
										 |  |  |             module = loader.load_module('_temp') | 
					
						
							| 
									
										
										
										
											2009-06-30 23:06:06 +00:00
										 |  |  |             self.assertTrue('testing_var' in module.__dict__, | 
					
						
							| 
									
										
										
										
											2009-02-01 02:05:11 +00:00
										 |  |  |                          "'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 +++') | 
					
						
							| 
									
										
										
										
											2010-07-03 21:48:25 +00:00
										 |  |  |             loader = _bootstrap._SourceFileLoader('_temp', mapping['_temp']) | 
					
						
							| 
									
										
										
										
											2009-08-27 23:49:21 +00:00
										 |  |  |             with self.assertRaises(SyntaxError): | 
					
						
							|  |  |  |                 loader.load_module(name) | 
					
						
							| 
									
										
										
										
											2009-02-01 02:05:11 +00:00
										 |  |  |             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('=') | 
					
						
							| 
									
										
										
										
											2010-07-03 21:48:25 +00:00
										 |  |  |             loader = _bootstrap._SourceFileLoader('_temp', mapping['_temp']) | 
					
						
							| 
									
										
										
										
											2009-08-27 23:49:21 +00:00
										 |  |  |             with self.assertRaises(SyntaxError): | 
					
						
							|  |  |  |                 loader.load_module('_temp') | 
					
						
							| 
									
										
										
										
											2009-06-30 23:06:06 +00:00
										 |  |  |             self.assertTrue('_temp' not in sys.modules) | 
					
						
							| 
									
										
										
										
											2009-01-18 00:24:28 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-07-03 22:18:47 +00:00
										 |  |  |     def test_file_from_empty_string_dir(self): | 
					
						
							|  |  |  |         # Loading a module found from an empty string entry on sys.path should | 
					
						
							|  |  |  |         # not only work, but keep all attributes relative. | 
					
						
							|  |  |  |         with open('_temp.py', 'w') as file: | 
					
						
							|  |  |  |             file.write("# test file for importlib") | 
					
						
							|  |  |  |         try: | 
					
						
							|  |  |  |             with util.uncache('_temp'): | 
					
						
							|  |  |  |                 loader = _bootstrap._SourceFileLoader('_temp', '_temp.py') | 
					
						
							|  |  |  |                 mod = loader.load_module('_temp') | 
					
						
							|  |  |  |                 self.assertEqual('_temp.py', mod.__file__) | 
					
						
							|  |  |  |                 self.assertEqual(imp.cache_from_source('_temp.py'), | 
					
						
							|  |  |  |                                  mod.__cached__) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         finally: | 
					
						
							|  |  |  |             os.unlink('_temp.py') | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-01-18 00:24:28 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | class BadBytecodeTest(unittest.TestCase): | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def import_(self, file, module_name): | 
					
						
							| 
									
										
										
										
											2010-07-03 21:48:25 +00:00
										 |  |  |         loader = self.loader(module_name, file) | 
					
						
							| 
									
										
										
										
											2009-01-18 00:24:28 +00:00
										 |  |  |         module = loader.load_module(module_name) | 
					
						
							| 
									
										
										
										
											2009-06-30 23:06:06 +00:00
										 |  |  |         self.assertTrue(module_name in sys.modules) | 
					
						
							| 
									
										
										
										
											2009-01-18 00:24:28 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-02-19 16:01:06 +00:00
										 |  |  |     def manipulate_bytecode(self, name, mapping, manipulator, *, | 
					
						
							|  |  |  |                             del_source=False): | 
					
						
							|  |  |  |         """Manipulate the bytecode of a module by passing it into a callable
 | 
					
						
							|  |  |  |         that returns what to use as the new bytecode."""
 | 
					
						
							|  |  |  |         try: | 
					
						
							|  |  |  |             del sys.modules['_temp'] | 
					
						
							|  |  |  |         except KeyError: | 
					
						
							|  |  |  |             pass | 
					
						
							|  |  |  |         py_compile.compile(mapping[name]) | 
					
						
							| 
									
										
										
										
											2010-07-03 21:48:25 +00:00
										 |  |  |         if not del_source: | 
					
						
							|  |  |  |             bytecode_path = imp.cache_from_source(mapping[name]) | 
					
						
							|  |  |  |         else: | 
					
						
							| 
									
										
										
										
											2010-02-19 16:01:06 +00:00
										 |  |  |             os.unlink(mapping[name]) | 
					
						
							| 
									
										
										
										
											2010-07-03 21:48:25 +00:00
										 |  |  |             bytecode_path = make_legacy_pyc(mapping[name]) | 
					
						
							|  |  |  |         if manipulator: | 
					
						
							|  |  |  |             with open(bytecode_path, 'rb') as file: | 
					
						
							|  |  |  |                 bc = file.read() | 
					
						
							|  |  |  |                 new_bc = manipulator(bc) | 
					
						
							|  |  |  |             with open(bytecode_path, 'wb') as file: | 
					
						
							|  |  |  |                 if new_bc is not None: | 
					
						
							|  |  |  |                     file.write(new_bc) | 
					
						
							| 
									
										
										
										
											2010-02-19 16:01:06 +00:00
										 |  |  |         return bytecode_path | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-07-03 21:48:25 +00:00
										 |  |  |     def _test_empty_file(self, test, *, del_source=False): | 
					
						
							| 
									
										
										
										
											2010-02-19 16:01:06 +00:00
										 |  |  |         with source_util.create_modules('_temp') as mapping: | 
					
						
							|  |  |  |             bc_path = self.manipulate_bytecode('_temp', mapping, | 
					
						
							| 
									
										
										
										
											2010-07-03 21:48:25 +00:00
										 |  |  |                                                 lambda bc: b'', | 
					
						
							|  |  |  |                                                 del_source=del_source) | 
					
						
							|  |  |  |             test('_temp', mapping, bc_path) | 
					
						
							| 
									
										
										
										
											2010-02-19 16:01:06 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     @source_util.writes_bytecode_files | 
					
						
							| 
									
										
										
										
											2010-07-03 21:48:25 +00:00
										 |  |  |     def _test_partial_magic(self, test, *, del_source=False): | 
					
						
							| 
									
										
										
										
											2010-02-19 16:01:06 +00:00
										 |  |  |         # When their are less than 4 bytes to a .pyc, regenerate it if | 
					
						
							|  |  |  |         # possible, else raise ImportError. | 
					
						
							|  |  |  |         with source_util.create_modules('_temp') as mapping: | 
					
						
							|  |  |  |             bc_path = self.manipulate_bytecode('_temp', mapping, | 
					
						
							| 
									
										
										
										
											2010-07-03 21:48:25 +00:00
										 |  |  |                                                 lambda bc: bc[:3], | 
					
						
							|  |  |  |                                                 del_source=del_source) | 
					
						
							|  |  |  |             test('_temp', mapping, bc_path) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def _test_magic_only(self, test, *, del_source=False): | 
					
						
							|  |  |  |         with source_util.create_modules('_temp') as mapping: | 
					
						
							|  |  |  |             bc_path = self.manipulate_bytecode('_temp', mapping, | 
					
						
							|  |  |  |                                                 lambda bc: bc[:4], | 
					
						
							|  |  |  |                                                 del_source=del_source) | 
					
						
							|  |  |  |             test('_temp', mapping, bc_path) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def _test_partial_timestamp(self, test, *, del_source=False): | 
					
						
							|  |  |  |         with source_util.create_modules('_temp') as mapping: | 
					
						
							|  |  |  |             bc_path = self.manipulate_bytecode('_temp', mapping, | 
					
						
							|  |  |  |                                                 lambda bc: bc[:7], | 
					
						
							|  |  |  |                                                 del_source=del_source) | 
					
						
							|  |  |  |             test('_temp', mapping, bc_path) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def _test_no_marshal(self, *, del_source=False): | 
					
						
							|  |  |  |         with source_util.create_modules('_temp') as mapping: | 
					
						
							|  |  |  |             bc_path = self.manipulate_bytecode('_temp', mapping, | 
					
						
							|  |  |  |                                                 lambda bc: bc[:8], | 
					
						
							|  |  |  |                                                 del_source=del_source) | 
					
						
							|  |  |  |             file_path = mapping['_temp'] if not del_source else bc_path | 
					
						
							|  |  |  |             with self.assertRaises(EOFError): | 
					
						
							|  |  |  |                 self.import_(file_path, '_temp') | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def _test_non_code_marshal(self, *, del_source=False): | 
					
						
							|  |  |  |         with source_util.create_modules('_temp') as mapping: | 
					
						
							|  |  |  |             bytecode_path = self.manipulate_bytecode('_temp', mapping, | 
					
						
							|  |  |  |                                     lambda bc: bc[:8] + marshal.dumps(b'abcd'), | 
					
						
							|  |  |  |                                     del_source=del_source) | 
					
						
							|  |  |  |             file_path = mapping['_temp'] if not del_source else bytecode_path | 
					
						
							| 
									
										
										
										
											2010-02-19 16:01:06 +00:00
										 |  |  |             with self.assertRaises(ImportError): | 
					
						
							| 
									
										
										
										
											2010-07-03 21:48:25 +00:00
										 |  |  |                 self.import_(file_path, '_temp') | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def _test_bad_marshal(self, *, del_source=False): | 
					
						
							|  |  |  |         with source_util.create_modules('_temp') as mapping: | 
					
						
							|  |  |  |             bytecode_path = self.manipulate_bytecode('_temp', mapping, | 
					
						
							|  |  |  |                                                 lambda bc: bc[:8] + b'<test>', | 
					
						
							|  |  |  |                                                 del_source=del_source) | 
					
						
							|  |  |  |             file_path = mapping['_temp'] if not del_source else bytecode_path | 
					
						
							|  |  |  |             with self.assertRaises(ValueError): | 
					
						
							|  |  |  |                 self.import_(file_path, '_temp') | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def _test_bad_magic(self, test, *, del_source=False): | 
					
						
							|  |  |  |         with source_util.create_modules('_temp') as mapping: | 
					
						
							|  |  |  |             bc_path = self.manipulate_bytecode('_temp', mapping, | 
					
						
							|  |  |  |                                     lambda bc: b'\x00\x00\x00\x00' + bc[4:]) | 
					
						
							|  |  |  |             test('_temp', mapping, bc_path) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | class SourceLoaderBadBytecodeTest(BadBytecodeTest): | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     loader = _bootstrap._SourceFileLoader | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     @source_util.writes_bytecode_files | 
					
						
							|  |  |  |     def test_empty_file(self): | 
					
						
							|  |  |  |         # When a .pyc is empty, regenerate it if possible, else raise | 
					
						
							|  |  |  |         # ImportError. | 
					
						
							|  |  |  |         def test(name, mapping, bytecode_path): | 
					
						
							|  |  |  |             self.import_(mapping[name], name) | 
					
						
							|  |  |  |             with open(bytecode_path, 'rb') as file: | 
					
						
							|  |  |  |                 self.assertGreater(len(file.read()), 8) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         self._test_empty_file(test) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def test_partial_magic(self): | 
					
						
							|  |  |  |         def test(name, mapping, bytecode_path): | 
					
						
							|  |  |  |             self.import_(mapping[name], name) | 
					
						
							|  |  |  |             with open(bytecode_path, 'rb') as file: | 
					
						
							|  |  |  |                 self.assertGreater(len(file.read()), 8) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         self._test_partial_magic(test) | 
					
						
							| 
									
										
										
										
											2010-02-19 16:01:06 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     @source_util.writes_bytecode_files | 
					
						
							|  |  |  |     def test_magic_only(self): | 
					
						
							|  |  |  |         # When there is only the magic number, regenerate the .pyc if possible, | 
					
						
							|  |  |  |         # else raise EOFError. | 
					
						
							| 
									
										
										
										
											2010-07-03 21:48:25 +00:00
										 |  |  |         def test(name, mapping, bytecode_path): | 
					
						
							|  |  |  |             self.import_(mapping[name], name) | 
					
						
							|  |  |  |             with open(bytecode_path, 'rb') as file: | 
					
						
							| 
									
										
										
										
											2010-02-19 16:01:06 +00:00
										 |  |  |                 self.assertGreater(len(file.read()), 8) | 
					
						
							| 
									
										
										
										
											2010-07-03 21:48:25 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     @source_util.writes_bytecode_files | 
					
						
							|  |  |  |     def test_bad_magic(self): | 
					
						
							|  |  |  |         # When the magic number is different, the bytecode should be | 
					
						
							|  |  |  |         # regenerated. | 
					
						
							|  |  |  |         def test(name, mapping, bytecode_path): | 
					
						
							|  |  |  |             self.import_(mapping[name], name) | 
					
						
							|  |  |  |             with open(bytecode_path, 'rb') as bytecode_file: | 
					
						
							|  |  |  |                 self.assertEqual(bytecode_file.read(4), imp.get_magic()) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         self._test_bad_magic(test) | 
					
						
							| 
									
										
										
										
											2010-02-19 16:01:06 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     @source_util.writes_bytecode_files | 
					
						
							|  |  |  |     def test_partial_timestamp(self): | 
					
						
							|  |  |  |         # When the timestamp is partial, regenerate the .pyc, else | 
					
						
							|  |  |  |         # raise EOFError. | 
					
						
							| 
									
										
										
										
											2010-07-03 21:48:25 +00:00
										 |  |  |         def test(name, mapping, bc_path): | 
					
						
							|  |  |  |             self.import_(mapping[name], name) | 
					
						
							| 
									
										
										
										
											2010-02-19 16:01:06 +00:00
										 |  |  |             with open(bc_path, 'rb') as file: | 
					
						
							|  |  |  |                 self.assertGreater(len(file.read()), 8) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     @source_util.writes_bytecode_files | 
					
						
							|  |  |  |     def test_no_marshal(self): | 
					
						
							|  |  |  |         # When there is only the magic number and timestamp, raise EOFError. | 
					
						
							| 
									
										
										
										
											2010-07-03 21:48:25 +00:00
										 |  |  |         self._test_no_marshal() | 
					
						
							| 
									
										
										
										
											2010-02-19 16:01:06 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-05-11 01:47:11 +00:00
										 |  |  |     @source_util.writes_bytecode_files | 
					
						
							| 
									
										
										
										
											2010-07-03 21:48:25 +00:00
										 |  |  |     def test_non_code_marshal(self): | 
					
						
							|  |  |  |         self._test_non_code_marshal() | 
					
						
							|  |  |  |         # XXX ImportError when sourceless | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     # [bad marshal] | 
					
						
							|  |  |  |     @source_util.writes_bytecode_files | 
					
						
							|  |  |  |     def test_bad_marshal(self): | 
					
						
							|  |  |  |         # Bad marshal data should raise a ValueError. | 
					
						
							|  |  |  |         self._test_bad_marshal() | 
					
						
							| 
									
										
										
										
											2009-01-18 00:24:28 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     # [bad timestamp] | 
					
						
							| 
									
										
										
										
											2009-05-11 01:47:11 +00:00
										 |  |  |     @source_util.writes_bytecode_files | 
					
						
							| 
									
										
										
										
											2010-07-03 21:48:25 +00:00
										 |  |  |     def test_old_timestamp(self): | 
					
						
							| 
									
										
										
										
											2010-02-19 16:01:06 +00:00
										 |  |  |         # When the timestamp is older than the source, bytecode should be | 
					
						
							|  |  |  |         # regenerated. | 
					
						
							| 
									
										
										
										
											2009-01-18 00:24:28 +00:00
										 |  |  |         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']) | 
					
						
							| 
									
										
										
										
											2010-04-17 00:19:56 +00:00
										 |  |  |             bytecode_path = imp.cache_from_source(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) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-11-07 23:55:05 +00:00
										 |  |  |     # [bytecode read-only] | 
					
						
							|  |  |  |     @source_util.writes_bytecode_files | 
					
						
							|  |  |  |     def test_read_only_bytecode(self): | 
					
						
							| 
									
										
										
										
											2010-02-19 16:01:06 +00:00
										 |  |  |         # When bytecode is read-only but should be rewritten, fail silently. | 
					
						
							| 
									
										
										
										
											2009-11-07 23:55:05 +00:00
										 |  |  |         with source_util.create_modules('_temp') as mapping: | 
					
						
							|  |  |  |             # Create bytecode that will need to be re-created. | 
					
						
							|  |  |  |             py_compile.compile(mapping['_temp']) | 
					
						
							| 
									
										
										
										
											2010-04-17 00:19:56 +00:00
										 |  |  |             bytecode_path = imp.cache_from_source(mapping['_temp']) | 
					
						
							| 
									
										
										
										
											2009-11-07 23:55:05 +00:00
										 |  |  |             with open(bytecode_path, 'r+b') as bytecode_file: | 
					
						
							|  |  |  |                 bytecode_file.seek(0) | 
					
						
							|  |  |  |                 bytecode_file.write(b'\x00\x00\x00\x00') | 
					
						
							|  |  |  |             # Make the bytecode read-only. | 
					
						
							|  |  |  |             os.chmod(bytecode_path, | 
					
						
							|  |  |  |                         stat.S_IRUSR | stat.S_IRGRP | stat.S_IROTH) | 
					
						
							|  |  |  |             try: | 
					
						
							|  |  |  |                 # Should not raise IOError! | 
					
						
							|  |  |  |                 self.import_(mapping['_temp'], '_temp') | 
					
						
							|  |  |  |             finally: | 
					
						
							|  |  |  |                 # Make writable for eventual clean-up. | 
					
						
							|  |  |  |                 os.chmod(bytecode_path, stat.S_IWUSR) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-01-18 00:24:28 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-07-03 21:48:25 +00:00
										 |  |  | class SourcelessLoaderBadBytecodeTest(BadBytecodeTest): | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     loader = _bootstrap._SourcelessFileLoader | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def test_empty_file(self): | 
					
						
							|  |  |  |         def test(name, mapping, bytecode_path): | 
					
						
							|  |  |  |             with self.assertRaises(ImportError): | 
					
						
							|  |  |  |                 self.import_(bytecode_path, name) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         self._test_empty_file(test, del_source=True) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def test_partial_magic(self): | 
					
						
							|  |  |  |         def test(name, mapping, bytecode_path): | 
					
						
							|  |  |  |             with self.assertRaises(ImportError): | 
					
						
							|  |  |  |                 self.import_(bytecode_path, name) | 
					
						
							|  |  |  |         self._test_partial_magic(test, del_source=True) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def test_magic_only(self): | 
					
						
							|  |  |  |         def test(name, mapping, bytecode_path): | 
					
						
							|  |  |  |             with self.assertRaises(EOFError): | 
					
						
							|  |  |  |                 self.import_(bytecode_path, name) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         self._test_magic_only(test, del_source=True) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def test_bad_magic(self): | 
					
						
							|  |  |  |         def test(name, mapping, bytecode_path): | 
					
						
							|  |  |  |             with self.assertRaises(ImportError): | 
					
						
							|  |  |  |                 self.import_(bytecode_path, name) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         self._test_bad_magic(test, del_source=True) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def test_partial_timestamp(self): | 
					
						
							|  |  |  |         def test(name, mapping, bytecode_path): | 
					
						
							|  |  |  |             with self.assertRaises(EOFError): | 
					
						
							|  |  |  |                 self.import_(bytecode_path, name) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         self._test_partial_timestamp(test, del_source=True) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def test_no_marshal(self): | 
					
						
							|  |  |  |         self._test_no_marshal(del_source=True) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def test_non_code_marshal(self): | 
					
						
							|  |  |  |         self._test_non_code_marshal(del_source=True) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-01-18 00:24:28 +00:00
										 |  |  | def test_main(): | 
					
						
							|  |  |  |     from test.support import run_unittest | 
					
						
							| 
									
										
										
										
											2010-07-03 21:48:25 +00:00
										 |  |  |     run_unittest(SimpleTest, | 
					
						
							|  |  |  |                  SourceLoaderBadBytecodeTest, | 
					
						
							|  |  |  |                  SourcelessLoaderBadBytecodeTest | 
					
						
							|  |  |  |                  ) | 
					
						
							| 
									
										
										
										
											2009-01-18 00:24:28 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | if __name__ == '__main__': | 
					
						
							|  |  |  |     test_main() |