]> git.ipfire.org Git - thirdparty/Python/cpython.git/commitdiff
bpo-44439: _ZipWriteFile.write() handle buffer protocol correctly (GH-29468)
authorMiss Islington (bot) <31488909+miss-islington@users.noreply.github.com>
Tue, 8 Mar 2022 10:04:54 +0000 (02:04 -0800)
committerGitHub <noreply@github.com>
Tue, 8 Mar 2022 10:04:54 +0000 (02:04 -0800)
Co-authored-by: Marco Ribeiro <marcoffee@users.noreply.github.com>
(cherry picked from commit 36dd7396fcd26d8bf9919d536d05d7000becbe5b)

Co-authored-by: Ma Lin <animalize@users.noreply.github.com>
Lib/test/test_zipfile.py
Lib/zipfile.py
Misc/NEWS.d/next/Library/2021-11-08-20-27-41.bpo-44439.I_8qro.rst [new file with mode: 0644]

index 999197a0e32dcc10770247a80f0dd0455f96912b..bd383d3f68552baa5e2b8cac5a89483192bd538c 100644 (file)
@@ -1,3 +1,4 @@
+import array
 import contextlib
 import importlib.util
 import io
@@ -1117,6 +1118,14 @@ class AbstractWriterTests:
             self.assertRaises(ValueError, w.write, b'')
             self.assertEqual(zipf.read('test'), data)
 
+    def test_issue44439(self):
+        q = array.array('Q', [1, 2, 3, 4, 5])
+        LENGTH = len(q) * q.itemsize
+        with zipfile.ZipFile(io.BytesIO(), 'w', self.compression) as zip:
+            with zip.open('data', 'w') as data:
+                self.assertEqual(data.write(q), LENGTH)
+            self.assertEqual(zip.getinfo('data').file_size, LENGTH)
+
 class StoredWriterTests(AbstractWriterTests, unittest.TestCase):
     compression = zipfile.ZIP_STORED
 
index 816f8582bbf6d5949537f1e42e1173b67bc4c19a..f72c6a4e0c25397651cfa6ffd566d94b400550f8 100644 (file)
@@ -1120,8 +1120,15 @@ class _ZipWriteFile(io.BufferedIOBase):
     def write(self, data):
         if self.closed:
             raise ValueError('I/O operation on closed file.')
-        nbytes = len(data)
+
+        # Accept any data that supports the buffer protocol
+        if isinstance(data, (bytes, bytearray)):
+            nbytes = len(data)
+        else:
+            data = memoryview(data)
+            nbytes = data.nbytes
         self._file_size += nbytes
+
         self._crc = crc32(data, self._crc)
         if self._compressor:
             data = self._compressor.compress(data)
diff --git a/Misc/NEWS.d/next/Library/2021-11-08-20-27-41.bpo-44439.I_8qro.rst b/Misc/NEWS.d/next/Library/2021-11-08-20-27-41.bpo-44439.I_8qro.rst
new file mode 100644 (file)
index 0000000..f4e562c
--- /dev/null
@@ -0,0 +1,2 @@
+Fix ``.write()`` method of a member file in ``ZipFile``, when the input data is
+an object that supports the buffer protocol, the file length may be wrong.