summaryrefslogtreecommitdiffstats
path: root/Lib
diff options
context:
space:
mode:
authorZackery Spytz <zspytz@gmail.com>2018-09-25 19:15:47 (GMT)
committerBarry Warsaw <barry@python.org>2018-09-25 19:15:47 (GMT)
commit5a5ce064b3baadcb79605c5a42ee3d0aee57cdfc (patch)
tree4e466db8516df40b4bd1d902175d8de26e724f3e /Lib
parent996859a90df51f84eab47351702cb59c6db4428a (diff)
downloadcpython-5a5ce064b3baadcb79605c5a42ee3d0aee57cdfc.zip
cpython-5a5ce064b3baadcb79605c5a42ee3d0aee57cdfc.tar.gz
cpython-5a5ce064b3baadcb79605c5a42ee3d0aee57cdfc.tar.bz2
bpo-5950: Support reading zips with comments in zipimport (#9548)
* bpo-5950: Support reading zips with comments in zipimport
Diffstat (limited to 'Lib')
-rw-r--r--Lib/test/test_zipimport.py15
-rw-r--r--Lib/zipimport.py36
2 files changed, 46 insertions, 5 deletions
diff --git a/Lib/test/test_zipimport.py b/Lib/test/test_zipimport.py
index 63e5672..e98b090 100644
--- a/Lib/test/test_zipimport.py
+++ b/Lib/test/test_zipimport.py
@@ -116,6 +116,9 @@ class UncompressedZipImportTestCase(ImportHooksBaseTestCase):
zinfo = ZipInfo(name, time.localtime(mtime))
zinfo.compress_type = self.compression
z.writestr(zinfo, data)
+ comment = kw.get("comment", None)
+ if comment is not None:
+ z.comment = comment
stuff = kw.get("stuff", None)
if stuff is not None:
@@ -665,6 +668,18 @@ class UncompressedZipImportTestCase(ImportHooksBaseTestCase):
with self.assertRaises(TypeError):
zipimport.zipimporter(memoryview(os.fsencode(filename)))
+ def testComment(self):
+ files = {TESTMOD + ".py": (NOW, test_src)}
+ self.doTest(".py", files, TESTMOD, comment=b"comment")
+
+ def testBeginningCruftAndComment(self):
+ files = {TESTMOD + ".py": (NOW, test_src)}
+ self.doTest(".py", files, TESTMOD, stuff=b"cruft" * 64, comment=b"hi")
+
+ def testLargestPossibleComment(self):
+ files = {TESTMOD + ".py": (NOW, test_src)}
+ self.doTest(".py", files, TESTMOD, comment=b"c" * ((1 << 16) - 1))
+
@support.requires_zlib
class CompressedZipImportTestCase(UncompressedZipImportTestCase):
diff --git a/Lib/zipimport.py b/Lib/zipimport.py
index 4017340..2c11f68f 100644
--- a/Lib/zipimport.py
+++ b/Lib/zipimport.py
@@ -38,6 +38,9 @@ _zip_directory_cache = {}
_module_type = type(sys)
+END_CENTRAL_DIR_SIZE = 22
+STRING_END_ARCHIVE = b'PK\x05\x06'
+MAX_COMMENT_LEN = (1 << 16) - 1
class zipimporter:
"""zipimporter(archivepath) -> zipimporter object
@@ -354,16 +357,39 @@ def _read_directory(archive):
with fp:
try:
- fp.seek(-22, 2)
+ fp.seek(-END_CENTRAL_DIR_SIZE, 2)
header_position = fp.tell()
- buffer = fp.read(22)
+ buffer = fp.read(END_CENTRAL_DIR_SIZE)
except OSError:
raise ZipImportError(f"can't read Zip file: {archive!r}", path=archive)
- if len(buffer) != 22:
+ if len(buffer) != END_CENTRAL_DIR_SIZE:
raise ZipImportError(f"can't read Zip file: {archive!r}", path=archive)
- if buffer[:4] != b'PK\x05\x06':
+ if buffer[:4] != STRING_END_ARCHIVE:
# Bad: End of Central Dir signature
- raise ZipImportError(f'not a Zip file: {archive!r}', path=archive)
+ # Check if there's a comment.
+ try:
+ fp.seek(0, 2)
+ file_size = fp.tell()
+ except OSError:
+ raise ZipImportError(f"can't read Zip file: {archive!r}",
+ path=archive)
+ max_comment_start = max(file_size - MAX_COMMENT_LEN -
+ END_CENTRAL_DIR_SIZE, 0)
+ try:
+ fp.seek(max_comment_start)
+ data = fp.read()
+ except OSError:
+ raise ZipImportError(f"can't read Zip file: {archive!r}",
+ path=archive)
+ pos = data.rfind(STRING_END_ARCHIVE)
+ if pos < 0:
+ raise ZipImportError(f'not a Zip file: {archive!r}',
+ path=archive)
+ buffer = data[pos:pos+END_CENTRAL_DIR_SIZE]
+ if len(buffer) != END_CENTRAL_DIR_SIZE:
+ raise ZipImportError(f"corrupt Zip file: {archive!r}",
+ path=archive)
+ header_position = file_size - len(data) + pos
header_size = _unpack_uint32(buffer[12:16])
header_offset = _unpack_uint32(buffer[16:20])