Brett Cannon | c049952 | 2012-05-11 14:48:41 -0400 | [diff] [blame] | 1 | import importlib |
Nick Coghlan | 9d3c61c | 2015-09-05 21:05:05 +1000 | [diff] [blame] | 2 | import importlib.util |
Guido van Rossum | 0ad59d4 | 2009-03-30 22:01:35 +0000 | [diff] [blame] | 3 | import os |
| 4 | import os.path |
Benjamin Peterson | b0274f2 | 2018-07-06 20:41:06 -0700 | [diff] [blame] | 5 | import py_compile |
Brett Cannon | 8a9583e | 2008-09-04 05:04:25 +0000 | [diff] [blame] | 6 | import sys |
Benjamin Peterson | ee8712c | 2008-05-20 21:35:26 +0000 | [diff] [blame] | 7 | from test import support |
Hai Shi | f7ba40b | 2020-06-25 18:38:51 +0800 | [diff] [blame] | 8 | from test.support import import_helper |
| 9 | from test.support import os_helper |
Benjamin Peterson | 42aa93b | 2017-12-09 10:26:52 -0800 | [diff] [blame] | 10 | from test.support import script_helper |
Brett Cannon | c049952 | 2012-05-11 14:48:41 -0400 | [diff] [blame] | 11 | import unittest |
| 12 | import warnings |
Brett Cannon | e4f41de | 2013-06-16 13:13:40 -0400 | [diff] [blame] | 13 | with warnings.catch_warnings(): |
Brett Cannon | 213b405 | 2015-10-30 14:41:06 -0700 | [diff] [blame] | 14 | warnings.simplefilter('ignore', DeprecationWarning) |
Brett Cannon | e4f41de | 2013-06-16 13:13:40 -0400 | [diff] [blame] | 15 | import imp |
Benjamin Peterson | 42aa93b | 2017-12-09 10:26:52 -0800 | [diff] [blame] | 16 | import _imp |
Neal Norwitz | 2294c0d | 2003-02-12 23:02:21 +0000 | [diff] [blame] | 17 | |
Brett Cannon | 130e481 | 2013-05-03 10:54:23 -0400 | [diff] [blame] | 18 | |
| 19 | def requires_load_dynamic(meth): |
| 20 | """Decorator to skip a test if not running under CPython or lacking |
| 21 | imp.load_dynamic().""" |
| 22 | meth = support.cpython_only(meth) |
| 23 | return unittest.skipIf(not hasattr(imp, 'load_dynamic'), |
| 24 | 'imp.load_dynamic() required')(meth) |
| 25 | |
| 26 | |
Thomas Wouters | 89f507f | 2006-12-13 04:49:30 +0000 | [diff] [blame] | 27 | class LockTests(unittest.TestCase): |
Tim Peters | 579bed7 | 2003-04-26 14:31:24 +0000 | [diff] [blame] | 28 | |
Thomas Wouters | 89f507f | 2006-12-13 04:49:30 +0000 | [diff] [blame] | 29 | """Very basic test of import lock functions.""" |
Tim Peters | 579bed7 | 2003-04-26 14:31:24 +0000 | [diff] [blame] | 30 | |
Thomas Wouters | 89f507f | 2006-12-13 04:49:30 +0000 | [diff] [blame] | 31 | def verify_lock_state(self, expected): |
Benjamin Peterson | c9c0f20 | 2009-06-30 23:06:06 +0000 | [diff] [blame] | 32 | self.assertEqual(imp.lock_held(), expected, |
Thomas Wouters | 89f507f | 2006-12-13 04:49:30 +0000 | [diff] [blame] | 33 | "expected imp.lock_held() to be %r" % expected) |
| 34 | def testLock(self): |
| 35 | LOOPS = 50 |
Tim Peters | 579bed7 | 2003-04-26 14:31:24 +0000 | [diff] [blame] | 36 | |
Thomas Wouters | 89f507f | 2006-12-13 04:49:30 +0000 | [diff] [blame] | 37 | # The import lock may already be held, e.g. if the test suite is run |
| 38 | # via "import test.autotest". |
| 39 | lock_held_at_start = imp.lock_held() |
| 40 | self.verify_lock_state(lock_held_at_start) |
Tim Peters | 579bed7 | 2003-04-26 14:31:24 +0000 | [diff] [blame] | 41 | |
Thomas Wouters | 89f507f | 2006-12-13 04:49:30 +0000 | [diff] [blame] | 42 | for i in range(LOOPS): |
| 43 | imp.acquire_lock() |
| 44 | self.verify_lock_state(True) |
Tim Peters | 579bed7 | 2003-04-26 14:31:24 +0000 | [diff] [blame] | 45 | |
Thomas Wouters | 89f507f | 2006-12-13 04:49:30 +0000 | [diff] [blame] | 46 | for i in range(LOOPS): |
Neal Norwitz | 2294c0d | 2003-02-12 23:02:21 +0000 | [diff] [blame] | 47 | imp.release_lock() |
Thomas Wouters | 89f507f | 2006-12-13 04:49:30 +0000 | [diff] [blame] | 48 | |
| 49 | # The original state should be restored now. |
| 50 | self.verify_lock_state(lock_held_at_start) |
| 51 | |
| 52 | if not lock_held_at_start: |
| 53 | try: |
| 54 | imp.release_lock() |
| 55 | except RuntimeError: |
| 56 | pass |
| 57 | else: |
| 58 | self.fail("release_lock() without lock should raise " |
| 59 | "RuntimeError") |
Neal Norwitz | 2294c0d | 2003-02-12 23:02:21 +0000 | [diff] [blame] | 60 | |
Guido van Rossum | ce3a72a | 2007-10-19 23:16:50 +0000 | [diff] [blame] | 61 | class ImportTests(unittest.TestCase): |
Alexander Belopolsky | e8f5832 | 2010-10-15 16:28:20 +0000 | [diff] [blame] | 62 | def setUp(self): |
| 63 | mod = importlib.import_module('test.encoded_modules') |
| 64 | self.test_strings = mod.test_strings |
| 65 | self.test_path = mod.__path__ |
| 66 | |
| 67 | def test_import_encoded_module(self): |
| 68 | for modname, encoding, teststr in self.test_strings: |
| 69 | mod = importlib.import_module('test.encoded_modules.' |
| 70 | 'module_' + modname) |
| 71 | self.assertEqual(teststr, mod.test) |
Guido van Rossum | ce3a72a | 2007-10-19 23:16:50 +0000 | [diff] [blame] | 72 | |
| 73 | def test_find_module_encoding(self): |
Alexander Belopolsky | e8f5832 | 2010-10-15 16:28:20 +0000 | [diff] [blame] | 74 | for mod, encoding, _ in self.test_strings: |
Brett Cannon | 749afa9 | 2010-10-29 23:47:23 +0000 | [diff] [blame] | 75 | with imp.find_module('module_' + mod, self.test_path)[0] as fd: |
| 76 | self.assertEqual(fd.encoding, encoding) |
Guido van Rossum | ce3a72a | 2007-10-19 23:16:50 +0000 | [diff] [blame] | 77 | |
Victor Stinner | fe7c5b5 | 2011-04-05 01:48:03 +0200 | [diff] [blame] | 78 | path = [os.path.dirname(__file__)] |
Brett Cannon | dd9a569 | 2012-04-20 12:59:59 -0400 | [diff] [blame] | 79 | with self.assertRaises(SyntaxError): |
| 80 | imp.find_module('badsyntax_pep3120', path) |
Victor Stinner | fe7c5b5 | 2011-04-05 01:48:03 +0200 | [diff] [blame] | 81 | |
Guido van Rossum | 40d20bc | 2007-10-22 00:09:51 +0000 | [diff] [blame] | 82 | def test_issue1267(self): |
Alexander Belopolsky | e8f5832 | 2010-10-15 16:28:20 +0000 | [diff] [blame] | 83 | for mod, encoding, _ in self.test_strings: |
| 84 | fp, filename, info = imp.find_module('module_' + mod, |
| 85 | self.test_path) |
Brett Cannon | 749afa9 | 2010-10-29 23:47:23 +0000 | [diff] [blame] | 86 | with fp: |
| 87 | self.assertNotEqual(fp, None) |
| 88 | self.assertEqual(fp.encoding, encoding) |
| 89 | self.assertEqual(fp.tell(), 0) |
| 90 | self.assertEqual(fp.readline(), '# test %s encoding\n' |
| 91 | % encoding) |
Guido van Rossum | 40d20bc | 2007-10-22 00:09:51 +0000 | [diff] [blame] | 92 | |
| 93 | fp, filename, info = imp.find_module("tokenize") |
Brett Cannon | 749afa9 | 2010-10-29 23:47:23 +0000 | [diff] [blame] | 94 | with fp: |
| 95 | self.assertNotEqual(fp, None) |
| 96 | self.assertEqual(fp.encoding, "utf-8") |
| 97 | self.assertEqual(fp.tell(), 0) |
| 98 | self.assertEqual(fp.readline(), |
| 99 | '"""Tokenization help for Python programs.\n') |
Guido van Rossum | 40d20bc | 2007-10-22 00:09:51 +0000 | [diff] [blame] | 100 | |
Brett Cannon | 8a9583e | 2008-09-04 05:04:25 +0000 | [diff] [blame] | 101 | def test_issue3594(self): |
| 102 | temp_mod_name = 'test_imp_helper' |
| 103 | sys.path.insert(0, '.') |
| 104 | try: |
Inada Naoki | c8e5eb9 | 2021-04-05 13:11:23 +0900 | [diff] [blame^] | 105 | with open(temp_mod_name + '.py', 'w', encoding="latin-1") as file: |
Brett Cannon | 8a9583e | 2008-09-04 05:04:25 +0000 | [diff] [blame] | 106 | file.write("# coding: cp1252\nu = 'test.test_imp'\n") |
| 107 | file, filename, info = imp.find_module(temp_mod_name) |
| 108 | file.close() |
Ezio Melotti | b3aedd4 | 2010-11-20 19:04:17 +0000 | [diff] [blame] | 109 | self.assertEqual(file.encoding, 'cp1252') |
Brett Cannon | 8a9583e | 2008-09-04 05:04:25 +0000 | [diff] [blame] | 110 | finally: |
| 111 | del sys.path[0] |
Hai Shi | f7ba40b | 2020-06-25 18:38:51 +0800 | [diff] [blame] | 112 | os_helper.unlink(temp_mod_name + '.py') |
| 113 | os_helper.unlink(temp_mod_name + '.pyc') |
Brett Cannon | 8a9583e | 2008-09-04 05:04:25 +0000 | [diff] [blame] | 114 | |
Guido van Rossum | 0ad59d4 | 2009-03-30 22:01:35 +0000 | [diff] [blame] | 115 | def test_issue5604(self): |
| 116 | # Test cannot cover imp.load_compiled function. |
| 117 | # Martin von Loewis note what shared library cannot have non-ascii |
| 118 | # character because init_xxx function cannot be compiled |
| 119 | # and issue never happens for dynamic modules. |
luzpaz | a5293b4 | 2017-11-05 07:37:50 -0600 | [diff] [blame] | 120 | # But sources modified to follow generic way for processing paths. |
Guido van Rossum | 0ad59d4 | 2009-03-30 22:01:35 +0000 | [diff] [blame] | 121 | |
Ezio Melotti | 435b531 | 2010-03-06 01:20:49 +0000 | [diff] [blame] | 122 | # the return encoding could be uppercase or None |
| 123 | fs_encoding = sys.getfilesystemencoding() |
Guido van Rossum | 0ad59d4 | 2009-03-30 22:01:35 +0000 | [diff] [blame] | 124 | |
| 125 | # covers utf-8 and Windows ANSI code pages |
| 126 | # one non-space symbol from every page |
| 127 | # (http://en.wikipedia.org/wiki/Code_page) |
| 128 | known_locales = { |
Ezio Melotti | 9a7d5ac | 2010-03-05 12:43:17 +0000 | [diff] [blame] | 129 | 'utf-8' : b'\xc3\xa4', |
Guido van Rossum | 0ad59d4 | 2009-03-30 22:01:35 +0000 | [diff] [blame] | 130 | 'cp1250' : b'\x8C', |
| 131 | 'cp1251' : b'\xc0', |
| 132 | 'cp1252' : b'\xc0', |
| 133 | 'cp1253' : b'\xc1', |
| 134 | 'cp1254' : b'\xc0', |
| 135 | 'cp1255' : b'\xe0', |
| 136 | 'cp1256' : b'\xe0', |
| 137 | 'cp1257' : b'\xc0', |
| 138 | 'cp1258' : b'\xc0', |
| 139 | } |
| 140 | |
Florent Xicluna | 21164ce | 2010-03-20 20:30:53 +0000 | [diff] [blame] | 141 | if sys.platform == 'darwin': |
| 142 | self.assertEqual(fs_encoding, 'utf-8') |
| 143 | # Mac OS X uses the Normal Form D decomposition |
| 144 | # http://developer.apple.com/mac/library/qa/qa2001/qa1173.html |
| 145 | special_char = b'a\xcc\x88' |
| 146 | else: |
| 147 | special_char = known_locales.get(fs_encoding) |
| 148 | |
Ezio Melotti | 9a7d5ac | 2010-03-05 12:43:17 +0000 | [diff] [blame] | 149 | if not special_char: |
Ezio Melotti | 76e0d1a | 2010-03-05 15:08:19 +0000 | [diff] [blame] | 150 | self.skipTest("can't run this test with %s as filesystem encoding" |
| 151 | % fs_encoding) |
| 152 | decoded_char = special_char.decode(fs_encoding) |
Ezio Melotti | 9a7d5ac | 2010-03-05 12:43:17 +0000 | [diff] [blame] | 153 | temp_mod_name = 'test_imp_helper_' + decoded_char |
| 154 | test_package_name = 'test_imp_helper_package_' + decoded_char |
| 155 | init_file_name = os.path.join(test_package_name, '__init__.py') |
| 156 | try: |
Ezio Melotti | 41a6b04 | 2010-03-06 01:50:25 +0000 | [diff] [blame] | 157 | # if the curdir is not in sys.path the test fails when run with |
| 158 | # ./python ./Lib/test/regrtest.py test_imp |
| 159 | sys.path.insert(0, os.curdir) |
Inada Naoki | c8e5eb9 | 2021-04-05 13:11:23 +0900 | [diff] [blame^] | 160 | with open(temp_mod_name + '.py', 'w', encoding="utf-8") as file: |
Ezio Melotti | 9a7d5ac | 2010-03-05 12:43:17 +0000 | [diff] [blame] | 161 | file.write('a = 1\n') |
| 162 | file, filename, info = imp.find_module(temp_mod_name) |
Brett Cannon | 749afa9 | 2010-10-29 23:47:23 +0000 | [diff] [blame] | 163 | with file: |
| 164 | self.assertIsNotNone(file) |
| 165 | self.assertTrue(filename[:-3].endswith(temp_mod_name)) |
| 166 | self.assertEqual(info[0], '.py') |
Serhiy Storchaka | 6787a38 | 2013-11-23 22:12:06 +0200 | [diff] [blame] | 167 | self.assertEqual(info[1], 'r') |
Brett Cannon | 749afa9 | 2010-10-29 23:47:23 +0000 | [diff] [blame] | 168 | self.assertEqual(info[2], imp.PY_SOURCE) |
Guido van Rossum | 0ad59d4 | 2009-03-30 22:01:35 +0000 | [diff] [blame] | 169 | |
Brett Cannon | 749afa9 | 2010-10-29 23:47:23 +0000 | [diff] [blame] | 170 | mod = imp.load_module(temp_mod_name, file, filename, info) |
| 171 | self.assertEqual(mod.a, 1) |
Guido van Rossum | 0ad59d4 | 2009-03-30 22:01:35 +0000 | [diff] [blame] | 172 | |
Brett Cannon | c049952 | 2012-05-11 14:48:41 -0400 | [diff] [blame] | 173 | with warnings.catch_warnings(): |
| 174 | warnings.simplefilter('ignore') |
| 175 | mod = imp.load_source(temp_mod_name, temp_mod_name + '.py') |
Ezio Melotti | 435b531 | 2010-03-06 01:20:49 +0000 | [diff] [blame] | 176 | self.assertEqual(mod.a, 1) |
Guido van Rossum | 0ad59d4 | 2009-03-30 22:01:35 +0000 | [diff] [blame] | 177 | |
Brett Cannon | c049952 | 2012-05-11 14:48:41 -0400 | [diff] [blame] | 178 | with warnings.catch_warnings(): |
| 179 | warnings.simplefilter('ignore') |
Ezio Melotti | e5e7a7c | 2013-03-16 21:49:20 +0200 | [diff] [blame] | 180 | if not sys.dont_write_bytecode: |
| 181 | mod = imp.load_compiled( |
| 182 | temp_mod_name, |
| 183 | imp.cache_from_source(temp_mod_name + '.py')) |
Ezio Melotti | 435b531 | 2010-03-06 01:20:49 +0000 | [diff] [blame] | 184 | self.assertEqual(mod.a, 1) |
Guido van Rossum | 0ad59d4 | 2009-03-30 22:01:35 +0000 | [diff] [blame] | 185 | |
Ezio Melotti | 9a7d5ac | 2010-03-05 12:43:17 +0000 | [diff] [blame] | 186 | if not os.path.exists(test_package_name): |
| 187 | os.mkdir(test_package_name) |
Inada Naoki | c8e5eb9 | 2021-04-05 13:11:23 +0900 | [diff] [blame^] | 188 | with open(init_file_name, 'w', encoding="utf-8") as file: |
Ezio Melotti | 9a7d5ac | 2010-03-05 12:43:17 +0000 | [diff] [blame] | 189 | file.write('b = 2\n') |
Brett Cannon | c049952 | 2012-05-11 14:48:41 -0400 | [diff] [blame] | 190 | with warnings.catch_warnings(): |
| 191 | warnings.simplefilter('ignore') |
| 192 | package = imp.load_package(test_package_name, test_package_name) |
Ezio Melotti | 435b531 | 2010-03-06 01:20:49 +0000 | [diff] [blame] | 193 | self.assertEqual(package.b, 2) |
Ezio Melotti | 9a7d5ac | 2010-03-05 12:43:17 +0000 | [diff] [blame] | 194 | finally: |
Ezio Melotti | 41a6b04 | 2010-03-06 01:50:25 +0000 | [diff] [blame] | 195 | del sys.path[0] |
Brett Cannon | f299abd | 2015-04-13 14:21:02 -0400 | [diff] [blame] | 196 | for ext in ('.py', '.pyc'): |
Hai Shi | f7ba40b | 2020-06-25 18:38:51 +0800 | [diff] [blame] | 197 | os_helper.unlink(temp_mod_name + ext) |
| 198 | os_helper.unlink(init_file_name + ext) |
| 199 | os_helper.rmtree(test_package_name) |
| 200 | os_helper.rmtree('__pycache__') |
Guido van Rossum | 0ad59d4 | 2009-03-30 22:01:35 +0000 | [diff] [blame] | 201 | |
Victor Stinner | c68b6aa | 2011-04-23 00:41:19 +0200 | [diff] [blame] | 202 | def test_issue9319(self): |
Antoine Pitrou | 1184690 | 2011-04-25 21:39:49 +0200 | [diff] [blame] | 203 | path = os.path.dirname(__file__) |
Victor Stinner | 7fdd0fe | 2011-04-23 01:24:11 +0200 | [diff] [blame] | 204 | self.assertRaises(SyntaxError, |
Antoine Pitrou | 1184690 | 2011-04-25 21:39:49 +0200 | [diff] [blame] | 205 | imp.find_module, "badsyntax_pep3120", [path]) |
Victor Stinner | c68b6aa | 2011-04-23 00:41:19 +0200 | [diff] [blame] | 206 | |
Nick Coghlan | 91b9f13 | 2012-09-01 00:13:45 +1000 | [diff] [blame] | 207 | def test_load_from_source(self): |
| 208 | # Verify that the imp module can correctly load and find .py files |
Hai Shi | f7ba40b | 2020-06-25 18:38:51 +0800 | [diff] [blame] | 209 | # XXX (ncoghlan): It would be nice to use import_helper.CleanImport |
Nick Coghlan | 91b9f13 | 2012-09-01 00:13:45 +1000 | [diff] [blame] | 210 | # here, but that breaks because the os module registers some |
| 211 | # handlers in copy_reg on import. Since CleanImport doesn't |
| 212 | # revert that registration, the module is left in a broken |
| 213 | # state after reversion. Reinitialising the module contents |
| 214 | # and just reverting os.environ to its previous state is an OK |
| 215 | # workaround |
| 216 | orig_path = os.path |
| 217 | orig_getenv = os.getenv |
Hai Shi | f7ba40b | 2020-06-25 18:38:51 +0800 | [diff] [blame] | 218 | with os_helper.EnvironmentVarGuard(): |
Nick Coghlan | 91b9f13 | 2012-09-01 00:13:45 +1000 | [diff] [blame] | 219 | x = imp.find_module("os") |
| 220 | self.addCleanup(x[0].close) |
| 221 | new_os = imp.load_module("os", *x) |
| 222 | self.assertIs(os, new_os) |
| 223 | self.assertIs(orig_path, new_os.path) |
| 224 | self.assertIsNot(orig_getenv, new_os.getenv) |
| 225 | |
Brett Cannon | 130e481 | 2013-05-03 10:54:23 -0400 | [diff] [blame] | 226 | @requires_load_dynamic |
Nick Coghlan | 91b9f13 | 2012-09-01 00:13:45 +1000 | [diff] [blame] | 227 | def test_issue15828_load_extensions(self): |
| 228 | # Issue 15828 picked up that the adapter between the old imp API |
| 229 | # and importlib couldn't handle C extensions |
| 230 | example = "_heapq" |
| 231 | x = imp.find_module(example) |
Brett Cannon | 848cdfd | 2012-08-31 11:31:20 -0400 | [diff] [blame] | 232 | file_ = x[0] |
| 233 | if file_ is not None: |
| 234 | self.addCleanup(file_.close) |
Nick Coghlan | 91b9f13 | 2012-09-01 00:13:45 +1000 | [diff] [blame] | 235 | mod = imp.load_module(example, *x) |
| 236 | self.assertEqual(mod.__name__, example) |
| 237 | |
Brett Cannon | 130e481 | 2013-05-03 10:54:23 -0400 | [diff] [blame] | 238 | @requires_load_dynamic |
Andrew Svetlov | 6b2cbeb | 2012-12-14 17:04:59 +0200 | [diff] [blame] | 239 | def test_issue16421_multiple_modules_in_one_dll(self): |
| 240 | # Issue 16421: loading several modules from the same compiled file fails |
| 241 | m = '_testimportmultiple' |
| 242 | fileobj, pathname, description = imp.find_module(m) |
| 243 | fileobj.close() |
| 244 | mod0 = imp.load_dynamic(m, pathname) |
Andrew Svetlov | ef9a43b | 2012-12-15 17:22:59 +0200 | [diff] [blame] | 245 | mod1 = imp.load_dynamic('_testimportmultiple_foo', pathname) |
| 246 | mod2 = imp.load_dynamic('_testimportmultiple_bar', pathname) |
Andrew Svetlov | 6b2cbeb | 2012-12-14 17:04:59 +0200 | [diff] [blame] | 247 | self.assertEqual(mod0.__name__, m) |
Andrew Svetlov | ef9a43b | 2012-12-15 17:22:59 +0200 | [diff] [blame] | 248 | self.assertEqual(mod1.__name__, '_testimportmultiple_foo') |
| 249 | self.assertEqual(mod2.__name__, '_testimportmultiple_bar') |
Andrew Svetlov | 6b2cbeb | 2012-12-14 17:04:59 +0200 | [diff] [blame] | 250 | with self.assertRaises(ImportError): |
| 251 | imp.load_dynamic('nonexistent', pathname) |
| 252 | |
Brett Cannon | 130e481 | 2013-05-03 10:54:23 -0400 | [diff] [blame] | 253 | @requires_load_dynamic |
Brett Cannon | f0434e6 | 2012-04-20 15:22:50 -0400 | [diff] [blame] | 254 | def test_load_dynamic_ImportError_path(self): |
| 255 | # Issue #1559549 added `name` and `path` attributes to ImportError |
| 256 | # in order to provide better detail. Issue #10854 implemented those |
| 257 | # attributes on import failures of extensions on Windows. |
| 258 | path = 'bogus file path' |
| 259 | name = 'extension' |
| 260 | with self.assertRaises(ImportError) as err: |
| 261 | imp.load_dynamic(name, path) |
| 262 | self.assertIn(path, err.exception.path) |
| 263 | self.assertEqual(name, err.exception.name) |
| 264 | |
Brett Cannon | 130e481 | 2013-05-03 10:54:23 -0400 | [diff] [blame] | 265 | @requires_load_dynamic |
Brett Cannon | 9d0f772 | 2013-05-03 10:37:08 -0400 | [diff] [blame] | 266 | def test_load_module_extension_file_is_None(self): |
| 267 | # When loading an extension module and the file is None, open one |
| 268 | # on the behalf of imp.load_dynamic(). |
| 269 | # Issue #15902 |
Brett Cannon | 8772b18 | 2013-05-04 17:54:57 -0400 | [diff] [blame] | 270 | name = '_testimportmultiple' |
Brett Cannon | 9d0f772 | 2013-05-03 10:37:08 -0400 | [diff] [blame] | 271 | found = imp.find_module(name) |
Benjamin Peterson | aa6f688 | 2013-05-11 16:29:03 -0500 | [diff] [blame] | 272 | if found[0] is not None: |
| 273 | found[0].close() |
Brett Cannon | 8772b18 | 2013-05-04 17:54:57 -0400 | [diff] [blame] | 274 | if found[2][2] != imp.C_EXTENSION: |
Zachary Ware | 9fe6d86 | 2013-12-08 00:20:35 -0600 | [diff] [blame] | 275 | self.skipTest("found module doesn't appear to be a C extension") |
Brett Cannon | 9d0f772 | 2013-05-03 10:37:08 -0400 | [diff] [blame] | 276 | imp.load_module(name, None, *found[1:]) |
| 277 | |
Nick Coghlan | 9d3c61c | 2015-09-05 21:05:05 +1000 | [diff] [blame] | 278 | @requires_load_dynamic |
| 279 | def test_issue24748_load_module_skips_sys_modules_check(self): |
| 280 | name = 'test.imp_dummy' |
| 281 | try: |
| 282 | del sys.modules[name] |
| 283 | except KeyError: |
| 284 | pass |
| 285 | try: |
| 286 | module = importlib.import_module(name) |
| 287 | spec = importlib.util.find_spec('_testmultiphase') |
| 288 | module = imp.load_dynamic(name, spec.origin) |
| 289 | self.assertEqual(module.__name__, name) |
| 290 | self.assertEqual(module.__spec__.name, name) |
| 291 | self.assertEqual(module.__spec__.origin, spec.origin) |
| 292 | self.assertRaises(AttributeError, getattr, module, 'dummy_name') |
| 293 | self.assertEqual(module.int_const, 1969) |
| 294 | self.assertIs(sys.modules[name], module) |
| 295 | finally: |
| 296 | try: |
| 297 | del sys.modules[name] |
| 298 | except KeyError: |
| 299 | pass |
| 300 | |
Brett Cannon | 997487d | 2013-06-07 13:26:53 -0400 | [diff] [blame] | 301 | @unittest.skipIf(sys.dont_write_bytecode, |
| 302 | "test meaningful only when writing bytecode") |
| 303 | def test_bug7732(self): |
Hai Shi | f7ba40b | 2020-06-25 18:38:51 +0800 | [diff] [blame] | 304 | with os_helper.temp_cwd(): |
| 305 | source = os_helper.TESTFN + '.py' |
Antoine Pitrou | bb2c45e | 2013-08-19 23:31:18 +0200 | [diff] [blame] | 306 | os.mkdir(source) |
Brett Cannon | 997487d | 2013-06-07 13:26:53 -0400 | [diff] [blame] | 307 | self.assertRaisesRegex(ImportError, '^No module', |
Hai Shi | f7ba40b | 2020-06-25 18:38:51 +0800 | [diff] [blame] | 308 | imp.find_module, os_helper.TESTFN, ["."]) |
Brett Cannon | 330cc52 | 2013-08-23 12:10:09 -0400 | [diff] [blame] | 309 | |
Brett Cannon | a4975a9 | 2013-08-23 11:45:57 -0400 | [diff] [blame] | 310 | def test_multiple_calls_to_get_data(self): |
| 311 | # Issue #18755: make sure multiple calls to get_data() can succeed. |
| 312 | loader = imp._LoadSourceCompatibility('imp', imp.__file__, |
Inada Naoki | c8e5eb9 | 2021-04-05 13:11:23 +0900 | [diff] [blame^] | 313 | open(imp.__file__, encoding="utf-8")) |
Brett Cannon | a4975a9 | 2013-08-23 11:45:57 -0400 | [diff] [blame] | 314 | loader.get_data(imp.__file__) # File should be closed |
| 315 | loader.get_data(imp.__file__) # Will need to create a newly opened file |
Brett Cannon | 997487d | 2013-06-07 13:26:53 -0400 | [diff] [blame] | 316 | |
Serhiy Storchaka | f7eae0a | 2017-06-28 08:30:06 +0300 | [diff] [blame] | 317 | def test_load_source(self): |
Victor Stinner | a505ecd | 2017-10-13 13:47:49 -0700 | [diff] [blame] | 318 | # Create a temporary module since load_source(name) modifies |
| 319 | # sys.modules[name] attributes like __loader___ |
| 320 | modname = f"tmp{__name__}" |
| 321 | mod = type(sys.modules[__name__])(modname) |
| 322 | with support.swap_item(sys.modules, modname, mod): |
| 323 | with self.assertRaisesRegex(ValueError, 'embedded null'): |
| 324 | imp.load_source(modname, __file__ + "\0") |
Serhiy Storchaka | f7eae0a | 2017-06-28 08:30:06 +0300 | [diff] [blame] | 325 | |
Oren Milman | 9974e1b | 2017-09-19 14:39:47 +0300 | [diff] [blame] | 326 | @support.cpython_only |
| 327 | def test_issue31315(self): |
| 328 | # There shouldn't be an assertion failure in imp.create_dynamic(), |
| 329 | # when spec.name is not a string. |
| 330 | create_dynamic = support.get_attribute(imp, 'create_dynamic') |
| 331 | class BadSpec: |
| 332 | name = None |
| 333 | origin = 'foo' |
| 334 | with self.assertRaises(TypeError): |
| 335 | create_dynamic(BadSpec()) |
| 336 | |
Nina Zakharenko | 69091cb | 2019-02-04 16:56:26 -0800 | [diff] [blame] | 337 | def test_issue_35321(self): |
| 338 | # Both _frozen_importlib and _frozen_importlib_external |
| 339 | # should have a spec origin of "frozen" and |
| 340 | # no need to clean up imports in this case. |
| 341 | |
| 342 | import _frozen_importlib_external |
| 343 | self.assertEqual(_frozen_importlib_external.__spec__.origin, "frozen") |
| 344 | |
| 345 | import _frozen_importlib |
| 346 | self.assertEqual(_frozen_importlib.__spec__.origin, "frozen") |
| 347 | |
Benjamin Peterson | 42aa93b | 2017-12-09 10:26:52 -0800 | [diff] [blame] | 348 | def test_source_hash(self): |
| 349 | self.assertEqual(_imp.source_hash(42, b'hi'), b'\xc6\xe7Z\r\x03:}\xab') |
| 350 | self.assertEqual(_imp.source_hash(43, b'hi'), b'\x85\x9765\xf8\x9a\x8b9') |
| 351 | |
| 352 | def test_pyc_invalidation_mode_from_cmdline(self): |
| 353 | cases = [ |
| 354 | ([], "default"), |
| 355 | (["--check-hash-based-pycs", "default"], "default"), |
| 356 | (["--check-hash-based-pycs", "always"], "always"), |
| 357 | (["--check-hash-based-pycs", "never"], "never"), |
| 358 | ] |
| 359 | for interp_args, expected in cases: |
| 360 | args = interp_args + [ |
| 361 | "-c", |
| 362 | "import _imp; print(_imp.check_hash_based_pycs)", |
| 363 | ] |
| 364 | res = script_helper.assert_python_ok(*args) |
| 365 | self.assertEqual(res.out.strip().decode('utf-8'), expected) |
| 366 | |
Benjamin Peterson | b0274f2 | 2018-07-06 20:41:06 -0700 | [diff] [blame] | 367 | def test_find_and_load_checked_pyc(self): |
| 368 | # issue 34056 |
Hai Shi | f7ba40b | 2020-06-25 18:38:51 +0800 | [diff] [blame] | 369 | with os_helper.temp_cwd(): |
Benjamin Peterson | b0274f2 | 2018-07-06 20:41:06 -0700 | [diff] [blame] | 370 | with open('mymod.py', 'wb') as fp: |
| 371 | fp.write(b'x = 42\n') |
| 372 | py_compile.compile( |
| 373 | 'mymod.py', |
| 374 | doraise=True, |
| 375 | invalidation_mode=py_compile.PycInvalidationMode.CHECKED_HASH, |
| 376 | ) |
| 377 | file, path, description = imp.find_module('mymod', path=['.']) |
| 378 | mod = imp.load_module('mymod', file, path, description) |
| 379 | self.assertEqual(mod.x, 42) |
| 380 | |
Guido van Rossum | 0ad59d4 | 2009-03-30 22:01:35 +0000 | [diff] [blame] | 381 | |
Nick Coghlan | 6ead552 | 2009-10-18 13:19:33 +0000 | [diff] [blame] | 382 | class ReloadTests(unittest.TestCase): |
| 383 | |
| 384 | """Very basic tests to make sure that imp.reload() operates just like |
| 385 | reload().""" |
| 386 | |
| 387 | def test_source(self): |
Hai Shi | f7ba40b | 2020-06-25 18:38:51 +0800 | [diff] [blame] | 388 | # XXX (ncoghlan): It would be nice to use test.import_helper.CleanImport |
Nick Coghlan | 6ead552 | 2009-10-18 13:19:33 +0000 | [diff] [blame] | 389 | # here, but that breaks because the os module registers some |
| 390 | # handlers in copy_reg on import. Since CleanImport doesn't |
| 391 | # revert that registration, the module is left in a broken |
| 392 | # state after reversion. Reinitialising the module contents |
| 393 | # and just reverting os.environ to its previous state is an OK |
| 394 | # workaround |
Hai Shi | f7ba40b | 2020-06-25 18:38:51 +0800 | [diff] [blame] | 395 | with os_helper.EnvironmentVarGuard(): |
Nick Coghlan | 6ead552 | 2009-10-18 13:19:33 +0000 | [diff] [blame] | 396 | import os |
| 397 | imp.reload(os) |
| 398 | |
| 399 | def test_extension(self): |
Hai Shi | f7ba40b | 2020-06-25 18:38:51 +0800 | [diff] [blame] | 400 | with import_helper.CleanImport('time'): |
Nick Coghlan | 6ead552 | 2009-10-18 13:19:33 +0000 | [diff] [blame] | 401 | import time |
| 402 | imp.reload(time) |
| 403 | |
| 404 | def test_builtin(self): |
Hai Shi | f7ba40b | 2020-06-25 18:38:51 +0800 | [diff] [blame] | 405 | with import_helper.CleanImport('marshal'): |
Nick Coghlan | 6ead552 | 2009-10-18 13:19:33 +0000 | [diff] [blame] | 406 | import marshal |
| 407 | imp.reload(marshal) |
Christian Heimes | 13a7a21 | 2008-01-07 17:13:09 +0000 | [diff] [blame] | 408 | |
Ezio Melotti | 056bafe | 2013-08-10 19:59:36 +0300 | [diff] [blame] | 409 | def test_with_deleted_parent(self): |
| 410 | # see #18681 |
| 411 | from html import parser |
Serhiy Storchaka | b212291 | 2013-08-11 20:12:20 +0300 | [diff] [blame] | 412 | html = sys.modules.pop('html') |
| 413 | def cleanup(): |
| 414 | sys.modules['html'] = html |
Ezio Melotti | 056bafe | 2013-08-10 19:59:36 +0300 | [diff] [blame] | 415 | self.addCleanup(cleanup) |
| 416 | with self.assertRaisesRegex(ImportError, 'html'): |
| 417 | imp.reload(parser) |
| 418 | |
Guido van Rossum | 40d20bc | 2007-10-22 00:09:51 +0000 | [diff] [blame] | 419 | |
Barry Warsaw | 28a691b | 2010-04-17 00:19:56 +0000 | [diff] [blame] | 420 | class PEP3147Tests(unittest.TestCase): |
| 421 | """Tests of PEP 3147.""" |
| 422 | |
| 423 | tag = imp.get_tag() |
| 424 | |
Brett Cannon | 19a2f59 | 2012-07-09 13:58:07 -0400 | [diff] [blame] | 425 | @unittest.skipUnless(sys.implementation.cache_tag is not None, |
| 426 | 'requires sys.implementation.cache_tag not be None') |
Barry Warsaw | 28a691b | 2010-04-17 00:19:56 +0000 | [diff] [blame] | 427 | def test_cache_from_source(self): |
| 428 | # Given the path to a .py file, return the path to its PEP 3147 |
| 429 | # defined .pyc file (i.e. under __pycache__). |
Brett Cannon | 410e88d | 2012-04-22 13:29:47 -0400 | [diff] [blame] | 430 | path = os.path.join('foo', 'bar', 'baz', 'qux.py') |
| 431 | expect = os.path.join('foo', 'bar', 'baz', '__pycache__', |
| 432 | 'qux.{}.pyc'.format(self.tag)) |
| 433 | self.assertEqual(imp.cache_from_source(path, True), expect) |
| 434 | |
Brett Cannon | 19a2f59 | 2012-07-09 13:58:07 -0400 | [diff] [blame] | 435 | @unittest.skipUnless(sys.implementation.cache_tag is not None, |
| 436 | 'requires sys.implementation.cache_tag to not be ' |
| 437 | 'None') |
Barry Warsaw | 28a691b | 2010-04-17 00:19:56 +0000 | [diff] [blame] | 438 | def test_source_from_cache(self): |
| 439 | # Given the path to a PEP 3147 defined .pyc file, return the path to |
| 440 | # its source. This tests the good path. |
Brett Cannon | 410e88d | 2012-04-22 13:29:47 -0400 | [diff] [blame] | 441 | path = os.path.join('foo', 'bar', 'baz', '__pycache__', |
| 442 | 'qux.{}.pyc'.format(self.tag)) |
| 443 | expect = os.path.join('foo', 'bar', 'baz', 'qux.py') |
| 444 | self.assertEqual(imp.source_from_cache(path), expect) |
Barry Warsaw | 28a691b | 2010-04-17 00:19:56 +0000 | [diff] [blame] | 445 | |
Barry Warsaw | 28a691b | 2010-04-17 00:19:56 +0000 | [diff] [blame] | 446 | |
Victor Stinner | 1a4d12d | 2010-08-13 13:07:29 +0000 | [diff] [blame] | 447 | class NullImporterTests(unittest.TestCase): |
Hai Shi | f7ba40b | 2020-06-25 18:38:51 +0800 | [diff] [blame] | 448 | @unittest.skipIf(os_helper.TESTFN_UNENCODABLE is None, |
Victor Stinner | 1a4d12d | 2010-08-13 13:07:29 +0000 | [diff] [blame] | 449 | "Need an undecodeable filename") |
| 450 | def test_unencodeable(self): |
Hai Shi | f7ba40b | 2020-06-25 18:38:51 +0800 | [diff] [blame] | 451 | name = os_helper.TESTFN_UNENCODABLE |
Victor Stinner | 1a4d12d | 2010-08-13 13:07:29 +0000 | [diff] [blame] | 452 | os.mkdir(name) |
| 453 | try: |
| 454 | self.assertRaises(ImportError, imp.NullImporter, name) |
| 455 | finally: |
| 456 | os.rmdir(name) |
| 457 | |
| 458 | |
Neal Norwitz | 2294c0d | 2003-02-12 23:02:21 +0000 | [diff] [blame] | 459 | if __name__ == "__main__": |
Brett Cannon | 95ea11f | 2013-05-03 10:57:08 -0400 | [diff] [blame] | 460 | unittest.main() |