1
0
mirror of https://github.com/gryf/fs-uae-wrapper.git synced 2025-12-19 12:28:12 +01:00
Files
fs-uae-wrapper/fs_uae_wrapper/file_archive.py
gryf 251fab377b Non existed archive will not be fatal
Moved logic which check for archive file explicitly to extract method.
That way get_archiver will return archiver for not (yet) existed
archives.
2017-01-02 13:13:55 +01:00

152 lines
3.8 KiB
Python

"""
File archive classes
"""
import os
import subprocess
import sys
class Archive(object):
"""Base class for archive support"""
ADD = ['a']
EXTRACT = ['x']
ARCH = 'false'
def __init__(self):
self.archiver = None
self.which()
def create(self, arch_name):
"""
Create archive using self.archiver and parameters in self.ADD
attribute
"""
result = subprocess.call([self.archiver] + self.ADD + [arch_name, '.'])
if result != 0:
sys.stderr.write("Unable to create archive `%s'\n" % arch_name)
return False
return True
def extract(self, arch_name):
"""
Create archive using self.archiver and parameters in self.ADD
attribute
"""
if not os.path.exists(arch_name):
sys.stderr.write("Archive `%s' doesn't exists.\n" % arch_name)
return False
result = subprocess.call([self.archiver] + self.EXTRACT + [arch_name])
if result != 0:
sys.stderr.write("Unable to extract archive `%s'\n" % arch_name)
return False
return True
def which(self):
"""
Check if there selected archiver is available in the system and place
it to the archiver attribute
"""
if isinstance(self.ARCH, list):
executables = self.ARCH
else:
executables = [self.ARCH]
for fname in executables:
for path in os.environ["PATH"].split(os.pathsep):
path = os.path.join(path.strip('"'), fname)
if os.path.isfile(path) and os.access(path, os.X_OK):
self.archiver = fname
return
self.archiver = None
class TarArchive(Archive):
ADD = ['cf']
EXTRACT = ['xf']
ARCH = 'tar'
class TarGzipArchive(TarArchive):
ADD = ['zcf']
class TarBzip2Archive(TarArchive):
ADD = ['jcf']
class TarXzArchive(TarArchive):
ADD = ['Jcf']
class LhaArchive(Archive):
ARCH = 'lha'
class ZipArchive(Archive):
ADD = ['a', '-tzip']
ARCH = '7z'
class SevenZArchive(Archive):
ARCH = '7z'
class LzxArchive(Archive):
EXTRACT = ['-x']
ARCH = 'unlzx'
@classmethod
def create(self, arch_name):
sys.stderr.write('Cannot create LZX archive. Only extracting is'
'supported\n')
return False
class RarArchive(Archive):
ARCH = ['rar', 'unrar']
def create(self, arch_name):
if self.archiver == 'unrar':
sys.stderr.write('Cannot create RAR archive. Only extracting is'
'supported by unrar.\n')
return False
result = subprocess.call([self.archiver] + self.ADD + [arch_name] +
sorted(os.listdir('.')))
if result != 0:
sys.stderr.write("Unable to create archive `%s'\n" % arch_name)
return False
return True
def get_archiver(arch_name):
"""Return right class for provided archive file name"""
_, ext = os.path.splitext(arch_name)
archivers = {'.tar': TarArchive,
'.tgz': TarGzipArchive,
'.tar.gz': TarGzipArchive,
'.tar.bz2': TarBzip2Archive,
'.tar.xz': TarXzArchive,
'.rar': RarArchive,
'.7z': SevenZArchive,
'.zip': ZipArchive,
'.lha': LhaArchive,
'.lzx': LzxArchive}
archiver = archivers.get(ext)
if not archiver:
sys.stderr.write("Unable find archive type for `%s'\n" % arch_name)
return None
archobj = archiver()
if archobj.archiver is None:
sys.stderr.write("Unable find executable for operating on files"
" `*%s'\n" % ext)
return None
return archobj