diff --git a/DeDRM_calibre_plugin/DeDRM_plugin/kfxdedrm.py b/DeDRM_calibre_plugin/DeDRM_plugin/kfxdedrm.py new file mode 100644 index 0000000..8885e09 --- /dev/null +++ b/DeDRM_calibre_plugin/DeDRM_plugin/kfxdedrm.py @@ -0,0 +1,108 @@ +#!/usr/bin/env python +# -*- coding: utf-8 -*- + +from __future__ import with_statement + +# Engine to remove drm from Kindle KFX ebooks + +import os +import shutil +import zipfile + +try: + from cStringIO import StringIO +except ImportError: + from StringIO import StringIO + +try: + import ion +except: + from calibre_plugins.dedrm import ion + + +__license__ = 'GPL v3' +__version__ = '1.0' + + +class KFXZipBook: + def __init__(self, infile): + self.infile = infile + self.voucher = None + self.decrypted = {} + + def getPIDMetaInfo(self): + return (None, None) + + def processBook(self, totalpids): + with zipfile.ZipFile(self.infile, 'r') as zf: + for filename in zf.namelist(): + data = zf.read(filename) + if data.startswith('\xeaDRMION\xee'): + if self.voucher is None: + self.decrypt_voucher(totalpids) + print u'Decrypting KFX DRMION: {0}'.format(filename) + outfile = StringIO() + ion.DrmIon(StringIO(data[8:-8]), lambda name: self.voucher).parse(outfile) + self.decrypted[filename] = outfile.getvalue() + + if not self.decrypted: + print(u'The .kfx-zip archive does not contain an encrypted DRMION file') + + def decrypt_voucher(self, totalpids): + with zipfile.ZipFile(self.infile, 'r') as zf: + for info in zf.infolist(): + if info.file_size < 0x10000: + data = zf.read(info.filename) + if data.startswith('\xe0\x01\x00\xea') and 'ProtectedData' in data: + break # found DRM voucher + else: + raise Exception(u'The .kfx-zip archive contains an encrypted DRMION file without a DRM voucher') + + print u'Decrypting KFX DRM voucher: {0}'.format(info.filename) + + for pid in [''] + totalpids: + for dsn_len,secret_len in [(0,0), (16,0), (16,40), (32,40), (40,40)]: + if len(pid) == dsn_len + secret_len: + break # split pid into DSN and account secret + else: + continue + + try: + voucher = ion.DrmIonVoucher(StringIO(data), pid[:dsn_len], pid[dsn_len:]) + voucher.parse() + voucher.decryptvoucher() + break + except: + pass + else: + raise Exception(u'Failed to decrypt KFX DRM voucher with any key') + + print u'KFX DRM voucher successfully decrypted' + + license_type = voucher.getlicensetype() + if license_type != "Purchase": + raise Exception((u'This book is licensed as {0}. ' + 'These tools are intended for use on purchased books.').format(license_type)) + + self.voucher = voucher + + def getBookTitle(self): + return os.path.splitext(os.path.split(self.infile)[1])[0] + + def getBookExtension(self): + return '.kfx-zip' + + def getBookType(self): + return 'KFX-ZIP' + + def cleanup(self): + pass + + def getFile(self, outpath): + if not self.decrypted: + shutil.copyfile(self.infile, outpath) + else: + with zipfile.ZipFile(self.infile, 'r') as zif: + with zipfile.ZipFile(outpath, 'w') as zof: + for info in zif.infolist(): + zof.writestr(info, self.decrypted.get(info.filename, zif.read(info.filename)))