Source code for rebasehelper.archive
# -*- coding: utf-8 -*-
#
# This tool helps you rebase your package to the latest version
# Copyright (C) 2013-2019 Red Hat, Inc.
#
# This program is free software; you can redistribute it and/or modify
# it under the terms of the GNU General Public License as published by
# the Free Software Foundation; either version 2 of the License, or
# (at your option) any later version.
#
# This program is distributed in the hope that it will be useful,
# but WITHOUT ANY WARRANTY; without even the implied warranty of
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
# GNU General Public License for more details.
#
# You should have received a copy of the GNU General Public License along
# with this program; if not, write to the Free Software Foundation, Inc.,
# 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
#
# Authors: Petr Hráček <phracek@redhat.com>
# Tomáš Hozza <thozza@redhat.com>
# Nikola Forró <nforro@redhat.com>
# František Nečas <fifinecas@seznam.cz>
import bz2
import logging
import lzma
import os
import shutil
import tarfile
import zipfile
from typing import Dict, Type, cast
from rebasehelper.logger import CustomLogger
from rebasehelper.helpers.path_helper import PathHelper
logger: CustomLogger = cast(CustomLogger, logging.getLogger(__name__))
# supported archive types
archive_types: Dict[str, Type['ArchiveTypeBase']] = {}
[docs]
def register_archive_type(archive):
archive_types[archive.EXTENSION] = archive
return archive
[docs]
class ArchiveTypeBase:
"""Base class for various archive types"""
EXTENSION: str = ''
[docs]
@classmethod
def match(cls, filename):
"""
Checks if the filename matches the archive type. If yes, returns
True, otherwise returns False.
"""
return filename.endswith(cls.EXTENSION)
[docs]
@classmethod
def open(cls, filename):
"""
Opens archive with the given filename and returns the proper
archive type object.
"""
raise NotImplementedError()
[docs]
@register_archive_type
class TarXzArchiveType(ArchiveTypeBase):
EXTENSION: str = '.tar.xz'
[docs]
@classmethod
def open(cls, filename):
return tarfile.open(mode='r', fileobj=lzma.LZMAFile(filename, 'r'))
[docs]
@register_archive_type
class TarBz2ArchiveType(ArchiveTypeBase):
EXTENSION: str = '.tar.bz2'
[docs]
@classmethod
def open(cls, filename):
if filename.endswith(TarBz2ArchiveType.EXTENSION):
return tarfile.TarFile.open(filename)
else:
return bz2.BZ2File(filename)
[docs]
@register_archive_type
class Bz2ArchiveType(TarBz2ArchiveType):
EXTENSION: str = '.bz2'
[docs]
@register_archive_type
class TarGzArchiveType(TarBz2ArchiveType):
EXTENSION: str = '.tar.gz'
[docs]
@classmethod
def open(cls, filename):
return tarfile.TarFile.open(filename)
[docs]
@register_archive_type
class TgzArchiveType(TarGzArchiveType):
EXTENSION: str = '.tgz'
[docs]
@register_archive_type
class TarArchiveType(TarGzArchiveType):
EXTENSION: str = '.tar'
[docs]
@register_archive_type
class CrateArchiveType(TarGzArchiveType):
EXTENSION: str = '.crate'
[docs]
@register_archive_type
class ZipArchiveType(ArchiveTypeBase):
EXTENSION: str = '.zip'
[docs]
@classmethod
def match(cls, filename):
return zipfile.is_zipfile(filename)
[docs]
@classmethod
def open(cls, filename):
return zipfile.ZipFile(filename, 'r')
[docs]
@register_archive_type
class GemArchiveType(ArchiveTypeBase):
EXTENSION: str = '.gem'
[docs]
class GemArchive:
def __init__(self, filename):
self.tmp = PathHelper.get_temp_dir()
TarArchiveType.extract(TarArchiveType.open(filename), '', self.tmp)
self.data = TarGzArchiveType.open(os.path.join(self.tmp, 'data.tar.gz'))
[docs]
def close(self):
self.data.close()
shutil.rmtree(self.tmp, onerror=lambda func, path, excinfo: shutil.rmtree(path))
[docs]
@classmethod
def open(cls, filename):
return cls.GemArchive(filename)
[docs]
class Archive:
"""Class representing an archive with sources"""
def __init__(self, filename):
self._filename = filename
self._archive_type = None
for archive_type in archive_types.values():
if archive_type.match(self._filename):
self._archive_type = archive_type
if self._archive_type is None:
raise NotImplementedError('Unsupported archive type')
[docs]
@classmethod
def get_supported_archives(cls):
return archive_types.keys()