# Copyright (c) 2011, SmartFile # All rights reserved. # # Redistribution and use in source and binary forms, with or without # modification, are permitted provided that the following conditions are met: # * Redistributions of source code must retain the above copyright # notice, this list of conditions and the following disclaimer. # * Redistributions in binary form must reproduce the above copyright # notice, this list of conditions and the following disclaimer in the # documentation and/or other materials provided with the distribution. # * Neither the name of the organization nor the # names of its contributors may be used to endorse or promote products # derived from this software without specific prior written permission. # # THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND # ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED # WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE # DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER BE LIABLE FOR ANY # DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES # (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; # LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND # ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT # (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS # SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. import time from libarchive import is_archive, Entry, SeekableArchive from tarfile import DEFAULT_FORMAT, USTAR_FORMAT, GNU_FORMAT, PAX_FORMAT, ENCODING from tarfile import REGTYPE, AREGTYPE, LNKTYPE, SYMTYPE, DIRTYPE, FIFOTYPE, CONTTYPE, CHRTYPE, BLKTYPE, GNUTYPE_SPARSE FORMAT_CONVERSION = { USTAR_FORMAT: 'tar', GNU_FORMAT: 'gnu', PAX_FORMAT: 'pax', } def is_tarfile(filename): return is_archive(filename, formats=('tar', 'gnu', 'pax')) def open(**kwargs): return TarFile(**kwargs) class TarInfo(Entry): def __init__(self, name): super(TarInfo, self).__init__(pathname=name) fromtarfile = Entry.from_archive def get_name(self): return self.pathname def set_name(self, value): self.pathname = value name = property(get_name, set_name) @property def get_type(self): for attr, type in ( ('isdir', DIRTYPE), ('isfile', REGTYPE), ('issym', SYMTYPE), ('isfifo', FIFOTYPE), ('ischr', CHRTYPE), ('isblk', BLKTYPE), ): if getattr(self, attr)(): return type def _get_missing(self): raise NotImplemented() def _set_missing(self, value): raise NotImplemented() pax_headers = property(_get_missing, _set_missing) class TarFile(SeekableArchive): def __init__(self, name=None, mode='r', fileobj=None, format=DEFAULT_FORMAT, tarinfo=TarInfo, encoding=ENCODING): if name: f = name elif fileobj: f = fileobj try: format = FORMAT_CONVERSION.get(format) except KeyError: raise Exception('Invalid tar format: %s' % format) super(TarFile, self).__init__(f, mode=mode, format=format, entry_class=tarinfo, encoding=encoding) getmember = SeekableArchive.getentry list = SeekableArchive.printlist extract = SeekableArchive.readpath extractfile = SeekableArchive.readstream def getmembers(self): return list(self) def getnames(self): return list(self.iterpaths) def next(self): pass # TODO: how to do this? def extract(self, member, path=None): if path is None: path = os.getcwd() if isinstance(member, basestring): f = os.path.join(path, member) else: f = os.path.join(path, member.pathname) return self.readpath(member, f) def add(self, name, arcname, recursive=True, exclude=None, filter=None): pass # TODO: implement this. def addfile(tarinfo, fileobj): return self.writepath(fileobj, tarinfo) def gettarinfo(name=None, arcname=None, fileobj=None): if name: f = name elif fileobj: f = fileobj entry = self.entry_class.from_file(f) if arcname: entry.pathname = arcname return entry def _get_missing(self): raise NotImplemented() def _set_missing(self, value): raise NotImplemented() pax_headers = property(_get_missing, _set_missing)