# Copyright (c) 2012-2014 Ciro Mattia Gonano # Copyright (c) 2013-2014 Pawel Jastrzebski # # Permission to use, copy, modify, and/or distribute this software for # any purpose with or without fee is hereby granted, provided that the # above copyright notice and this permission notice appear in all # copies. # # THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL # WARRANTIES WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED # WARRANTIES OF MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE # AUTHOR BE LIABLE FOR ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL # DAMAGES OR ANY DAMAGES WHATSOEVER RESULTING FROM LOSS OF USE, DATA # OR PROFITS, WHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE OR OTHER # TORTIOUS ACTION, ARISING OUT OF OR IN CONNECTION WITH THE USE OR # PERFORMANCE OF THIS SOFTWARE. # __license__ = 'ISC' __copyright__ = '2012-2014, Ciro Mattia Gonano , Pawel Jastrzebski ' __docformat__ = 'restructuredtext en' import sys import os from zipfile import is_zipfile, ZipFile from subprocess import STDOUT, PIPE from psutil import Popen from shutil import move, copy from . import rarfile class CBxArchive: def __init__(self, origFileName): self.origFileName = origFileName if is_zipfile(origFileName): self.compressor = 'zip' elif rarfile.is_rarfile(origFileName): self.compressor = 'rar' elif origFileName.endswith('.7z') or origFileName.endswith('.cb7'): self.compressor = '7z' else: self.compressor = None def isCbxFile(self): return self.compressor is not None def extractCBZ(self, targetdir): cbzFile = ZipFile(self.origFileName) filelist = [] for f in cbzFile.namelist(): if f.startswith('__MACOSX') or f.endswith('.DS_Store') or f.endswith('thumbs.db'): pass # skip MacOS special files elif f.endswith('/'): try: os.makedirs(os.path.join(targetdir, f)) except Exception: pass # the dir exists so we are going to extract the images only. else: filelist.append(f) cbzFile.extractall(targetdir, filelist) def extractCBR(self, targetdir): cbrFile = rarfile.RarFile(self.origFileName) filelist = [] for f in cbrFile.namelist(): if f.startswith('__MACOSX') or f.endswith('.DS_Store') or f.endswith('thumbs.db'): pass # skip MacOS special files elif f.endswith('/'): try: os.makedirs(os.path.join(targetdir, f)) except Exception: pass # the dir exists so we are going to extract the images only. else: filelist.append(f) cbrFile.extractall(targetdir, filelist) def extractCB7(self, targetdir): # Workaround for some wide UTF-8 + Popen abnormalities if sys.platform.startswith('darwin'): copy(self.origFileName, os.path.join(os.path.dirname(self.origFileName), 'TMP_KCC_TMP')) self.origFileName = os.path.join(os.path.dirname(self.origFileName), 'TMP_KCC_TMP') output = Popen('7za x "' + self.origFileName + '" -xr!__MACOSX -xr!.DS_Store -xr!thumbs.db -o"' + targetdir + '"', stdout=PIPE, stderr=STDOUT, shell=True) extracted = False for line in output.stdout: if b"Everything is Ok" in line: extracted = True if sys.platform.startswith('darwin'): os.remove(self.origFileName) if not extracted: raise OSError def extract(self, targetdir): if self.compressor == 'rar': self.extractCBR(targetdir) elif self.compressor == 'zip': self.extractCBZ(targetdir) elif self.compressor == '7z': self.extractCB7(targetdir) adir = os.listdir(targetdir) if 'ComicInfo.xml' in adir: adir.remove('ComicInfo.xml') if len(adir) == 1 and os.path.isdir(os.path.join(targetdir, adir[0])): for f in os.listdir(os.path.join(targetdir, adir[0])): # If directory names contain UTF-8 chars shutil.move can't clean up the mess alone if os.path.isdir(os.path.join(targetdir, f)): os.replace(os.path.join(targetdir, adir[0], f), os.path.join(targetdir, adir[0], f + '-A')) f += '-A' move(os.path.join(targetdir, adir[0], f), targetdir) os.rmdir(os.path.join(targetdir, adir[0])) return targetdir