mirror of
https://github.com/gryf/e-uae-wrapper.git
synced 2026-02-07 09:35:45 +01:00
Ported file archiver from fs-uae-wrapper
This commit is contained in:
194
e_uae_wrapper/file_archive.py
Normal file
194
e_uae_wrapper/file_archive.py
Normal file
@@ -0,0 +1,194 @@
|
||||
"""
|
||||
File archive classes
|
||||
"""
|
||||
import os
|
||||
import subprocess
|
||||
import re
|
||||
import logging
|
||||
|
||||
from e_uae_wrapper import path
|
||||
|
||||
|
||||
class Archive(object):
|
||||
"""Base class for archive support"""
|
||||
ADD = ['a']
|
||||
EXTRACT = ['x']
|
||||
ARCH = 'false'
|
||||
|
||||
def __init__(self):
|
||||
self.archiver = path.which(self.ARCH)
|
||||
self._compress = self.archiver
|
||||
self._decompress = self.archiver
|
||||
|
||||
def create(self, arch_name, files=None):
|
||||
"""
|
||||
Create archive. Return True on success, False otherwise.
|
||||
"""
|
||||
files = files if files else ['.']
|
||||
logging.debug("Calling `%s %s %s %s'.", self._compress,
|
||||
" ".join(self.ADD), arch_name, " ".join(files))
|
||||
result = subprocess.call([self._compress] + self.ADD + [arch_name]
|
||||
+ files)
|
||||
if result != 0:
|
||||
logging.error("Unable to create archive `%s'.", arch_name)
|
||||
return False
|
||||
return True
|
||||
|
||||
def extract(self, arch_name):
|
||||
"""
|
||||
Extract archive. Return True on success, False otherwise.
|
||||
"""
|
||||
if not os.path.exists(arch_name):
|
||||
logging.error("Archive `%s' doesn't exists.", arch_name)
|
||||
return False
|
||||
|
||||
logging.debug("Calling `%s %s %s'.", self._compress,
|
||||
" ".join(self.ADD), arch_name)
|
||||
result = subprocess.call([self._decompress] + self.EXTRACT +
|
||||
[arch_name])
|
||||
if result != 0:
|
||||
logging.error("Unable to extract archive `%s'.", arch_name)
|
||||
return False
|
||||
return True
|
||||
|
||||
|
||||
class TarArchive(Archive):
|
||||
ADD = ['cf']
|
||||
EXTRACT = ['xf']
|
||||
ARCH = 'tar'
|
||||
|
||||
def create(self, arch_name, files=None):
|
||||
files = files if files else sorted(os.listdir('.'))
|
||||
logging.debug("Calling `%s %s %s %s'.", self._compress,
|
||||
" ".join(self.ADD), arch_name, " ".join(files))
|
||||
result = subprocess.call([self._compress] + self.ADD + [arch_name] +
|
||||
files)
|
||||
if result != 0:
|
||||
logging.error("Unable to create archive `%s'.", arch_name)
|
||||
return False
|
||||
return True
|
||||
|
||||
|
||||
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', 'zip']
|
||||
|
||||
def __init__(self):
|
||||
super(ZipArchive, self).__init__()
|
||||
if self.archiver == 'zip':
|
||||
self._decompress = path.which('unzip')
|
||||
ZipArchive.ADD = ['-r']
|
||||
ZipArchive.EXTRACT = []
|
||||
|
||||
|
||||
class SevenZArchive(Archive):
|
||||
ARCH = '7z'
|
||||
|
||||
|
||||
class LzxArchive(Archive):
|
||||
EXTRACT = ['-x']
|
||||
ARCH = 'unlzx'
|
||||
|
||||
@classmethod
|
||||
def create(self, arch_name, files=None):
|
||||
logging.error('Cannot create LZX archive. Only extracting is'
|
||||
'supported.')
|
||||
return False
|
||||
|
||||
|
||||
class RarArchive(Archive):
|
||||
ARCH = ['rar', 'unrar']
|
||||
|
||||
def create(self, arch_name, files=None):
|
||||
files = files if files else sorted(os.listdir('.'))
|
||||
if self.archiver == 'unrar':
|
||||
logging.error('Cannot create RAR archive. Only extracting is'
|
||||
'supported by unrar.')
|
||||
return False
|
||||
|
||||
logging.debug("Calling `%s %s %s %s'.", self._compress,
|
||||
" ".join(self.ADD), arch_name, " ".join(files))
|
||||
result = subprocess.call([self._compress] + self.ADD + [arch_name] +
|
||||
files)
|
||||
if result != 0:
|
||||
logging.error("Unable to create archive `%s'.", arch_name)
|
||||
return False
|
||||
return True
|
||||
|
||||
|
||||
class Archivers(object):
|
||||
"""Archivers class"""
|
||||
archivers = [{'arch': TarArchive, 'name': 'tar', 'ext': ['tar']},
|
||||
{'arch': TarGzipArchive, 'name': 'tgz',
|
||||
'ext': ['tar.gz', 'tgz']},
|
||||
{'arch': TarBzip2Archive, 'name': 'tar.bz2',
|
||||
'ext': ['tar.bz2']},
|
||||
{'arch': TarXzArchive, 'name': 'tar.xz', 'ext': ['tar.xz']},
|
||||
{'arch': RarArchive, 'name': 'rar', 'ext': ['rar']},
|
||||
{'arch': SevenZArchive, 'name': '7z', 'ext': ['7z']},
|
||||
{'arch': ZipArchive, 'name': 'zip', 'ext': ['zip']},
|
||||
{'arch': LhaArchive, 'name': 'lha', 'ext': ['lha', 'lzh']},
|
||||
{'arch': LzxArchive, 'name': 'lzx', 'ext': ['lzx']}]
|
||||
|
||||
@classmethod
|
||||
def get(cls, extension):
|
||||
"""
|
||||
Get the archive class or None
|
||||
"""
|
||||
for arch in cls.archivers:
|
||||
if extension in arch['ext']:
|
||||
return arch['arch']
|
||||
return None
|
||||
|
||||
@classmethod
|
||||
def get_extension_by_name(cls, name):
|
||||
"""
|
||||
Get the first defined extension for the archive format
|
||||
"""
|
||||
for arch in cls.archivers:
|
||||
if name == arch['name']:
|
||||
return '.' + arch['ext'][0]
|
||||
return None
|
||||
|
||||
|
||||
def get_archiver(arch_name):
|
||||
"""Return right class for provided archive file name"""
|
||||
|
||||
_, ext = os.path.splitext(arch_name)
|
||||
re_tar = re.compile('.*(.[tT][aA][rR].[^.]+$)')
|
||||
result = re_tar.match(arch_name)
|
||||
|
||||
if result:
|
||||
ext = result.groups()[0]
|
||||
|
||||
if ext:
|
||||
ext = ext[1:]
|
||||
|
||||
archiver = Archivers.get(ext)
|
||||
if not archiver:
|
||||
logging.error("Unable find archive type for `%s'.", arch_name)
|
||||
return None
|
||||
|
||||
archobj = archiver()
|
||||
if archobj.archiver is None:
|
||||
logging.error("Unable find executable for operating on files `*%s'.",
|
||||
ext)
|
||||
return None
|
||||
|
||||
return archobj
|
||||
22
e_uae_wrapper/path.py
Normal file
22
e_uae_wrapper/path.py
Normal file
@@ -0,0 +1,22 @@
|
||||
"""
|
||||
Misc utilities
|
||||
"""
|
||||
import os
|
||||
|
||||
|
||||
def which(executables):
|
||||
"""
|
||||
Check if there selected archiver is available in the system and place it
|
||||
to the archiver attribute
|
||||
"""
|
||||
|
||||
if not isinstance(executables, list):
|
||||
executables = [executables]
|
||||
|
||||
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):
|
||||
return fname
|
||||
|
||||
return None
|
||||
Reference in New Issue
Block a user