2012-11-02 13:54:17 -07:00
|
|
|
"""
|
|
|
|
A python class to represent a single comic, be it file or folder of images
|
|
|
|
"""
|
|
|
|
|
2012-11-06 12:56:30 -08:00
|
|
|
"""
|
|
|
|
Copyright 2012 Anthony Beville
|
|
|
|
|
|
|
|
Licensed under the Apache License, Version 2.0 (the "License");
|
|
|
|
you may not use this file except in compliance with the License.
|
|
|
|
You may obtain a copy of the License at
|
|
|
|
|
|
|
|
http://www.apache.org/licenses/LICENSE-2.0
|
|
|
|
|
|
|
|
Unless required by applicable law or agreed to in writing, software
|
|
|
|
distributed under the License is distributed on an "AS IS" BASIS,
|
|
|
|
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
|
|
See the License for the specific language governing permissions and
|
|
|
|
limitations under the License.
|
|
|
|
"""
|
|
|
|
|
2012-11-02 13:54:17 -07:00
|
|
|
import zipfile
|
|
|
|
import os
|
|
|
|
import struct
|
2012-11-05 11:20:13 -08:00
|
|
|
import sys
|
|
|
|
import tempfile
|
2012-11-06 12:29:18 -08:00
|
|
|
import subprocess
|
|
|
|
import platform
|
2012-11-27 10:01:19 -08:00
|
|
|
if platform.system() == "Windows":
|
|
|
|
import _subprocess
|
2012-11-06 12:29:18 -08:00
|
|
|
import time
|
2012-11-05 11:20:13 -08:00
|
|
|
|
|
|
|
sys.path.insert(0, os.path.abspath(".") )
|
|
|
|
import UnRAR2
|
|
|
|
from UnRAR2.rar_exceptions import *
|
2012-11-02 13:54:17 -07:00
|
|
|
|
|
|
|
from options import Options, MetaDataStyle
|
|
|
|
from comicinfoxml import ComicInfoXml
|
|
|
|
from comicbookinfo import ComicBookInfo
|
2012-12-02 12:17:39 -08:00
|
|
|
from comet import CoMet
|
2012-11-02 13:54:17 -07:00
|
|
|
from genericmetadata import GenericMetadata
|
|
|
|
from filenameparser import FileNameParser
|
|
|
|
|
|
|
|
|
2012-11-05 13:34:23 -08:00
|
|
|
class ZipArchiver:
|
2012-11-05 11:46:48 -08:00
|
|
|
|
|
|
|
def __init__( self, path ):
|
|
|
|
self.path = path
|
|
|
|
|
|
|
|
def getArchiveComment( self ):
|
|
|
|
zf = zipfile.ZipFile( self.path, 'r' )
|
|
|
|
comment = zf.comment
|
|
|
|
zf.close()
|
|
|
|
return comment
|
|
|
|
|
|
|
|
def setArchiveComment( self, comment ):
|
2012-11-27 16:33:51 -08:00
|
|
|
return self.writeZipComment( self.path, comment )
|
2012-11-05 11:46:48 -08:00
|
|
|
|
|
|
|
def readArchiveFile( self, archive_file ):
|
|
|
|
zf = zipfile.ZipFile( self.path, 'r' )
|
|
|
|
data = zf.read( archive_file )
|
|
|
|
zf.close()
|
|
|
|
return data
|
|
|
|
|
|
|
|
def removeArchiveFile( self, archive_file ):
|
2012-11-27 16:33:51 -08:00
|
|
|
try:
|
|
|
|
self.rebuildZipFile( [ archive_file ] )
|
|
|
|
except:
|
|
|
|
return False
|
|
|
|
else:
|
|
|
|
return True
|
|
|
|
|
2012-11-05 11:46:48 -08:00
|
|
|
def writeArchiveFile( self, archive_file, data ):
|
|
|
|
# At the moment, no other option but to rebuild the whole
|
|
|
|
# zip archive w/o the indicated file. Very sucky, but maybe
|
|
|
|
# another solution can be found
|
2012-11-27 16:33:51 -08:00
|
|
|
try:
|
|
|
|
self.rebuildZipFile( [ archive_file ] )
|
|
|
|
|
|
|
|
#now just add the archive file as a new one
|
|
|
|
zf = zipfile.ZipFile(self.path, mode='a', compression=zipfile.ZIP_DEFLATED )
|
|
|
|
zf.writestr( archive_file, data )
|
|
|
|
zf.close()
|
|
|
|
return True
|
|
|
|
except:
|
|
|
|
return False
|
|
|
|
|
2012-11-05 11:46:48 -08:00
|
|
|
def getArchiveFilenameList( self ):
|
|
|
|
zf = zipfile.ZipFile( self.path, 'r' )
|
|
|
|
namelist = zf.namelist()
|
|
|
|
zf.close()
|
|
|
|
return namelist
|
|
|
|
|
|
|
|
# zip helper func
|
|
|
|
def rebuildZipFile( self, exclude_list ):
|
|
|
|
|
|
|
|
# this recompresses the zip archive, without the files in the exclude_list
|
2012-11-27 16:33:51 -08:00
|
|
|
#print "Rebuilding zip {0} without {1}".format( self.path, exclude_list )
|
2012-11-23 20:54:36 -08:00
|
|
|
|
|
|
|
# generate temp file
|
|
|
|
tmp_fd, tmp_name = tempfile.mkstemp( dir=os.path.dirname(self.path) )
|
|
|
|
os.close( tmp_fd )
|
|
|
|
|
2012-11-05 11:46:48 -08:00
|
|
|
zin = zipfile.ZipFile (self.path, 'r')
|
2012-11-23 20:54:36 -08:00
|
|
|
zout = zipfile.ZipFile (tmp_name, 'w')
|
2012-11-05 11:46:48 -08:00
|
|
|
for item in zin.infolist():
|
|
|
|
buffer = zin.read(item.filename)
|
|
|
|
if ( item.filename not in exclude_list ):
|
|
|
|
zout.writestr(item, buffer)
|
|
|
|
|
|
|
|
#preserve the old comment
|
|
|
|
zout.comment = zin.comment
|
|
|
|
|
|
|
|
zout.close()
|
|
|
|
zin.close()
|
|
|
|
|
|
|
|
# replace with the new file
|
|
|
|
os.remove( self.path )
|
2012-11-23 20:54:36 -08:00
|
|
|
os.rename( tmp_name, self.path )
|
2012-11-02 13:54:17 -07:00
|
|
|
|
|
|
|
|
2012-11-05 11:46:48 -08:00
|
|
|
def writeZipComment( self, filename, comment ):
|
|
|
|
"""
|
|
|
|
This is a custom function for writing a comment to a zip file,
|
|
|
|
since the built-in one doesn't seem to work on Windows and Mac OS/X
|
2012-11-02 13:54:17 -07:00
|
|
|
|
2012-11-05 11:46:48 -08:00
|
|
|
Fortunately, the zip comment is at the end of the file, and it's
|
|
|
|
easy to manipulate. See this website for more info:
|
|
|
|
see: http://en.wikipedia.org/wiki/Zip_(file_format)#Structure
|
|
|
|
"""
|
2012-11-02 13:54:17 -07:00
|
|
|
|
2012-11-05 11:46:48 -08:00
|
|
|
#get file size
|
|
|
|
statinfo = os.stat(filename)
|
|
|
|
file_length = statinfo.st_size
|
2012-11-02 13:54:17 -07:00
|
|
|
|
2012-11-27 16:33:51 -08:00
|
|
|
try:
|
|
|
|
fo = open(filename, "r+b")
|
2012-11-02 13:54:17 -07:00
|
|
|
|
2012-11-27 16:33:51 -08:00
|
|
|
#the starting position, relative to EOF
|
|
|
|
pos = -4
|
2012-11-05 11:46:48 -08:00
|
|
|
|
2012-11-27 16:33:51 -08:00
|
|
|
found = False
|
|
|
|
value = bytearray()
|
2012-11-02 13:54:17 -07:00
|
|
|
|
2012-11-27 16:33:51 -08:00
|
|
|
# walk backwards to find the "End of Central Directory" record
|
|
|
|
while ( not found ) and ( -pos != file_length ):
|
|
|
|
# seek, relative to EOF
|
|
|
|
fo.seek( pos, 2)
|
2012-11-02 13:54:17 -07:00
|
|
|
|
2012-11-27 16:33:51 -08:00
|
|
|
value = fo.read( 4 )
|
|
|
|
|
|
|
|
#look for the end of central directory signature
|
|
|
|
if bytearray(value) == bytearray([ 0x50, 0x4b, 0x05, 0x06 ]):
|
|
|
|
found = True
|
|
|
|
else:
|
|
|
|
# not found, step back another byte
|
|
|
|
pos = pos - 1
|
|
|
|
#print pos,"{1} int: {0:x}".format(bytearray(value)[0], value)
|
2012-11-05 11:46:48 -08:00
|
|
|
|
2012-11-27 16:33:51 -08:00
|
|
|
if found:
|
|
|
|
|
|
|
|
# now skip forward 20 bytes to the comment length word
|
|
|
|
pos += 20
|
|
|
|
fo.seek( pos, 2)
|
2012-11-05 11:46:48 -08:00
|
|
|
|
2012-11-27 16:33:51 -08:00
|
|
|
# Pack the length of the comment string
|
|
|
|
format = "H" # one 2-byte integer
|
|
|
|
comment_length = struct.pack(format, len(comment)) # pack integer in a binary string
|
|
|
|
|
|
|
|
# write out the length
|
|
|
|
fo.write( comment_length )
|
|
|
|
fo.seek( pos+2, 2)
|
|
|
|
|
|
|
|
# write out the comment itself
|
|
|
|
fo.write( comment )
|
|
|
|
fo.truncate()
|
|
|
|
fo.close()
|
|
|
|
else:
|
|
|
|
raise Exception('Failed to write comment to zip file!')
|
|
|
|
except:
|
|
|
|
return False
|
2012-11-05 11:46:48 -08:00
|
|
|
else:
|
2012-11-27 16:33:51 -08:00
|
|
|
return True
|
2012-12-05 21:45:53 -08:00
|
|
|
|
|
|
|
def copyFromArchive( self, otherArchive ):
|
|
|
|
# Replace the current zip with one copied from another archive
|
|
|
|
try:
|
|
|
|
zout = zipfile.ZipFile (self.path, 'w')
|
|
|
|
for fname in otherArchive.getArchiveFilenameList():
|
|
|
|
data = otherArchive.readArchiveFile( fname )
|
|
|
|
zout.writestr( fname, data )
|
|
|
|
zout.close()
|
|
|
|
|
|
|
|
#preserve the old comment
|
|
|
|
comment = otherArchive.getArchiveComment()
|
|
|
|
if comment is not None:
|
|
|
|
if not self.writeZipComment( self.path, comment ):
|
|
|
|
return False
|
|
|
|
except:
|
|
|
|
return False
|
|
|
|
else:
|
|
|
|
return True
|
|
|
|
|
2012-11-02 13:54:17 -07:00
|
|
|
|
2012-11-05 11:46:48 -08:00
|
|
|
#------------------------------------------
|
|
|
|
# RAR implementation
|
|
|
|
|
2012-11-05 13:34:23 -08:00
|
|
|
class RarArchiver:
|
2012-11-05 11:46:48 -08:00
|
|
|
|
|
|
|
def __init__( self, path ):
|
|
|
|
self.path = path
|
|
|
|
self.rar_exe_path = None
|
2012-11-06 12:29:18 -08:00
|
|
|
self.devnull = open(os.devnull, "w")
|
2012-11-05 11:46:48 -08:00
|
|
|
|
2012-11-06 12:29:18 -08:00
|
|
|
# windows only, keeps the cmd.exe from popping up
|
|
|
|
if platform.system() == "Windows":
|
|
|
|
self.startupinfo = subprocess.STARTUPINFO()
|
2012-11-23 14:05:56 -08:00
|
|
|
self.startupinfo.dwFlags |= _subprocess.STARTF_USESHOWWINDOW
|
2012-11-06 12:29:18 -08:00
|
|
|
else:
|
|
|
|
self.startupinfo = None
|
|
|
|
|
|
|
|
def __del__(self):
|
|
|
|
self.devnull.close()
|
|
|
|
|
2012-11-05 11:46:48 -08:00
|
|
|
def getArchiveComment( self ):
|
2012-11-02 13:54:17 -07:00
|
|
|
|
2012-11-05 11:46:48 -08:00
|
|
|
rarc = UnRAR2.RarFile( self.path )
|
|
|
|
return rarc.comment
|
|
|
|
|
|
|
|
def setArchiveComment( self, comment ):
|
|
|
|
|
|
|
|
if self.rar_exe_path is not None:
|
2012-11-27 16:33:51 -08:00
|
|
|
try:
|
|
|
|
# write comment to temp file
|
|
|
|
tmp_fd, tmp_name = tempfile.mkstemp()
|
|
|
|
f = os.fdopen(tmp_fd, 'w+b')
|
|
|
|
f.write( comment )
|
|
|
|
f.close()
|
2012-11-05 11:46:48 -08:00
|
|
|
|
2012-11-27 16:33:51 -08:00
|
|
|
# use external program to write comment to Rar archive
|
|
|
|
subprocess.call([self.rar_exe_path, 'c', '-c-', '-z' + tmp_name, self.path],
|
|
|
|
startupinfo=self.startupinfo,
|
|
|
|
stdout=self.devnull)
|
|
|
|
|
|
|
|
if platform.system() == "Darwin":
|
|
|
|
time.sleep(1)
|
|
|
|
|
|
|
|
os.remove( tmp_name)
|
|
|
|
except:
|
|
|
|
return False
|
|
|
|
else:
|
|
|
|
return True
|
|
|
|
else:
|
|
|
|
return False
|
|
|
|
|
2012-11-05 11:46:48 -08:00
|
|
|
def readArchiveFile( self, archive_file ):
|
|
|
|
|
|
|
|
entries = UnRAR2.RarFile( self.path ).read_files( archive_file )
|
|
|
|
|
|
|
|
#entries is a list of of tuples: ( rarinfo, filedata)
|
|
|
|
if (len(entries) == 1):
|
|
|
|
return entries[0][1]
|
|
|
|
else:
|
|
|
|
return ""
|
|
|
|
|
|
|
|
def writeArchiveFile( self, archive_file, data ):
|
|
|
|
|
|
|
|
if self.rar_exe_path is not None:
|
2012-11-27 16:33:51 -08:00
|
|
|
try:
|
|
|
|
tmp_folder = tempfile.mkdtemp()
|
2012-11-05 11:46:48 -08:00
|
|
|
|
2012-11-27 16:33:51 -08:00
|
|
|
tmp_file = os.path.join( tmp_folder, archive_file )
|
2012-11-05 11:46:48 -08:00
|
|
|
|
2012-11-27 16:33:51 -08:00
|
|
|
f = open(tmp_file, 'w')
|
|
|
|
f.write( data )
|
|
|
|
f.close()
|
2012-11-05 11:46:48 -08:00
|
|
|
|
2012-11-27 16:33:51 -08:00
|
|
|
# use external program to write file to Rar archive
|
|
|
|
subprocess.call([self.rar_exe_path, 'a', '-c-', '-ep', self.path, tmp_file],
|
|
|
|
startupinfo=self.startupinfo,
|
|
|
|
stdout=self.devnull)
|
|
|
|
|
|
|
|
if platform.system() == "Darwin":
|
|
|
|
time.sleep(1)
|
|
|
|
os.remove( tmp_file)
|
|
|
|
os.rmdir( tmp_folder)
|
|
|
|
except:
|
|
|
|
return False
|
|
|
|
else:
|
|
|
|
return True
|
|
|
|
else:
|
|
|
|
return False
|
|
|
|
|
2012-11-05 11:46:48 -08:00
|
|
|
def removeArchiveFile( self, archive_file ):
|
|
|
|
if self.rar_exe_path is not None:
|
2012-11-27 16:33:51 -08:00
|
|
|
try:
|
|
|
|
# use external program to remove file from Rar archive
|
|
|
|
subprocess.call([self.rar_exe_path, 'd','-c-', self.path, archive_file],
|
|
|
|
startupinfo=self.startupinfo,
|
|
|
|
stdout=self.devnull)
|
|
|
|
|
|
|
|
if platform.system() == "Darwin":
|
|
|
|
time.sleep(1)
|
|
|
|
except:
|
|
|
|
return False
|
|
|
|
else:
|
|
|
|
return True
|
|
|
|
else:
|
|
|
|
return False
|
|
|
|
|
2012-11-05 11:46:48 -08:00
|
|
|
def getArchiveFilenameList( self ):
|
|
|
|
|
|
|
|
rarc = UnRAR2.RarFile( self.path )
|
|
|
|
|
|
|
|
return [ item.filename for item in rarc.infolist() ]
|
2012-11-02 13:54:17 -07:00
|
|
|
|
2012-11-05 11:46:48 -08:00
|
|
|
#------------------------------------------
|
|
|
|
# Folder implementation
|
2012-11-05 13:34:23 -08:00
|
|
|
class FolderArchiver:
|
2012-11-02 13:54:17 -07:00
|
|
|
|
2012-11-05 11:46:48 -08:00
|
|
|
def __init__( self, path ):
|
|
|
|
self.path = path
|
2012-11-05 13:34:23 -08:00
|
|
|
self.comment_file_name = "ComicTaggerFolderComment.txt"
|
2012-11-05 11:46:48 -08:00
|
|
|
|
|
|
|
def getArchiveComment( self ):
|
2012-11-05 13:34:23 -08:00
|
|
|
return self.readArchiveFile( self.comment_file_name )
|
|
|
|
|
2012-11-05 11:46:48 -08:00
|
|
|
def setArchiveComment( self, comment ):
|
2012-11-27 16:33:51 -08:00
|
|
|
return self.writeArchiveFile( self.comment_file_name, comment )
|
2012-11-05 13:34:23 -08:00
|
|
|
|
|
|
|
def readArchiveFile( self, archive_file ):
|
|
|
|
|
2012-11-06 12:29:18 -08:00
|
|
|
data = ""
|
2012-11-05 13:34:23 -08:00
|
|
|
fname = os.path.join( self.path, archive_file )
|
|
|
|
try:
|
|
|
|
with open( fname, 'rb' ) as f:
|
|
|
|
data = f.read()
|
|
|
|
f.close()
|
|
|
|
except IOError as e:
|
|
|
|
pass
|
|
|
|
|
|
|
|
return data
|
|
|
|
|
2012-11-05 11:46:48 -08:00
|
|
|
def writeArchiveFile( self, archive_file, data ):
|
2012-11-05 13:34:23 -08:00
|
|
|
|
|
|
|
fname = os.path.join( self.path, archive_file )
|
|
|
|
try:
|
|
|
|
with open(fname, 'w+') as f:
|
|
|
|
f.write( data )
|
|
|
|
f.close()
|
2012-11-27 16:33:51 -08:00
|
|
|
except:
|
|
|
|
return False
|
|
|
|
else:
|
|
|
|
return True
|
2012-11-05 13:34:23 -08:00
|
|
|
|
2012-11-05 11:46:48 -08:00
|
|
|
def removeArchiveFile( self, archive_file ):
|
2012-11-05 13:34:23 -08:00
|
|
|
|
|
|
|
fname = os.path.join( self.path, archive_file )
|
|
|
|
try:
|
|
|
|
os.remove( fname )
|
|
|
|
except:
|
2012-11-27 16:33:51 -08:00
|
|
|
return False
|
|
|
|
else:
|
|
|
|
return True
|
2012-11-05 13:34:23 -08:00
|
|
|
|
2012-11-05 11:46:48 -08:00
|
|
|
def getArchiveFilenameList( self ):
|
2012-11-05 13:34:23 -08:00
|
|
|
return self.listFiles( self.path )
|
|
|
|
|
|
|
|
def listFiles( self, folder ):
|
|
|
|
|
|
|
|
itemlist = list()
|
|
|
|
|
|
|
|
for item in os.listdir( folder ):
|
|
|
|
itemlist.append( item )
|
|
|
|
if os.path.isdir( item ):
|
|
|
|
itemlist.extend( self.listFiles( os.path.join( folder, item ) ))
|
|
|
|
|
|
|
|
return itemlist
|
2012-11-05 11:46:48 -08:00
|
|
|
|
2012-11-02 13:54:17 -07:00
|
|
|
#------------------------------------------
|
2012-11-05 11:46:48 -08:00
|
|
|
# Unknown implementation
|
2012-11-05 13:34:23 -08:00
|
|
|
class UnknownArchiver:
|
2012-11-05 11:46:48 -08:00
|
|
|
|
|
|
|
def __init__( self, path ):
|
|
|
|
self.path = path
|
2012-11-02 13:54:17 -07:00
|
|
|
|
2012-11-05 11:46:48 -08:00
|
|
|
def getArchiveComment( self ):
|
|
|
|
return ""
|
|
|
|
def setArchiveComment( self, comment ):
|
2012-11-27 16:33:51 -08:00
|
|
|
return False
|
2012-11-05 11:46:48 -08:00
|
|
|
def readArchiveFilen( self ):
|
|
|
|
return ""
|
|
|
|
def writeArchiveFile( self, archive_file, data ):
|
2012-11-27 16:33:51 -08:00
|
|
|
return False
|
2012-11-05 11:46:48 -08:00
|
|
|
def removeArchiveFile( self, archive_file ):
|
2012-11-27 16:33:51 -08:00
|
|
|
return False
|
2012-11-05 11:46:48 -08:00
|
|
|
def getArchiveFilenameList( self ):
|
|
|
|
return []
|
2012-11-02 13:54:17 -07:00
|
|
|
|
2012-11-05 11:46:48 -08:00
|
|
|
#------------------------------------------------------------------
|
2012-11-02 13:54:17 -07:00
|
|
|
class ComicArchive:
|
|
|
|
|
2012-11-05 11:20:13 -08:00
|
|
|
class ArchiveType:
|
|
|
|
Zip, Rar, Folder, Unknown = range(4)
|
|
|
|
|
2012-11-02 13:54:17 -07:00
|
|
|
def __init__( self, path ):
|
|
|
|
self.path = path
|
|
|
|
self.ci_xml_filename = 'ComicInfo.xml'
|
2012-12-05 14:17:30 -08:00
|
|
|
self.comet_default_filename = 'CoMet.xml'
|
|
|
|
self.comet_filename = None
|
2012-11-05 11:20:13 -08:00
|
|
|
|
|
|
|
if self.zipTest():
|
|
|
|
self.archive_type = self.ArchiveType.Zip
|
2012-11-05 11:46:48 -08:00
|
|
|
self.archiver = ZipArchiver( self.path )
|
2012-11-05 11:20:13 -08:00
|
|
|
|
|
|
|
elif self.rarTest():
|
|
|
|
self.archive_type = self.ArchiveType.Rar
|
2012-11-05 11:46:48 -08:00
|
|
|
self.archiver = RarArchiver( self.path )
|
|
|
|
|
2012-11-05 11:20:13 -08:00
|
|
|
elif os.path.isdir( self.path ):
|
|
|
|
self.archive_type = self.ArchiveType.Folder
|
2012-11-05 11:46:48 -08:00
|
|
|
self.archiver = FolderArchiver( self.path )
|
2012-11-05 11:20:13 -08:00
|
|
|
else:
|
|
|
|
self.archive_type = self.ArchiveType.Unknown
|
2012-11-05 11:46:48 -08:00
|
|
|
self.archiver = UnknownArchiver( self.path )
|
2012-11-05 11:20:13 -08:00
|
|
|
|
|
|
|
def setExternalRarProgram( self, rar_exe_path ):
|
2012-11-05 11:46:48 -08:00
|
|
|
if self.isRar():
|
|
|
|
self.archiver.rar_exe_path = rar_exe_path
|
2012-11-05 11:20:13 -08:00
|
|
|
|
|
|
|
def zipTest( self ):
|
2012-11-02 13:54:17 -07:00
|
|
|
return zipfile.is_zipfile( self.path )
|
2012-11-05 11:20:13 -08:00
|
|
|
|
|
|
|
def rarTest( self ):
|
|
|
|
try:
|
|
|
|
rarc = UnRAR2.RarFile( self.path )
|
2012-11-05 13:34:23 -08:00
|
|
|
except: # InvalidRARArchive:
|
2012-11-05 11:20:13 -08:00
|
|
|
return False
|
|
|
|
else:
|
|
|
|
return True
|
|
|
|
|
|
|
|
|
|
|
|
def isZip( self ):
|
|
|
|
return self.archive_type == self.ArchiveType.Zip
|
2012-11-02 13:54:17 -07:00
|
|
|
|
2012-11-05 11:20:13 -08:00
|
|
|
def isRar( self ):
|
|
|
|
return self.archive_type == self.ArchiveType.Rar
|
|
|
|
|
2012-11-02 13:54:17 -07:00
|
|
|
def isFolder( self ):
|
2012-11-05 11:20:13 -08:00
|
|
|
return self.archive_type == self.ArchiveType.Folder
|
|
|
|
|
2012-11-27 11:37:04 -08:00
|
|
|
def isWritable( self ):
|
2012-11-05 11:20:13 -08:00
|
|
|
if self.archive_type == self.ArchiveType.Unknown :
|
|
|
|
return False
|
|
|
|
|
2012-11-05 11:46:48 -08:00
|
|
|
elif self.isRar() and self.archiver.rar_exe_path is None:
|
2012-11-05 11:20:13 -08:00
|
|
|
return False
|
|
|
|
|
|
|
|
elif not os.access(self.path, os.W_OK):
|
|
|
|
return False
|
2012-11-27 11:37:04 -08:00
|
|
|
|
|
|
|
elif ((self.archive_type != self.ArchiveType.Folder) and
|
|
|
|
(not os.access( os.path.dirname( os.path.abspath(self.path)), os.W_OK ))):
|
|
|
|
return False
|
2012-11-02 13:54:17 -07:00
|
|
|
|
2012-11-05 11:20:13 -08:00
|
|
|
return True
|
2012-11-02 13:54:17 -07:00
|
|
|
|
2012-11-20 13:19:08 -08:00
|
|
|
def isWritableForStyle( self, data_style ):
|
|
|
|
|
|
|
|
if self.isRar() and data_style == MetaDataStyle.CBI:
|
|
|
|
return False
|
|
|
|
|
|
|
|
return self.isWritable()
|
|
|
|
|
2012-11-02 13:54:17 -07:00
|
|
|
def seemsToBeAComicArchive( self ):
|
|
|
|
|
2012-11-17 16:32:01 -08:00
|
|
|
# Do we even care about extensions??
|
|
|
|
ext = os.path.splitext(self.path)[1].lower()
|
2012-11-05 11:20:13 -08:00
|
|
|
|
2012-11-17 16:32:01 -08:00
|
|
|
if (
|
|
|
|
( self.isZip() or self.isRar() or self.isFolder() )
|
2012-11-05 11:20:13 -08:00
|
|
|
and
|
2012-11-20 13:19:08 -08:00
|
|
|
( self.getNumberOfPages() > 2)
|
2012-11-05 11:20:13 -08:00
|
|
|
|
2012-11-02 13:54:17 -07:00
|
|
|
):
|
|
|
|
return True
|
|
|
|
else:
|
|
|
|
return False
|
|
|
|
|
|
|
|
def readMetadata( self, style ):
|
|
|
|
|
|
|
|
if style == MetaDataStyle.CIX:
|
|
|
|
return self.readCIX()
|
|
|
|
elif style == MetaDataStyle.CBI:
|
|
|
|
return self.readCBI()
|
2012-12-02 12:17:39 -08:00
|
|
|
elif style == MetaDataStyle.COMET:
|
|
|
|
return self.readCoMet()
|
2012-11-02 13:54:17 -07:00
|
|
|
else:
|
|
|
|
return GenericMetadata()
|
|
|
|
|
|
|
|
def writeMetadata( self, metadata, style ):
|
|
|
|
|
|
|
|
if style == MetaDataStyle.CIX:
|
2012-11-27 16:33:51 -08:00
|
|
|
return self.writeCIX( metadata )
|
2012-11-02 13:54:17 -07:00
|
|
|
elif style == MetaDataStyle.CBI:
|
2012-11-27 16:33:51 -08:00
|
|
|
return self.writeCBI( metadata )
|
2012-12-02 12:17:39 -08:00
|
|
|
elif style == MetaDataStyle.COMET:
|
|
|
|
return self.writeCoMet( metadata )
|
2012-11-02 13:54:17 -07:00
|
|
|
|
2012-11-04 11:24:48 -08:00
|
|
|
def hasMetadata( self, style ):
|
2012-11-02 13:54:17 -07:00
|
|
|
|
|
|
|
if style == MetaDataStyle.CIX:
|
|
|
|
return self.hasCIX()
|
|
|
|
elif style == MetaDataStyle.CBI:
|
|
|
|
return self.hasCBI()
|
2012-12-02 12:17:39 -08:00
|
|
|
elif style == MetaDataStyle.COMET:
|
|
|
|
return self.hasCoMet()
|
2012-11-02 13:54:17 -07:00
|
|
|
else:
|
|
|
|
return False
|
|
|
|
|
2012-11-04 11:24:48 -08:00
|
|
|
def removeMetadata( self, style ):
|
|
|
|
if style == MetaDataStyle.CIX:
|
2012-11-27 16:33:51 -08:00
|
|
|
return self.removeCIX()
|
2012-11-04 11:24:48 -08:00
|
|
|
elif style == MetaDataStyle.CBI:
|
2012-11-27 16:33:51 -08:00
|
|
|
return self.removeCBI()
|
2012-12-02 12:17:39 -08:00
|
|
|
elif style == MetaDataStyle.COMET:
|
|
|
|
return self.removeCoMet()
|
2012-11-02 13:54:17 -07:00
|
|
|
|
|
|
|
def getCoverPage(self):
|
|
|
|
|
2012-11-14 17:25:01 -08:00
|
|
|
# assume first page is the cover (for now)
|
|
|
|
return self.getPage( 0 )
|
|
|
|
|
|
|
|
def getPage( self, index ):
|
|
|
|
|
2012-11-16 11:32:46 -08:00
|
|
|
image_data = None
|
|
|
|
|
|
|
|
filename = self.getPageName( index )
|
|
|
|
|
|
|
|
if filename is not None:
|
|
|
|
image_data = self.archiver.readArchiveFile( filename )
|
|
|
|
|
|
|
|
return image_data
|
|
|
|
|
|
|
|
def getPageName( self, index ):
|
|
|
|
|
|
|
|
page_list = self.getPageNameList()
|
|
|
|
|
|
|
|
num_pages = len( page_list )
|
2012-11-14 17:25:01 -08:00
|
|
|
if num_pages == 0 or index >= num_pages:
|
2012-11-02 13:54:17 -07:00
|
|
|
return None
|
2012-11-16 11:32:46 -08:00
|
|
|
|
|
|
|
return page_list[index]
|
|
|
|
|
|
|
|
def getPageNameList( self , sort_list=True):
|
|
|
|
|
2012-11-02 13:54:17 -07:00
|
|
|
# get the list file names in the archive, and sort
|
2012-11-05 11:46:48 -08:00
|
|
|
files = self.archiver.getArchiveFilenameList()
|
2012-11-07 22:25:00 -08:00
|
|
|
|
2012-11-14 17:25:01 -08:00
|
|
|
# seems like some archive creators are on Windows, and don't know about case-sensitivity!
|
2012-11-16 11:32:46 -08:00
|
|
|
if sort_list:
|
|
|
|
files.sort(key=lambda x: x.lower())
|
2012-11-02 13:54:17 -07:00
|
|
|
|
2012-11-14 17:25:01 -08:00
|
|
|
# make a sub-list of image files
|
|
|
|
page_list = []
|
2012-11-02 13:54:17 -07:00
|
|
|
for name in files:
|
|
|
|
if ( name[-4:].lower() in [ ".jpg", "jpeg", ".png" ] ):
|
2012-11-14 17:25:01 -08:00
|
|
|
page_list.append(name)
|
2012-11-16 11:32:46 -08:00
|
|
|
|
|
|
|
return page_list
|
2012-11-02 13:54:17 -07:00
|
|
|
|
2012-11-16 11:32:46 -08:00
|
|
|
def getNumberOfPages( self ):
|
2012-11-02 13:54:17 -07:00
|
|
|
|
2012-11-16 11:32:46 -08:00
|
|
|
return len( self.getPageNameList( sort_list=False ) )
|
2012-11-02 13:54:17 -07:00
|
|
|
|
|
|
|
def readCBI( self ):
|
2012-11-19 12:28:19 -08:00
|
|
|
raw_cbi = self.readRawCBI()
|
|
|
|
if raw_cbi is None:
|
2012-11-02 13:54:17 -07:00
|
|
|
return GenericMetadata()
|
|
|
|
|
2012-11-19 12:28:19 -08:00
|
|
|
return ComicBookInfo().metadataFromString( raw_cbi )
|
|
|
|
|
|
|
|
def readRawCBI( self ):
|
|
|
|
if ( not self.hasCBI() ):
|
|
|
|
return None
|
|
|
|
|
|
|
|
return self.archiver.getArchiveComment()
|
2012-11-02 13:54:17 -07:00
|
|
|
|
2012-12-02 12:17:39 -08:00
|
|
|
def hasCBI(self):
|
|
|
|
#if ( not ( self.isZip() or self.isRar()) or not self.seemsToBeAComicArchive() ):
|
|
|
|
if not self.seemsToBeAComicArchive():
|
|
|
|
return False
|
2012-11-02 13:54:17 -07:00
|
|
|
|
2012-12-02 12:17:39 -08:00
|
|
|
comment = self.archiver.getArchiveComment()
|
|
|
|
return ComicBookInfo().validateString( comment )
|
|
|
|
|
2012-11-19 12:28:19 -08:00
|
|
|
def writeCBI( self, metadata ):
|
2012-11-02 13:54:17 -07:00
|
|
|
cbi_string = ComicBookInfo().stringFromMetadata( metadata )
|
2012-11-27 16:33:51 -08:00
|
|
|
return self.archiver.setArchiveComment( cbi_string )
|
2012-11-02 13:54:17 -07:00
|
|
|
|
2012-11-04 11:24:48 -08:00
|
|
|
def removeCBI( self ):
|
2012-11-27 16:33:51 -08:00
|
|
|
return self.archiver.setArchiveComment( "" )
|
2012-11-04 11:24:48 -08:00
|
|
|
|
2012-11-02 13:54:17 -07:00
|
|
|
def readCIX( self ):
|
2012-11-19 12:28:19 -08:00
|
|
|
raw_cix = self.readRawCIX()
|
|
|
|
if raw_cix is None:
|
|
|
|
return GenericMetadata()
|
|
|
|
|
|
|
|
return ComicInfoXml().metadataFromString( raw_cix )
|
|
|
|
|
|
|
|
def readRawCIX( self ):
|
2012-11-05 11:20:13 -08:00
|
|
|
if not self.hasCIX():
|
2012-12-03 17:15:12 -08:00
|
|
|
print self.path, "doesn't have ComicInfo.xml data!"
|
2012-11-19 12:28:19 -08:00
|
|
|
return None
|
2012-11-05 11:20:13 -08:00
|
|
|
|
2012-11-19 12:28:19 -08:00
|
|
|
return self.archiver.readArchiveFile( self.ci_xml_filename )
|
2012-11-02 13:54:17 -07:00
|
|
|
|
|
|
|
def writeCIX(self, metadata):
|
|
|
|
|
2012-11-05 11:20:13 -08:00
|
|
|
if metadata is not None:
|
2012-12-05 14:17:30 -08:00
|
|
|
metadata.pageCount = self.getNumberOfPages()
|
2012-11-04 11:24:48 -08:00
|
|
|
cix_string = ComicInfoXml().stringFromMetadata( metadata )
|
2012-11-27 16:33:51 -08:00
|
|
|
return self.archiver.writeArchiveFile( self.ci_xml_filename, cix_string )
|
|
|
|
else:
|
|
|
|
return False
|
|
|
|
|
2012-11-04 11:24:48 -08:00
|
|
|
def removeCIX( self ):
|
|
|
|
|
2012-11-27 16:33:51 -08:00
|
|
|
return self.archiver.removeArchiveFile( self.ci_xml_filename )
|
2012-11-04 11:24:48 -08:00
|
|
|
|
2012-11-02 13:54:17 -07:00
|
|
|
def hasCIX(self):
|
2012-11-05 11:20:13 -08:00
|
|
|
if not self.seemsToBeAComicArchive():
|
|
|
|
return False
|
2012-11-05 11:46:48 -08:00
|
|
|
elif self.ci_xml_filename in self.archiver.getArchiveFilenameList():
|
2012-11-05 11:20:13 -08:00
|
|
|
return True
|
|
|
|
else:
|
|
|
|
return False
|
2012-11-02 13:54:17 -07:00
|
|
|
|
2012-11-05 11:20:13 -08:00
|
|
|
|
2012-12-02 12:17:39 -08:00
|
|
|
def readCoMet( self ):
|
|
|
|
raw_comet = self.readRawCoMet()
|
|
|
|
if raw_comet is None:
|
|
|
|
return GenericMetadata()
|
|
|
|
|
|
|
|
return CoMet().metadataFromString( raw_comet )
|
|
|
|
|
|
|
|
def readRawCoMet( self ):
|
|
|
|
if not self.hasCoMet():
|
2012-12-03 17:15:12 -08:00
|
|
|
print self.path, "doesn't have CoMet data!"
|
2012-12-02 12:17:39 -08:00
|
|
|
return None
|
|
|
|
|
|
|
|
return self.archiver.readArchiveFile( self.comet_filename )
|
|
|
|
|
|
|
|
def writeCoMet(self, metadata):
|
|
|
|
|
|
|
|
if metadata is not None:
|
2012-12-05 14:17:30 -08:00
|
|
|
if not self.hasCoMet():
|
|
|
|
self.comet_filename = self.comet_default_filename
|
|
|
|
|
|
|
|
metadata.pageCount = self.getNumberOfPages()
|
2012-12-02 12:17:39 -08:00
|
|
|
comet_string = CoMet().stringFromMetadata( metadata )
|
|
|
|
return self.archiver.writeArchiveFile( self.comet_filename, comet_string )
|
|
|
|
else:
|
|
|
|
return False
|
|
|
|
|
|
|
|
def removeCoMet( self ):
|
2012-12-05 14:17:30 -08:00
|
|
|
if self.hasCoMet():
|
|
|
|
retcode = self.archiver.removeArchiveFile( self.comet_filename )
|
|
|
|
self.comet_filename = None
|
|
|
|
return retcode
|
|
|
|
return True
|
2012-12-02 12:17:39 -08:00
|
|
|
|
|
|
|
def hasCoMet(self):
|
|
|
|
if not self.seemsToBeAComicArchive():
|
|
|
|
return False
|
2012-12-05 14:17:30 -08:00
|
|
|
|
|
|
|
#Use the existence of self.comet_filename as a cue that the tag block exists
|
|
|
|
if self.comet_filename is None:
|
|
|
|
#TODO look at all xml files in root, and search for CoMet data, get first
|
|
|
|
for n in self.archiver.getArchiveFilenameList():
|
|
|
|
if ( os.path.dirname(n) == "" and
|
|
|
|
os.path.splitext(n)[1].lower() == '.xml'):
|
|
|
|
# read in XML file, and validate it
|
|
|
|
data = self.archiver.readArchiveFile( n )
|
|
|
|
if CoMet().validateString( data ):
|
|
|
|
# since we found it, save it!
|
|
|
|
self.comet_filename = n
|
|
|
|
return True
|
|
|
|
# if we made it through the loop, no CoMet here...
|
2012-11-02 13:54:17 -07:00
|
|
|
return False
|
2012-12-05 14:17:30 -08:00
|
|
|
|
|
|
|
else:
|
|
|
|
return True
|
2012-11-05 11:20:13 -08:00
|
|
|
|
2012-11-02 13:54:17 -07:00
|
|
|
|
|
|
|
def metadataFromFilename( self ):
|
|
|
|
|
|
|
|
metadata = GenericMetadata()
|
|
|
|
|
|
|
|
fnp = FileNameParser()
|
|
|
|
fnp.parseFilename( self.path )
|
|
|
|
|
|
|
|
if fnp.issue != "":
|
2012-11-19 11:57:16 -08:00
|
|
|
metadata.issue = fnp.issue
|
2012-11-02 13:54:17 -07:00
|
|
|
if fnp.series != "":
|
|
|
|
metadata.series = fnp.series
|
|
|
|
if fnp.volume != "":
|
2012-11-19 11:57:16 -08:00
|
|
|
metadata.volume = fnp.volume
|
2012-11-02 13:54:17 -07:00
|
|
|
if fnp.year != "":
|
2012-11-19 11:57:16 -08:00
|
|
|
metadata.year = fnp.year
|
2012-11-18 21:15:16 -08:00
|
|
|
if fnp.issue_count != "":
|
|
|
|
metadata.issueCount = fnp.issue_count
|
2012-11-02 13:54:17 -07:00
|
|
|
|
|
|
|
metadata.isEmpty = False
|
2012-11-18 19:55:40 -08:00
|
|
|
|
2012-11-05 11:20:13 -08:00
|
|
|
return metadata
|
2012-12-05 21:45:53 -08:00
|
|
|
|
|
|
|
def exportAsZip( self, zipfilename ):
|
|
|
|
if self.archive_type == self.ArchiveType.Zip:
|
|
|
|
# nothing to do, we're already a zip
|
|
|
|
return True
|
|
|
|
|
|
|
|
zip_archiver = ZipArchiver( zipfilename )
|
|
|
|
return zip_archiver.copyFromArchive( self.archiver )
|
|
|
|
|