136 lines
4.6 KiB

  1. # Copyright (c) 2011, SmartFile <btimby@smartfile.com>
  2. # All rights reserved.
  3. #
  4. # Redistribution and use in source and binary forms, with or without
  5. # modification, are permitted provided that the following conditions are met:
  6. # * Redistributions of source code must retain the above copyright
  7. # notice, this list of conditions and the following disclaimer.
  8. # * Redistributions in binary form must reproduce the above copyright
  9. # notice, this list of conditions and the following disclaimer in the
  10. # documentation and/or other materials provided with the distribution.
  11. # * Neither the name of the organization nor the
  12. # names of its contributors may be used to endorse or promote products
  13. # derived from this software without specific prior written permission.
  14. #
  15. # THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND
  16. # ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
  17. # WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
  18. # DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER BE LIABLE FOR ANY
  19. # DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
  20. # (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
  21. # LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
  22. # ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
  23. # (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
  24. # SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
  25. import time
  26. from libarchive import is_archive, Entry, SeekableArchive
  27. from tarfile import DEFAULT_FORMAT, USTAR_FORMAT, GNU_FORMAT, PAX_FORMAT, ENCODING
  28. from tarfile import REGTYPE, AREGTYPE, LNKTYPE, SYMTYPE, DIRTYPE, FIFOTYPE, CONTTYPE, CHRTYPE, BLKTYPE, GNUTYPE_SPARSE
  29. FORMAT_CONVERSION = {
  30. USTAR_FORMAT: 'tar',
  31. GNU_FORMAT: 'gnu',
  32. PAX_FORMAT: 'pax',
  33. }
  34. def is_tarfile(filename):
  35. return is_archive(filename, formats=('tar', 'gnu', 'pax'))
  36. def open(**kwargs):
  37. return TarFile(**kwargs)
  38. class TarInfo(Entry):
  39. def __init__(self, name):
  40. super(TarInfo, self).__init__(pathname=name)
  41. fromtarfile = Entry.from_archive
  42. def get_name(self):
  43. return self.pathname
  44. def set_name(self, value):
  45. self.pathname = value
  46. name = property(get_name, set_name)
  47. @property
  48. def get_type(self):
  49. for attr, type in (
  50. ('isdir', DIRTYPE), ('isfile', REGTYPE), ('issym', SYMTYPE),
  51. ('isfifo', FIFOTYPE), ('ischr', CHRTYPE), ('isblk', BLKTYPE),
  52. ):
  53. if getattr(self, attr)():
  54. return type
  55. def _get_missing(self):
  56. raise NotImplemented()
  57. def _set_missing(self, value):
  58. raise NotImplemented()
  59. pax_headers = property(_get_missing, _set_missing)
  60. class TarFile(SeekableArchive):
  61. def __init__(self, name=None, mode='r', fileobj=None, format=DEFAULT_FORMAT, tarinfo=TarInfo, encoding=ENCODING):
  62. if name:
  63. f = name
  64. elif fileobj:
  65. f = fileobj
  66. try:
  67. format = FORMAT_CONVERSION.get(format)
  68. except KeyError:
  69. raise Exception('Invalid tar format: %s' % format)
  70. super(TarFile, self).__init__(f, mode=mode, format=format, entry_class=tarinfo, encoding=encoding)
  71. getmember = SeekableArchive.getentry
  72. list = SeekableArchive.printlist
  73. extract = SeekableArchive.readpath
  74. extractfile = SeekableArchive.readstream
  75. def getmembers(self):
  76. return list(self)
  77. def getnames(self):
  78. return list(self.iterpaths)
  79. def next(self):
  80. pass # TODO: how to do this?
  81. def extract(self, member, path=None):
  82. if path is None:
  83. path = os.getcwd()
  84. if isinstance(member, basestring):
  85. f = os.path.join(path, member)
  86. else:
  87. f = os.path.join(path, member.pathname)
  88. return self.readpath(member, f)
  89. def add(self, name, arcname, recursive=True, exclude=None, filter=None):
  90. pass # TODO: implement this.
  91. def addfile(tarinfo, fileobj):
  92. return self.writepath(fileobj, tarinfo)
  93. def gettarinfo(name=None, arcname=None, fileobj=None):
  94. if name:
  95. f = name
  96. elif fileobj:
  97. f = fileobj
  98. entry = self.entry_class.from_file(f)
  99. if arcname:
  100. entry.pathname = arcname
  101. return entry
  102. def _get_missing(self):
  103. raise NotImplemented()
  104. def _set_missing(self, value):
  105. raise NotImplemented()
  106. pax_headers = property(_get_missing, _set_missing)