You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
 
 
 
 
 

283 lines
9.1 KiB

  1. #!/usr/bin/env python
  2. # coding=utf-8
  3. #
  4. # Copyright (c) 2011, SmartFile <btimby@smartfile.com>
  5. # All rights reserved.
  6. #
  7. # Redistribution and use in source and binary forms, with or without
  8. # modification, are permitted provided that the following conditions are met:
  9. # * Redistributions of source code must retain the above copyright
  10. # notice, this list of conditions and the following disclaimer.
  11. # * Redistributions in binary form must reproduce the above copyright
  12. # notice, this list of conditions and the following disclaimer in the
  13. # documentation and/or other materials provided with the distribution.
  14. # * Neither the name of the organization nor the
  15. # names of its contributors may be used to endorse or promote products
  16. # derived from this software without specific prior written permission.
  17. #
  18. # THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND
  19. # ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
  20. # WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
  21. # DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER BE LIABLE FOR ANY
  22. # DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
  23. # (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
  24. # LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
  25. # ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
  26. # (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
  27. # SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
  28. import os, unittest, tempfile, random, string, sys
  29. import zipfile
  30. import io
  31. from libarchive import Archive, is_archive_name, is_archive
  32. from libarchive.zip import is_zipfile, ZipFile, ZipEntry
  33. PY3 = sys.version_info[0] == 3
  34. TMPDIR = tempfile.mkdtemp(suffix='.python-libarchive')
  35. ZIPFILE = 'test.zip'
  36. ZIPPATH = os.path.join(TMPDIR, ZIPFILE)
  37. FILENAMES = [
  38. 'test1.txt',
  39. 'foo',
  40. # TODO: test non-ASCII chars.
  41. #'álért.txt',
  42. ]
  43. def make_temp_files():
  44. if not os.path.exists(ZIPPATH):
  45. for name in FILENAMES:
  46. with open(os.path.join(TMPDIR, name), 'w') as f:
  47. f.write(''.join(random.sample(string.ascii_letters, 10)))
  48. def make_temp_archive():
  49. make_temp_files()
  50. with zipfile.ZipFile(ZIPPATH, mode="w") as z:
  51. for name in FILENAMES:
  52. z.write(os.path.join(TMPDIR, name), arcname=name)
  53. class TestIsArchiveName(unittest.TestCase):
  54. def test_formats(self):
  55. self.assertEqual(is_archive_name('foo'), None)
  56. self.assertEqual(is_archive_name('foo.txt'), None)
  57. self.assertEqual(is_archive_name('foo.txt.gz'), None)
  58. self.assertEqual(is_archive_name('foo.tar.gz'), 'tar')
  59. self.assertEqual(is_archive_name('foo.tar.bz2'), 'tar')
  60. self.assertEqual(is_archive_name('foo.zip'), 'zip')
  61. self.assertEqual(is_archive_name('foo.rar'), 'rar')
  62. self.assertEqual(is_archive_name('foo.iso'), 'iso')
  63. self.assertEqual(is_archive_name('foo.rpm'), 'cpio')
  64. class TestIsArchiveZip(unittest.TestCase):
  65. def setUp(self):
  66. make_temp_archive()
  67. def test_zip(self):
  68. self.assertEqual(is_archive(ZIPPATH), True)
  69. self.assertEqual(is_archive(ZIPPATH, formats=('zip', )), True)
  70. self.assertEqual(is_archive(ZIPPATH, formats=('tar', )), False)
  71. class TestIsArchiveTar(unittest.TestCase):
  72. def test_tar(self):
  73. pass
  74. # TODO: incorporate tests from:
  75. # http://hg.python.org/cpython/file/a6e1d926cd98/Lib/test/test_zipfile.py
  76. class TestZipRead(unittest.TestCase):
  77. def setUp(self):
  78. make_temp_archive()
  79. self.f = open(ZIPPATH, mode='r')
  80. def tearDown(self):
  81. self.f.close()
  82. def test_iszipfile(self):
  83. self.assertEqual(is_zipfile('/dev/null'), False)
  84. self.assertEqual(is_zipfile(ZIPPATH), True)
  85. def test_iterate(self):
  86. z = ZipFile(self.f, 'r')
  87. count = 0
  88. for e in z:
  89. count += 1
  90. self.assertEqual(count, len(FILENAMES), 'Did not enumerate correct number of items in archive.')
  91. def test_deferred_close_by_archive(self):
  92. """ Test archive deferred close without a stream. """
  93. z = ZipFile(self.f, 'r')
  94. self.assertIsNotNone(z._a)
  95. self.assertIsNone(z._stream)
  96. z.close()
  97. self.assertIsNone(z._a)
  98. def test_deferred_close_by_stream(self):
  99. """ Ensure archive closes self if stream is closed first. """
  100. z = ZipFile(self.f, 'r')
  101. stream = z.readstream(FILENAMES[0])
  102. stream.close()
  103. # Make sure archive stays open after stream is closed.
  104. self.assertIsNotNone(z._a)
  105. self.assertIsNone(z._stream)
  106. z.close()
  107. self.assertIsNone(z._a)
  108. self.assertTrue(stream.closed)
  109. def test_close_stream_first(self):
  110. """ Ensure that archive stays open after being closed if a stream is
  111. open. Further, ensure closing the stream closes the archive. """
  112. z = ZipFile(self.f, 'r')
  113. stream = z.readstream(FILENAMES[0])
  114. z.close()
  115. try:
  116. stream.read()
  117. except:
  118. self.fail("Reading stream from closed archive failed!")
  119. stream.close()
  120. # Now the archive should close.
  121. self.assertIsNone(z._a)
  122. self.assertTrue(stream.closed)
  123. self.assertIsNone(z._stream)
  124. def test_filenames(self):
  125. z = ZipFile(self.f, 'r')
  126. names = []
  127. for e in z:
  128. if PY3:
  129. names.append(e.filename)
  130. else:
  131. names.append(e.filename[0])
  132. self.assertEqual(names, FILENAMES, 'File names differ in archive.')
  133. #~ def test_non_ascii(self):
  134. #~ pass
  135. def test_extract_str(self):
  136. pass
  137. class TestZipWrite(unittest.TestCase):
  138. def setUp(self):
  139. make_temp_files()
  140. self.f = open(ZIPPATH, mode='w')
  141. def tearDown(self):
  142. self.f.close()
  143. def test_writepath(self):
  144. z = ZipFile(self.f, 'w')
  145. for fname in FILENAMES:
  146. with open(os.path.join(TMPDIR, fname), 'r') as f:
  147. z.writepath(f)
  148. z.close()
  149. def test_writepath_directory(self):
  150. """ Test writing a directory. """
  151. z = ZipFile(self.f, 'w')
  152. z.writepath(None, pathname='/testdir', folder=True)
  153. z.writepath(None, pathname='/testdir/testinside', folder=True)
  154. z.close()
  155. self.f.close()
  156. f = open(ZIPPATH, mode='r')
  157. z = ZipFile(f, 'r')
  158. entries = z.infolist()
  159. assert len(entries) == 2
  160. assert entries[0].isdir()
  161. z.close()
  162. f.close()
  163. def test_writestream(self):
  164. z = ZipFile(self.f, 'w')
  165. for fname in FILENAMES:
  166. full_path = os.path.join(TMPDIR, fname)
  167. i = open(full_path)
  168. o = z.writestream(fname)
  169. while True:
  170. data = i.read(1)
  171. if not data:
  172. break
  173. if PY3:
  174. o.write(data)
  175. else:
  176. o.write(unicode(data))
  177. o.close()
  178. i.close()
  179. z.close()
  180. def test_writestream_unbuffered(self):
  181. z = ZipFile(self.f, 'w')
  182. for fname in FILENAMES:
  183. full_path = os.path.join(TMPDIR, fname)
  184. i = open(full_path)
  185. o = z.writestream(fname, os.path.getsize(full_path))
  186. while True:
  187. data = i.read(1)
  188. if not data:
  189. break
  190. if PY3:
  191. o.write(data)
  192. else:
  193. o.write(unicode(data))
  194. o.close()
  195. i.close()
  196. z.close()
  197. def test_deferred_close_by_archive(self):
  198. """ Test archive deferred close without a stream. """
  199. z = ZipFile(self.f, 'w')
  200. o = z.writestream(FILENAMES[0])
  201. z.close()
  202. self.assertIsNotNone(z._a)
  203. self.assertIsNotNone(z._stream)
  204. if PY3:
  205. o.write('testdata')
  206. else:
  207. o.write(unicode('testdata'))
  208. o.close()
  209. self.assertIsNone(z._a)
  210. self.assertIsNone(z._stream)
  211. z.close()
  212. class TestHighLevelAPI(unittest.TestCase):
  213. def setUp(self):
  214. make_temp_archive()
  215. def _test_listing_content(self, f):
  216. """ Test helper capturing file paths while iterating the archive. """
  217. found = []
  218. with Archive(f) as a:
  219. for entry in a:
  220. found.append(entry.pathname)
  221. self.assertEqual(set(found), set(FILENAMES))
  222. def test_open_by_name(self):
  223. """ Test an archive opened directly by name. """
  224. self._test_listing_content(ZIPPATH)
  225. def test_open_by_named_fobj(self):
  226. """ Test an archive using a file-like object opened by name. """
  227. with open(ZIPPATH, 'rb') as f:
  228. self._test_listing_content(f)
  229. def test_open_by_unnamed_fobj(self):
  230. """ Test an archive using file-like object opened by fileno(). """
  231. with open(ZIPPATH, 'rb') as zf:
  232. with io.FileIO(zf.fileno(), mode='r', closefd=False) as f:
  233. self._test_listing_content(f)
  234. if __name__ == '__main__':
  235. unittest.main()