]> granicus.if.org Git - python/commitdiff
Issue #28449: tarfile.open() with mode "r" or "r:" now tries to open a tar
authorSerhiy Storchaka <storchaka@gmail.com>
Sun, 30 Oct 2016 18:52:29 +0000 (20:52 +0200)
committerSerhiy Storchaka <storchaka@gmail.com>
Sun, 30 Oct 2016 18:52:29 +0000 (20:52 +0200)
file with compression before trying to open it without compression.  Otherwise
it had 50% chance failed with ignore_zeros=True.

Lib/tarfile.py
Lib/test/test_tarfile.py
Misc/NEWS

index 721f9d7f9180316cbb1c90ea23135b3785adfec9..999a99b978fced11befb850c3025c12a27064c99 100755 (executable)
@@ -1549,7 +1549,9 @@ class TarFile(object):
 
         if mode in ("r", "r:*"):
             # Find out which *open() is appropriate for opening the file.
-            for comptype in cls.OPEN_METH:
+            def not_compressed(comptype):
+                return cls.OPEN_METH[comptype] == 'taropen'
+            for comptype in sorted(cls.OPEN_METH, key=not_compressed):
                 func = getattr(cls, cls.OPEN_METH[comptype])
                 if fileobj is not None:
                     saved_pos = fileobj.tell()
index abfb34dfb812d810eba3f116764801c25ea11c99..1efb841bb22dc2f0922a606a00c0d396fa5df3cc 100644 (file)
@@ -3,6 +3,7 @@ import os
 import io
 from hashlib import md5
 from contextlib import contextmanager
+from random import Random
 
 import unittest
 import unittest.mock
@@ -349,12 +350,17 @@ class CommonReadTest(ReadTest):
 
     def test_ignore_zeros(self):
         # Test TarFile's ignore_zeros option.
+        # generate 512 pseudorandom bytes
+        data = Random(0).getrandbits(512*8).to_bytes(512, 'big')
         for char in (b'\0', b'a'):
             # Test if EOFHeaderError ('\0') and InvalidHeaderError ('a')
             # are ignored correctly.
             with self.open(tmpname, "w") as fobj:
                 fobj.write(char * 1024)
-                fobj.write(tarfile.TarInfo("foo").tobuf())
+                tarinfo = tarfile.TarInfo("foo")
+                tarinfo.size = len(data)
+                fobj.write(tarinfo.tobuf())
+                fobj.write(data)
 
             tar = tarfile.open(tmpname, mode="r", ignore_zeros=True)
             try:
index f7226fddf07add14a20d70064954a52ce2f35a52..298d3aa64e2889a616cae8bf729c53778cfd6b3d 100644 (file)
--- a/Misc/NEWS
+++ b/Misc/NEWS
@@ -113,6 +113,10 @@ Core and Builtins
 Library
 -------
 
+- Issue #28449: tarfile.open() with mode "r" or "r:" now tries to open a tar
+  file with compression before trying to open it without compression.  Otherwise
+  it had 50% chance failed with ignore_zeros=True.
+
 - Issue #23262: The webbrowser module now supports Firefox 36+ and derived
   browsers.  Based on patch by Oleg Broytman.