2011-06-16 11:59:20 +06:00
#!/usr/bin/env python
2012-11-07 19:14:25 +06:00
# vim:fileencoding=UTF-8:ts=4:sw=4:sta:et:sts=4:ai
2011-06-16 11:59:20 +06:00
from __future__ import with_statement
from calibre . customize import FileTypePlugin
from calibre . gui2 import is_ok_to_use_qt
2012-03-07 00:24:28 +06:00
from calibre . utils . config import config_dir
from calibre . constants import iswindows , isosx
2011-06-16 11:59:20 +06:00
# from calibre.ptempfile import PersistentTemporaryDirectory
import sys
import os
import re
2012-11-20 19:28:12 +06:00
import time
2012-03-07 00:24:28 +06:00
from zipfile import ZipFile
2011-06-16 11:59:20 +06:00
class K4DeDRM ( FileTypePlugin ) :
2012-09-09 06:45:24 +06:00
name = ' Kindle and Mobipocket DeDRM ' # Name of the plugin
description = ' Removes DRM from eInk Kindle, Kindle 4 Mac and Kindle 4 PC ebooks, and from Mobipocket ebooks. Provided by the work of many including DiapDealer, SomeUpdates, IHeartCabbages, CMBDTC, Skindle, DarkReverser, mdlnx, ApprenticeAlf, etc. '
2011-06-16 11:59:20 +06:00
supported_platforms = [ ' osx ' , ' windows ' , ' linux ' ] # Platforms this plugin will run on
2012-09-09 06:45:24 +06:00
author = ' DiapDealer, SomeUpdates, mdlnx, Apprentice Alf ' # The author of this plugin
2012-11-20 19:28:12 +06:00
version = ( 0 , 4 , 7 ) # The version number of this plugin
2012-05-16 22:15:43 +06:00
file_types = set ( [ ' prc ' , ' mobi ' , ' azw ' , ' azw1 ' , ' azw3 ' , ' azw4 ' , ' tpz ' ] ) # The file types that this plugin will be applied to
2011-06-16 11:59:20 +06:00
on_import = True # Run this plugin during the import
2012-09-09 06:45:24 +06:00
priority = 520 # run this plugin before earlier versions
2011-06-16 11:59:20 +06:00
minimum_calibre_version = ( 0 , 7 , 55 )
2012-09-09 06:45:24 +06:00
2012-03-07 00:24:28 +06:00
def initialize ( self ) :
"""
Dynamic modules can ' t be imported/loaded from a zipfile... so this routine
runs whenever the plugin gets initialized . This will extract the appropriate
library for the target OS and copy it to the ' alfcrypto ' subdirectory of
calibre ' s configuration directory. That ' alfcrypto ' directory is then
inserted into the syspath ( as the very first entry ) in the run function
so the CDLL stuff will work in the alfcrypto . py script .
"""
if iswindows :
names = [ ' alfcrypto.dll ' , ' alfcrypto64.dll ' ]
elif isosx :
names = [ ' libalfcrypto.dylib ' ]
else :
2012-11-07 19:14:25 +06:00
names = [ ' libalfcrypto32.so ' , ' libalfcrypto64.so ' , ' alfcrypto.py ' , ' alfcrypto.dll ' , ' alfcrypto64.dll ' , ' getk4pcpids.py ' , ' mobidedrm.py ' , ' kgenpids.py ' , ' k4pcutils.py ' , ' topazextract.py ' , ' outputfix.py ' ]
2012-03-07 00:24:28 +06:00
lib_dict = self . load_resources ( names )
self . alfdir = os . path . join ( config_dir , ' alfcrypto ' )
if not os . path . exists ( self . alfdir ) :
os . mkdir ( self . alfdir )
for entry , data in lib_dict . items ( ) :
file_path = os . path . join ( self . alfdir , entry )
with open ( file_path , ' wb ' ) as f :
f . write ( data )
2011-06-16 11:59:20 +06:00
def run ( self , path_to_ebook ) :
2012-11-20 19:28:12 +06:00
# add the alfcrypto directory to sys.path so alfcrypto.py
2012-03-07 00:24:28 +06:00
# will be able to locate the custom lib(s) for CDLL import.
sys . path . insert ( 0 , self . alfdir )
# Had to move these imports here so the custom libs can be
# extracted to the appropriate places beforehand these routines
# look for them.
2012-11-07 19:14:25 +06:00
from calibre_plugins . k4mobidedrm import kgenpids , topazextract , mobidedrm , outputfix
if sys . stdout . encoding == None :
sys . stdout = outputfix . getwriter ( ' utf-8 ' ) ( sys . stdout )
else :
sys . stdout = outputfix . getwriter ( sys . stdout . encoding ) ( sys . stdout )
if sys . stderr . encoding == None :
sys . stderr = outputfix . getwriter ( ' utf-8 ' ) ( sys . stderr )
else :
sys . stderr = outputfix . getwriter ( sys . stderr . encoding ) ( sys . stderr )
2012-03-07 00:24:28 +06:00
2011-09-05 12:17:36 +06:00
plug_ver = ' . ' . join ( str ( self . version ) . strip ( ' () ' ) . replace ( ' ' , ' ' ) . split ( ' , ' ) )
2011-06-16 11:59:20 +06:00
k4 = True
pids = [ ]
serials = [ ]
kInfoFiles = [ ]
2012-11-20 19:28:12 +06:00
starttime = time . time ( )
print " K4MobiDeDRM plugin v {0:s} : Starting " . format ( plug_ver )
2012-09-09 06:45:24 +06:00
self . config ( )
2012-11-20 19:28:12 +06:00
2011-06-16 11:59:20 +06:00
# Get supplied list of PIDs to try from plugin customization.
2012-09-09 06:45:24 +06:00
pidstringlistt = self . pids_string . split ( ' , ' )
for pid in pidstringlistt :
pid = str ( pid ) . strip ( )
if len ( pid ) == 10 or len ( pid ) == 8 :
pids . append ( pid )
else :
if len ( pid ) > 0 :
print " ' %s ' is not a valid Mobipocket PID. " % pid
2012-11-20 19:28:12 +06:00
2012-09-09 06:45:24 +06:00
# For linux, get PIDs by calling the right routines under WINE
if sys . platform . startswith ( ' linux ' ) :
k4 = False
pids . extend ( self . WINEgetPIDs ( path_to_ebook ) )
2012-11-20 19:28:12 +06:00
2012-09-09 06:45:24 +06:00
# Get supplied list of Kindle serial numbers to try from plugin customization.
serialstringlistt = self . serials_string . split ( ' , ' )
for serial in serialstringlistt :
2012-11-20 19:28:12 +06:00
serial = str ( serial ) . replace ( " " , " " )
2012-09-09 06:45:24 +06:00
if len ( serial ) == 16 and serial [ 0 ] == ' B ' :
serials . append ( serial )
else :
if len ( serial ) > 0 :
print " ' %s ' is not a valid Kindle serial number. " % serial
2012-11-20 19:28:12 +06:00
2011-06-16 11:59:20 +06:00
# Load any kindle info files (*.info) included Calibre's config directory.
try :
2012-09-09 06:45:24 +06:00
print ' K4MobiDeDRM v %s : Calibre configuration directory = %s ' % ( plug_ver , config_dir )
files = os . listdir ( config_dir )
2011-06-16 11:59:20 +06:00
filefilter = re . compile ( " \ .info$| \ .kinf$ " , re . IGNORECASE )
files = filter ( filefilter . search , files )
if files :
for filename in files :
2012-09-09 06:45:24 +06:00
fpath = os . path . join ( config_dir , filename )
2011-06-16 11:59:20 +06:00
kInfoFiles . append ( fpath )
2011-09-05 12:17:36 +06:00
print ' K4MobiDeDRM v %s : Kindle info/kinf file %s found in config folder. ' % ( plug_ver , filename )
2011-06-16 11:59:20 +06:00
except IOError :
2011-09-05 12:17:36 +06:00
print ' K4MobiDeDRM v %s : Error reading kindle info/kinf files from config directory. ' % plug_ver
2011-06-16 11:59:20 +06:00
pass
mobi = True
magic3 = file ( path_to_ebook , ' rb ' ) . read ( 3 )
if magic3 == ' TPZ ' :
mobi = False
bookname = os . path . splitext ( os . path . basename ( path_to_ebook ) ) [ 0 ]
if mobi :
mb = mobidedrm . MobiBook ( path_to_ebook )
else :
mb = topazextract . TopazBook ( path_to_ebook )
title = mb . getBookTitle ( )
md1 , md2 = mb . getPIDMetaInfo ( )
2012-11-20 19:28:12 +06:00
pids . extend ( kgenpids . getPidList ( md1 , md2 , k4 , serials , kInfoFiles ) )
print " K4MobiDeDRM plugin v {2:s} : Found {1:d} keys to try after {0:.1f} seconds " . format ( time . time ( ) - starttime , len ( pids ) , plug_ver )
2011-06-16 11:59:20 +06:00
try :
2012-11-20 19:28:12 +06:00
mb . processBook ( pids )
2011-06-16 11:59:20 +06:00
2011-09-05 12:17:36 +06:00
except mobidedrm . DrmException , e :
2011-06-16 11:59:20 +06:00
#if you reached here then no luck raise and exception
if is_ok_to_use_qt ( ) :
from PyQt4 . Qt import QMessageBox
2011-09-05 12:17:36 +06:00
d = QMessageBox ( QMessageBox . Warning , " K4MobiDeDRM v %s Plugin " % plug_ver , " Error: " + str ( e ) + " ... %s \n " % path_to_ebook )
2011-06-16 11:59:20 +06:00
d . show ( )
d . raise_ ( )
d . exec_ ( )
2012-11-20 19:28:12 +06:00
raise Exception ( " K4MobiDeDRM plugin v {1:s} Error: {2:s} after {0:.1f} seconds " . format ( time . time ( ) - starttime , plug_ver , str ( e ) ) )
2011-09-05 12:17:36 +06:00
except topazextract . TpzDRMError , e :
2011-06-16 11:59:20 +06:00
#if you reached here then no luck raise and exception
if is_ok_to_use_qt ( ) :
2012-03-07 00:24:28 +06:00
from PyQt4 . Qt import QMessageBox
d = QMessageBox ( QMessageBox . Warning , " K4MobiDeDRM v %s Plugin " % plug_ver , " Error: " + str ( e ) + " ... %s \n " % path_to_ebook )
d . show ( )
d . raise_ ( )
d . exec_ ( )
2012-11-20 19:28:12 +06:00
raise Exception ( " K4MobiDeDRM plugin v {1:s} Error: {2:s} after {0:.1f} seconds " . format ( time . time ( ) - starttime , plug_ver , str ( e ) ) )
2011-06-16 11:59:20 +06:00
2012-11-20 19:28:12 +06:00
print " K4MobiDeDRM plugin v {1:s} : Successfully decrypted book after {0:.1f} seconds " . format ( time . time ( ) - starttime , plug_ver )
2011-06-16 11:59:20 +06:00
if mobi :
2011-09-05 12:17:36 +06:00
if mb . getPrintReplica ( ) :
of = self . temporary_file ( bookname + ' .azw4 ' )
2012-11-20 19:28:12 +06:00
print ' K4MobiDeDRM plugin v %s : Print Replica format detected. ' % plug_ver
2012-05-16 22:15:43 +06:00
elif mb . getMobiVersion ( ) > = 8 :
2012-11-20 19:28:12 +06:00
print ' K4MobiDeDRM plugin v %s : Stand-alone KF8 format detected. ' % plug_ver
2012-05-16 22:15:43 +06:00
of = self . temporary_file ( bookname + ' .azw3 ' )
2011-09-05 12:17:36 +06:00
else :
of = self . temporary_file ( bookname + ' .mobi ' )
2011-06-16 11:59:20 +06:00
mb . getMobiFile ( of . name )
2012-11-20 19:28:12 +06:00
print " K4MobiDeDRM plugin v {1:s} : Saved decrypted book after {0:.1f} seconds " . format ( time . time ( ) - starttime , plug_ver )
2011-09-05 12:17:36 +06:00
else :
2011-06-16 11:59:20 +06:00
of = self . temporary_file ( bookname + ' .htmlz ' )
mb . getHTMLZip ( of . name )
mb . cleanup ( )
2012-11-20 19:28:12 +06:00
print " K4MobiDeDRM plugin v {1:s} : Saved decrypted Topaz HTMLZ after {0:.1f} seconds " . format ( time . time ( ) - starttime , plug_ver )
2011-06-16 11:59:20 +06:00
return of . name
2012-09-09 06:45:24 +06:00
def WINEgetPIDs ( self , infile ) :
import subprocess
from subprocess import Popen , PIPE , STDOUT
import subasyncio
from subasyncio import Process
print " Getting PIDs from WINE "
outfile = os . path . join ( self . alfdir + ' winepids.txt ' )
2012-09-11 22:57:03 +06:00
# Remove any previous winepids.txt file.
if os . path . exists ( outfile ) :
os . remove ( outfile )
2012-09-09 06:45:24 +06:00
cmdline = ' wine python.exe ' \
+ ' " ' + self . alfdir + ' /getk4pcpids.py " ' \
+ ' " ' + infile + ' " ' \
+ ' " ' + outfile + ' " '
env = os . environ
2012-11-20 19:28:12 +06:00
2012-09-09 06:45:24 +06:00
print " My wine_prefix from tweaks is " , self . wine_prefix
if ( " WINEPREFIX " in env ) :
print " Using WINEPREFIX from the environment: " , env [ " WINEPREFIX " ]
elif ( self . wine_prefix is not None ) :
env [ ' WINEPREFIX ' ] = self . wine_prefix
print " Using WINEPREFIX from tweaks: " , self . wine_prefix
else :
print " No wine prefix used "
print cmdline
2012-09-11 22:57:03 +06:00
try :
cmdline = cmdline . encode ( sys . getfilesystemencoding ( ) )
p2 = Process ( cmdline , shell = True , bufsize = 1 , stdin = None , stdout = sys . stdout , stderr = STDOUT , close_fds = False )
result = p2 . wait ( " wait " )
except Exception , e :
print " WINE subprocess error " , str ( e )
return [ ]
print " WINE subprocess returned " , result
2012-11-20 19:28:12 +06:00
2012-09-09 06:45:24 +06:00
WINEpids = [ ]
2012-09-11 22:57:03 +06:00
if os . path . exists ( outfile ) :
try :
customvalues = file ( outfile , ' r ' ) . readline ( ) . split ( ' , ' )
for customvalue in customvalues :
customvalue = str ( customvalue )
customvalue = customvalue . strip ( )
if len ( customvalue ) == 10 or len ( customvalue ) == 8 :
WINEpids . append ( customvalue )
else :
print " ' %s ' is not a valid PID. " % customvalue
except Exception , e :
print " Error parsing winepids.txt: " , str ( e )
return [ ]
else :
print " No PIDs generated by Wine Python subprocess. "
2012-09-09 06:45:24 +06:00
return WINEpids
def is_customizable ( self ) :
# return true to allow customization via the Plugin->Preferences.
return True
def config_widget ( self ) :
# It is important to put this import statement here rather than at the
# top of the module as importing the config class will also cause the
# GUI libraries to be loaded, which we do not want when using calibre
# from the command line
from calibre_plugins . k4mobidedrm . config import ConfigWidget
return config . ConfigWidget ( )
2012-11-20 19:28:12 +06:00
2012-09-09 06:45:24 +06:00
def config ( self ) :
from calibre_plugins . k4mobidedrm . config import prefs
2012-11-20 19:28:12 +06:00
2012-09-09 06:45:24 +06:00
self . pids_string = prefs [ ' pids ' ]
self . serials_string = prefs [ ' serials ' ]
self . wine_prefix = prefs [ ' WINEPREFIX ' ]
2012-11-20 19:28:12 +06:00
2012-09-09 06:45:24 +06:00
def save_settings ( self , config_widget ) :
'''
Save the settings specified by the user with config_widget .
'''
config_widget . save_settings ( )
self . config ( )
2012-03-07 00:24:28 +06:00
def load_resources ( self , names ) :
ans = { }
with ZipFile ( self . plugin_path , ' r ' ) as zf :
for candidate in zf . namelist ( ) :
if candidate in names :
ans [ candidate ] = zf . read ( candidate )
2012-11-20 19:28:12 +06:00
return ans