blob: d2758b4d68dfdc1ea3a1f3f33bd2b86e356e2ca7 [file] [log] [blame]
Just van Rossum52e14d62002-12-30 22:08:05 +00001import sys
2import os
3import marshal
4import imp
5import struct
6import time
Neal Norwitzb155b622006-01-23 07:52:13 +00007import unittest
Just van Rossum52e14d62002-12-30 22:08:05 +00008
9import zlib # implied prerequisite
10from zipfile import ZipFile, ZipInfo, ZIP_STORED, ZIP_DEFLATED
11from test import test_support
12from test.test_importhooks import ImportHooksBaseTestCase, test_src, test_co
13
14import zipimport
Thomas Wouters49fd7fa2006-04-21 10:40:58 +000015import linecache
16import doctest
17import inspect
Guido van Rossum34d19282007-08-09 01:03:29 +000018import io
Thomas Wouters49fd7fa2006-04-21 10:40:58 +000019from traceback import extract_tb, extract_stack, print_tb
20raise_src = 'def do_raise(): raise TypeError\n'
Just van Rossum52e14d62002-12-30 22:08:05 +000021
Neal Norwitzb155b622006-01-23 07:52:13 +000022# so we only run testAFakeZlib once if this test is run repeatedly
23# which happens when we look for ref leaks
24test_imported = False
25
26
Just van Rossum52e14d62002-12-30 22:08:05 +000027def make_pyc(co, mtime):
28 data = marshal.dumps(co)
Jack Jansen472e7db2003-01-08 16:37:03 +000029 if type(mtime) is type(0.0):
Tim Petersf2715e02003-02-19 02:35:07 +000030 # Mac mtimes need a bit of special casing
31 if mtime < 0x7fffffff:
32 mtime = int(mtime)
33 else:
Guido van Rossume2a383d2007-01-15 16:59:06 +000034 mtime = int(-0x100000000 + int(mtime))
Jack Jansen472e7db2003-01-08 16:37:03 +000035 pyc = imp.get_magic() + struct.pack("<i", int(mtime)) + data
Just van Rossum52e14d62002-12-30 22:08:05 +000036 return pyc
37
Tim Peters68f2d002006-01-23 22:19:24 +000038def module_path_to_dotted_name(path):
39 return path.replace(os.sep, '.')
40
Just van Rossum52e14d62002-12-30 22:08:05 +000041NOW = time.time()
42test_pyc = make_pyc(test_co, NOW)
43
44
45if __debug__:
46 pyc_ext = ".pyc"
47else:
48 pyc_ext = ".pyo"
49
50
51TESTMOD = "ziptestmodule"
52TESTPACK = "ziptestpackage"
Just van Rossumd35c6db2003-01-02 12:55:48 +000053TESTPACK2 = "ziptestpackage2"
Skip Montanaro7a98be22007-08-16 14:35:24 +000054TEMP_ZIP = os.path.abspath("junk95142.zip")
Just van Rossum52e14d62002-12-30 22:08:05 +000055
56class UncompressedZipImportTestCase(ImportHooksBaseTestCase):
57
58 compression = ZIP_STORED
59
60 def setUp(self):
61 # We're reusing the zip archive path, so we must clear the
Thomas Wouters49fd7fa2006-04-21 10:40:58 +000062 # cached directory info and linecache
63 linecache.clearcache()
Just van Rossum52e14d62002-12-30 22:08:05 +000064 zipimport._zip_directory_cache.clear()
65 ImportHooksBaseTestCase.setUp(self)
66
Thomas Heller354e3d92003-07-22 18:10:15 +000067 def doTest(self, expected_ext, files, *modules, **kw):
Just van Rossum52e14d62002-12-30 22:08:05 +000068 z = ZipFile(TEMP_ZIP, "w")
69 try:
70 for name, (mtime, data) in files.items():
71 zinfo = ZipInfo(name, time.localtime(mtime))
72 zinfo.compress_type = self.compression
73 z.writestr(zinfo, data)
74 z.close()
Thomas Heller354e3d92003-07-22 18:10:15 +000075
76 stuff = kw.get("stuff", None)
77 if stuff is not None:
78 # Prepend 'stuff' to the start of the zipfile
79 f = open(TEMP_ZIP, "rb")
80 data = f.read()
81 f.close()
82
83 f = open(TEMP_ZIP, "wb")
84 f.write(stuff)
85 f.write(data)
86 f.close()
87
Just van Rossum52e14d62002-12-30 22:08:05 +000088 sys.path.insert(0, TEMP_ZIP)
89
90 mod = __import__(".".join(modules), globals(), locals(),
91 ["__dummy__"])
Thomas Wouters49fd7fa2006-04-21 10:40:58 +000092
93 call = kw.get('call')
94 if call is not None:
95 call(mod)
96
Just van Rossum9a3129c2003-01-03 11:18:56 +000097 if expected_ext:
98 file = mod.get_file()
99 self.assertEquals(file, os.path.join(TEMP_ZIP,
Just van Rossum6706c4d2003-01-09 22:27:10 +0000100 *modules) + expected_ext)
Just van Rossum52e14d62002-12-30 22:08:05 +0000101 finally:
102 z.close()
103 os.remove(TEMP_ZIP)
104
105 def testAFakeZlib(self):
106 #
107 # This could cause a stack overflow before: importing zlib.py
108 # from a compressed archive would cause zlib to be imported
109 # which would find zlib.py in the archive, which would... etc.
110 #
111 # This test *must* be executed first: it must be the first one
112 # to trigger zipimport to import zlib (zipimport caches the
113 # zlib.decompress function object, after which the problem being
114 # tested here wouldn't be a problem anymore...
115 # (Hence the 'A' in the test method name: to make it the first
116 # item in a list sorted by name, like unittest.makeSuite() does.)
117 #
Just van Rossum59498542003-11-18 23:00:55 +0000118 # This test fails on platforms on which the zlib module is
119 # statically linked, but the problem it tests for can't
120 # occur in that case (builtin modules are always found first),
121 # so we'll simply skip it then. Bug #765456.
122 #
123 if "zlib" in sys.builtin_module_names:
124 return
Just van Rossum52e14d62002-12-30 22:08:05 +0000125 if "zlib" in sys.modules:
126 del sys.modules["zlib"]
127 files = {"zlib.py": (NOW, test_src)}
128 try:
129 self.doTest(".py", files, "zlib")
130 except ImportError:
131 if self.compression != ZIP_DEFLATED:
132 self.fail("expected test to not raise ImportError")
133 else:
134 if self.compression != ZIP_STORED:
135 self.fail("expected test to raise ImportError")
136
137 def testPy(self):
138 files = {TESTMOD + ".py": (NOW, test_src)}
139 self.doTest(".py", files, TESTMOD)
140
141 def testPyc(self):
142 files = {TESTMOD + pyc_ext: (NOW, test_pyc)}
143 self.doTest(pyc_ext, files, TESTMOD)
144
145 def testBoth(self):
146 files = {TESTMOD + ".py": (NOW, test_src),
147 TESTMOD + pyc_ext: (NOW, test_pyc)}
148 self.doTest(pyc_ext, files, TESTMOD)
149
Just van Rossum9a3129c2003-01-03 11:18:56 +0000150 def testEmptyPy(self):
151 files = {TESTMOD + ".py": (NOW, "")}
152 self.doTest(None, files, TESTMOD)
153
Just van Rossum52e14d62002-12-30 22:08:05 +0000154 def testBadMagic(self):
155 # make pyc magic word invalid, forcing loading from .py
Guido van Rossum254348e2007-11-21 19:29:53 +0000156 badmagic_pyc = bytearray(test_pyc)
Guido van Rossumad8d3002007-08-03 18:40:49 +0000157 badmagic_pyc[0] ^= 0x04 # flip an arbitrary bit
Just van Rossum52e14d62002-12-30 22:08:05 +0000158 files = {TESTMOD + ".py": (NOW, test_src),
159 TESTMOD + pyc_ext: (NOW, badmagic_pyc)}
160 self.doTest(".py", files, TESTMOD)
161
162 def testBadMagic2(self):
163 # make pyc magic word invalid, causing an ImportError
Guido van Rossum254348e2007-11-21 19:29:53 +0000164 badmagic_pyc = bytearray(test_pyc)
Guido van Rossumad8d3002007-08-03 18:40:49 +0000165 badmagic_pyc[0] ^= 0x04 # flip an arbitrary bit
Just van Rossum52e14d62002-12-30 22:08:05 +0000166 files = {TESTMOD + pyc_ext: (NOW, badmagic_pyc)}
167 try:
168 self.doTest(".py", files, TESTMOD)
169 except ImportError:
170 pass
171 else:
172 self.fail("expected ImportError; import from bad pyc")
173
174 def testBadMTime(self):
Guido van Rossum254348e2007-11-21 19:29:53 +0000175 badtime_pyc = bytearray(test_pyc)
Guido van Rossumad8d3002007-08-03 18:40:49 +0000176 badtime_pyc[7] ^= 0x02 # flip the second bit -- not the first as that one
177 # isn't stored in the .py's mtime in the zip archive.
Just van Rossum52e14d62002-12-30 22:08:05 +0000178 files = {TESTMOD + ".py": (NOW, test_src),
179 TESTMOD + pyc_ext: (NOW, badtime_pyc)}
180 self.doTest(".py", files, TESTMOD)
181
182 def testPackage(self):
183 packdir = TESTPACK + os.sep
184 files = {packdir + "__init__" + pyc_ext: (NOW, test_pyc),
185 packdir + TESTMOD + pyc_ext: (NOW, test_pyc)}
186 self.doTest(pyc_ext, files, TESTPACK, TESTMOD)
187
188 def testDeepPackage(self):
189 packdir = TESTPACK + os.sep
Just van Rossumd35c6db2003-01-02 12:55:48 +0000190 packdir2 = packdir + TESTPACK2 + os.sep
Just van Rossum52e14d62002-12-30 22:08:05 +0000191 files = {packdir + "__init__" + pyc_ext: (NOW, test_pyc),
192 packdir2 + "__init__" + pyc_ext: (NOW, test_pyc),
193 packdir2 + TESTMOD + pyc_ext: (NOW, test_pyc)}
Just van Rossumd35c6db2003-01-02 12:55:48 +0000194 self.doTest(pyc_ext, files, TESTPACK, TESTPACK2, TESTMOD)
Just van Rossum52e14d62002-12-30 22:08:05 +0000195
Neal Norwitzb155b622006-01-23 07:52:13 +0000196 def testZipImporterMethods(self):
197 packdir = TESTPACK + os.sep
198 packdir2 = packdir + TESTPACK2 + os.sep
199 files = {packdir + "__init__" + pyc_ext: (NOW, test_pyc),
200 packdir2 + "__init__" + pyc_ext: (NOW, test_pyc),
201 packdir2 + TESTMOD + pyc_ext: (NOW, test_pyc)}
202
203 z = ZipFile(TEMP_ZIP, "w")
204 try:
205 for name, (mtime, data) in files.items():
206 zinfo = ZipInfo(name, time.localtime(mtime))
207 zinfo.compress_type = self.compression
208 z.writestr(zinfo, data)
209 z.close()
210
211 zi = zipimport.zipimporter(TEMP_ZIP)
212 self.assertEquals(zi.is_package(TESTPACK), True)
213 zi.load_module(TESTPACK)
Tim Petersbc29c1a2006-01-23 21:28:42 +0000214
Neal Norwitzb155b622006-01-23 07:52:13 +0000215 self.assertEquals(zi.is_package(packdir + '__init__'), False)
216 self.assertEquals(zi.is_package(packdir + TESTPACK2), True)
217 self.assertEquals(zi.is_package(packdir2 + TESTMOD), False)
218
219 mod_name = packdir2 + TESTMOD
Tim Peters68f2d002006-01-23 22:19:24 +0000220 mod = __import__(module_path_to_dotted_name(mod_name))
Neal Norwitzb155b622006-01-23 07:52:13 +0000221 self.assertEquals(zi.get_source(TESTPACK), None)
222 self.assertEquals(zi.get_source(mod_name), None)
Christian Heimes7f044312008-01-06 17:05:40 +0000223
224 # test prefix and archivepath members
225 zi2 = zipimport.zipimporter(TEMP_ZIP + os.sep + TESTPACK)
226 self.assertEquals(zi2.archive, TEMP_ZIP)
227 self.assertEquals(zi2.prefix, TESTPACK + os.sep)
Neal Norwitzb155b622006-01-23 07:52:13 +0000228 finally:
229 z.close()
230 os.remove(TEMP_ZIP)
231
Just van Rossum52e14d62002-12-30 22:08:05 +0000232 def testGetData(self):
233 z = ZipFile(TEMP_ZIP, "w")
234 z.compression = self.compression
235 try:
236 name = "testdata.dat"
Guido van Rossumad8d3002007-08-03 18:40:49 +0000237 data = bytes(x for x in range(256))
Just van Rossum52e14d62002-12-30 22:08:05 +0000238 z.writestr(name, data)
239 z.close()
240 zi = zipimport.zipimporter(TEMP_ZIP)
241 self.assertEquals(data, zi.get_data(name))
Neal Norwitzb155b622006-01-23 07:52:13 +0000242 self.assert_('zipimporter object' in repr(zi))
Just van Rossum52e14d62002-12-30 22:08:05 +0000243 finally:
244 z.close()
245 os.remove(TEMP_ZIP)
246
247 def testImporterAttr(self):
248 src = """if 1: # indent hack
249 def get_file():
250 return __file__
Guido van Rossumad8d3002007-08-03 18:40:49 +0000251 if __loader__.get_data("some.data") != b"some data":
Collin Winter828f04a2007-08-31 00:04:24 +0000252 raise AssertionError("bad data")\n"""
Just van Rossum52e14d62002-12-30 22:08:05 +0000253 pyc = make_pyc(compile(src, "<???>", "exec"), NOW)
254 files = {TESTMOD + pyc_ext: (NOW, pyc),
255 "some.data": (NOW, "some data")}
256 self.doTest(pyc_ext, files, TESTMOD)
257
Thomas Heller354e3d92003-07-22 18:10:15 +0000258 def testImport_WithStuff(self):
259 # try importing from a zipfile which contains additional
260 # stuff at the beginning of the file
261 files = {TESTMOD + ".py": (NOW, test_src)}
262 self.doTest(".py", files, TESTMOD,
Guido van Rossum85825dc2007-08-27 17:03:28 +0000263 stuff=b"Some Stuff"*31)
Just van Rossum52e14d62002-12-30 22:08:05 +0000264
Thomas Wouters49fd7fa2006-04-21 10:40:58 +0000265 def assertModuleSource(self, module):
266 self.assertEqual(inspect.getsource(module), test_src)
267
268 def testGetSource(self):
269 files = {TESTMOD + ".py": (NOW, test_src)}
270 self.doTest(".py", files, TESTMOD, call=self.assertModuleSource)
271
272 def testGetCompiledSource(self):
273 pyc = make_pyc(compile(test_src, "<???>", "exec"), NOW)
274 files = {TESTMOD + ".py": (NOW, test_src),
275 TESTMOD + pyc_ext: (NOW, pyc)}
276 self.doTest(pyc_ext, files, TESTMOD, call=self.assertModuleSource)
277
278 def runDoctest(self, callback):
279 files = {TESTMOD + ".py": (NOW, test_src),
280 "xyz.txt": (NOW, ">>> log.append(True)\n")}
281 self.doTest(".py", files, TESTMOD, call=callback)
282
283 def doDoctestFile(self, module):
284 log = []
285 old_master, doctest.master = doctest.master, None
286 try:
287 doctest.testfile(
288 'xyz.txt', package=module, module_relative=True,
289 globs=locals()
290 )
291 finally:
292 doctest.master = old_master
293 self.assertEqual(log,[True])
294
295 def testDoctestFile(self):
296 self.runDoctest(self.doDoctestFile)
297
298 def doDoctestSuite(self, module):
299 log = []
300 doctest.DocFileTest(
301 'xyz.txt', package=module, module_relative=True,
302 globs=locals()
303 ).run()
304 self.assertEqual(log,[True])
305
306 def testDoctestSuite(self):
307 self.runDoctest(self.doDoctestSuite)
308
309
310 def doTraceback(self, module):
311 try:
312 module.do_raise()
313 except:
314 tb = sys.exc_info()[2].tb_next
315
316 f,lno,n,line = extract_tb(tb, 1)[0]
317 self.assertEqual(line, raise_src.strip())
318
319 f,lno,n,line = extract_stack(tb.tb_frame, 1)[0]
320 self.assertEqual(line, raise_src.strip())
321
Guido van Rossum34d19282007-08-09 01:03:29 +0000322 s = io.StringIO()
Thomas Wouters49fd7fa2006-04-21 10:40:58 +0000323 print_tb(tb, 1, s)
324 self.failUnless(s.getvalue().endswith(raise_src))
325 else:
326 raise AssertionError("This ought to be impossible")
327
328 def testTraceback(self):
329 files = {TESTMOD + ".py": (NOW, raise_src)}
330 self.doTest(None, files, TESTMOD, call=self.doTraceback)
331
332
Just van Rossum52e14d62002-12-30 22:08:05 +0000333class CompressedZipImportTestCase(UncompressedZipImportTestCase):
334 compression = ZIP_DEFLATED
335
336
Neal Norwitzb155b622006-01-23 07:52:13 +0000337class BadFileZipImportTestCase(unittest.TestCase):
338 def assertZipFailure(self, filename):
339 self.assertRaises(zipimport.ZipImportError,
340 zipimport.zipimporter, filename)
341
342 def testNoFile(self):
343 self.assertZipFailure('AdfjdkFJKDFJjdklfjs')
344
345 def testEmptyFilename(self):
346 self.assertZipFailure('')
347
348 def testBadArgs(self):
349 self.assertRaises(TypeError, zipimport.zipimporter, None)
350 self.assertRaises(TypeError, zipimport.zipimporter, TESTMOD, kwd=None)
351
352 def testFilenameTooLong(self):
353 self.assertZipFailure('A' * 33000)
354
355 def testEmptyFile(self):
356 test_support.unlink(TESTMOD)
357 open(TESTMOD, 'w+').close()
358 self.assertZipFailure(TESTMOD)
359
360 def testFileUnreadable(self):
361 test_support.unlink(TESTMOD)
362 fd = os.open(TESTMOD, os.O_CREAT, 000)
Tim Peters68f2d002006-01-23 22:19:24 +0000363 try:
364 os.close(fd)
365 self.assertZipFailure(TESTMOD)
366 finally:
367 # If we leave "the read-only bit" set on Windows, nothing can
368 # delete TESTMOD, and later tests suffer bogus failures.
Guido van Rossumcd16bf62007-06-13 18:07:49 +0000369 os.chmod(TESTMOD, 0o666)
Tim Peters68f2d002006-01-23 22:19:24 +0000370 test_support.unlink(TESTMOD)
Neal Norwitzb155b622006-01-23 07:52:13 +0000371
372 def testNotZipFile(self):
373 test_support.unlink(TESTMOD)
374 fp = open(TESTMOD, 'w+')
375 fp.write('a' * 22)
376 fp.close()
377 self.assertZipFailure(TESTMOD)
378
Neal Norwitzdbc95f42006-01-23 08:48:03 +0000379 # XXX: disabled until this works on Big-endian machines
380 def _testBogusZipFile(self):
Neal Norwitzb155b622006-01-23 07:52:13 +0000381 test_support.unlink(TESTMOD)
382 fp = open(TESTMOD, 'w+')
383 fp.write(struct.pack('=I', 0x06054B50))
384 fp.write('a' * 18)
385 fp.close()
386 z = zipimport.zipimporter(TESTMOD)
387
388 try:
389 self.assertRaises(TypeError, z.find_module, None)
390 self.assertRaises(TypeError, z.load_module, None)
391 self.assertRaises(TypeError, z.is_package, None)
392 self.assertRaises(TypeError, z.get_code, None)
393 self.assertRaises(TypeError, z.get_data, None)
394 self.assertRaises(TypeError, z.get_source, None)
395
396 error = zipimport.ZipImportError
397 self.assertEqual(z.find_module('abc'), None)
398
399 self.assertRaises(error, z.load_module, 'abc')
400 self.assertRaises(error, z.get_code, 'abc')
401 self.assertRaises(IOError, z.get_data, 'abc')
402 self.assertRaises(error, z.get_source, 'abc')
403 self.assertRaises(error, z.is_package, 'abc')
404 finally:
405 zipimport._zip_directory_cache.clear()
406
407
408def cleanup():
409 # this is necessary if test is run repeated (like when finding leaks)
410 global test_imported
411 if test_imported:
412 zipimport._zip_directory_cache.clear()
413 if hasattr(UncompressedZipImportTestCase, 'testAFakeZlib'):
414 delattr(UncompressedZipImportTestCase, 'testAFakeZlib')
415 if hasattr(CompressedZipImportTestCase, 'testAFakeZlib'):
416 delattr(CompressedZipImportTestCase, 'testAFakeZlib')
417 test_imported = True
418
Neal Norwitz5c1ba532003-02-17 18:05:20 +0000419def test_main():
Neal Norwitzb155b622006-01-23 07:52:13 +0000420 cleanup()
421 try:
422 test_support.run_unittest(
423 UncompressedZipImportTestCase,
424 CompressedZipImportTestCase,
425 BadFileZipImportTestCase,
426 )
427 finally:
428 test_support.unlink(TESTMOD)
Neal Norwitz5c1ba532003-02-17 18:05:20 +0000429
430if __name__ == "__main__":
431 test_main()