Commit 2524fdef authored by Serhiy Storchaka's avatar Serhiy Storchaka Committed by GitHub

bpo-36434: Properly handle writing errors in ZIP files. (GH-12559)

Errors during writing no longer prevent to properly close
the ZIP file.
parent 7a465cb5
...@@ -402,6 +402,43 @@ class AbstractTestsWithSourceFile: ...@@ -402,6 +402,43 @@ class AbstractTestsWithSourceFile:
self.assertEqual(one_info._compresslevel, 1) self.assertEqual(one_info._compresslevel, 1)
self.assertEqual(nine_info._compresslevel, 9) self.assertEqual(nine_info._compresslevel, 9)
def test_writing_errors(self):
class BrokenFile(io.BytesIO):
def write(self, data):
nonlocal count
if count is not None:
if count == stop:
raise OSError
count += 1
super().write(data)
stop = 0
while True:
testfile = BrokenFile()
count = None
with zipfile.ZipFile(testfile, 'w', self.compression) as zipfp:
with zipfp.open('file1', 'w') as f:
f.write(b'data1')
count = 0
try:
with zipfp.open('file2', 'w') as f:
f.write(b'data2')
except OSError:
stop += 1
else:
break
finally:
count = None
with zipfile.ZipFile(io.BytesIO(testfile.getvalue())) as zipfp:
self.assertEqual(zipfp.namelist(), ['file1'])
self.assertEqual(zipfp.read('file1'), b'data1')
with zipfile.ZipFile(io.BytesIO(testfile.getvalue())) as zipfp:
self.assertEqual(zipfp.namelist(), ['file1', 'file2'])
self.assertEqual(zipfp.read('file1'), b'data1')
self.assertEqual(zipfp.read('file2'), b'data2')
def tearDown(self): def tearDown(self):
unlink(TESTFN) unlink(TESTFN)
unlink(TESTFN2) unlink(TESTFN2)
......
...@@ -1105,6 +1105,7 @@ class _ZipWriteFile(io.BufferedIOBase): ...@@ -1105,6 +1105,7 @@ class _ZipWriteFile(io.BufferedIOBase):
def close(self): def close(self):
if self.closed: if self.closed:
return return
try:
super().close() super().close()
# Flush any data from the compressor, and update header info # Flush any data from the compressor, and update header info
if self._compressor: if self._compressor:
...@@ -1127,11 +1128,11 @@ class _ZipWriteFile(io.BufferedIOBase): ...@@ -1127,11 +1128,11 @@ class _ZipWriteFile(io.BufferedIOBase):
else: else:
if not self._zip64: if not self._zip64:
if self._file_size > ZIP64_LIMIT: if self._file_size > ZIP64_LIMIT:
raise RuntimeError('File size unexpectedly exceeded ZIP64 ' raise RuntimeError(
'limit') 'File size unexpectedly exceeded ZIP64 limit')
if self._compress_size > ZIP64_LIMIT: if self._compress_size > ZIP64_LIMIT:
raise RuntimeError('Compressed size unexpectedly exceeded ' raise RuntimeError(
'ZIP64 limit') 'Compressed size unexpectedly exceeded ZIP64 limit')
# Seek backwards and write file header (which will now include # Seek backwards and write file header (which will now include
# correct CRC and file sizes) # correct CRC and file sizes)
...@@ -1141,11 +1142,13 @@ class _ZipWriteFile(io.BufferedIOBase): ...@@ -1141,11 +1142,13 @@ class _ZipWriteFile(io.BufferedIOBase):
self._fileobj.write(self._zinfo.FileHeader(self._zip64)) self._fileobj.write(self._zinfo.FileHeader(self._zip64))
self._fileobj.seek(self._zipfile.start_dir) self._fileobj.seek(self._zipfile.start_dir)
self._zipfile._writing = False
# Successfully written: Add file to our caches # Successfully written: Add file to our caches
self._zipfile.filelist.append(self._zinfo) self._zipfile.filelist.append(self._zinfo)
self._zipfile.NameToInfo[self._zinfo.filename] = self._zinfo self._zipfile.NameToInfo[self._zinfo.filename] = self._zinfo
finally:
self._zipfile._writing = False
class ZipFile: class ZipFile:
""" Class with methods to open, read, write, close, list zip files. """ Class with methods to open, read, write, close, list zip files.
......
Errors during writing to a ZIP file no longer prevent to properly close it.
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment