2020-08-22 16:31:00 -04:00
|
|
|
from __future__ import division, print_function, unicode_literals
|
|
|
|
import sys
|
|
|
|
import os
|
|
|
|
import re
|
|
|
|
|
|
|
|
from glob import glob
|
|
|
|
from shutil import copyfile
|
|
|
|
|
2020-08-23 13:07:24 -04:00
|
|
|
from sqlalchemy.exc import SQLAlchemyError
|
|
|
|
|
2020-10-27 11:06:43 +01:00
|
|
|
from cps.services.worker import CalibreTask
|
2020-12-08 08:01:42 +01:00
|
|
|
from cps import db
|
2020-08-22 16:31:00 -04:00
|
|
|
from cps import logger, config
|
|
|
|
from cps.subproc_wrapper import process_open
|
|
|
|
from flask_babel import gettext as _
|
|
|
|
|
2020-08-30 08:49:53 +02:00
|
|
|
from cps.tasks.mail import TaskEmail
|
2020-08-23 21:21:55 -04:00
|
|
|
from cps import gdriveutils
|
2020-08-22 16:31:00 -04:00
|
|
|
log = logger.create()
|
|
|
|
|
|
|
|
|
|
|
|
class TaskConvert(CalibreTask):
|
2020-10-03 21:43:48 +02:00
|
|
|
def __init__(self, file_path, bookid, taskMessage, settings, kindle_mail, user=None):
|
2020-08-23 21:21:55 -04:00
|
|
|
super(TaskConvert, self).__init__(taskMessage)
|
2020-08-22 16:31:00 -04:00
|
|
|
self.file_path = file_path
|
|
|
|
self.bookid = bookid
|
|
|
|
self.settings = settings
|
|
|
|
self.kindle_mail = kindle_mail
|
2020-10-03 21:43:48 +02:00
|
|
|
self.user = user
|
2020-08-22 16:31:00 -04:00
|
|
|
|
|
|
|
self.results = dict()
|
|
|
|
|
|
|
|
def run(self, worker_thread):
|
|
|
|
self.worker_thread = worker_thread
|
2020-10-04 12:09:52 +02:00
|
|
|
if config.config_use_google_drive:
|
2020-12-08 08:04:46 +01:00
|
|
|
worker_db = db.CalibreDB(expire_on_commit=False)
|
2020-12-08 08:01:42 +01:00
|
|
|
cur_book = worker_db.get_book(self.bookid)
|
|
|
|
data = worker_db.get_book_format(self.bookid, self.settings['old_book_format'])
|
2020-10-04 12:09:52 +02:00
|
|
|
df = gdriveutils.getFileFromEbooksFolder(cur_book.path,
|
|
|
|
data.name + "." + self.settings['old_book_format'].lower())
|
|
|
|
if df:
|
|
|
|
datafile = os.path.join(config.config_calibre_dir,
|
|
|
|
cur_book.path,
|
|
|
|
data.name + u"." + self.settings['old_book_format'].lower())
|
|
|
|
if not os.path.exists(os.path.join(config.config_calibre_dir, cur_book.path)):
|
|
|
|
os.makedirs(os.path.join(config.config_calibre_dir, cur_book.path))
|
|
|
|
df.GetContentFile(datafile)
|
2020-12-08 08:04:46 +01:00
|
|
|
worker_db.session.close()
|
2020-10-04 12:09:52 +02:00
|
|
|
else:
|
|
|
|
error_message = _(u"%(format)s not found on Google Drive: %(fn)s",
|
|
|
|
format=self.settings['old_book_format'],
|
|
|
|
fn=data.name + "." + self.settings['old_book_format'].lower())
|
2020-12-08 08:04:46 +01:00
|
|
|
worker_db.session.close()
|
2020-10-04 12:09:52 +02:00
|
|
|
return error_message
|
|
|
|
|
2020-08-22 16:31:00 -04:00
|
|
|
filename = self._convert_ebook_format()
|
2020-10-04 13:59:33 +02:00
|
|
|
if config.config_use_google_drive:
|
|
|
|
os.remove(self.file_path + u'.' + self.settings['old_book_format'].lower())
|
2020-08-23 21:35:05 -04:00
|
|
|
|
2020-08-23 21:21:55 -04:00
|
|
|
if filename:
|
|
|
|
if config.config_use_google_drive:
|
2020-10-04 12:09:52 +02:00
|
|
|
# Upload files to gdrive
|
2020-08-23 21:21:55 -04:00
|
|
|
gdriveutils.updateGdriveCalibreFromLocal()
|
2020-10-04 12:09:52 +02:00
|
|
|
self._handleSuccess()
|
2020-08-23 21:21:55 -04:00
|
|
|
if self.kindle_mail:
|
|
|
|
# if we're sending to kindle after converting, create a one-off task and run it immediately
|
|
|
|
# todo: figure out how to incorporate this into the progress
|
2020-08-23 21:35:05 -04:00
|
|
|
try:
|
2021-01-03 19:27:24 +01:00
|
|
|
worker_thread.add(self.user, TaskEmail(self.settings['subject'],
|
|
|
|
self.results["path"],
|
|
|
|
filename,
|
|
|
|
self.settings,
|
|
|
|
self.kindle_mail,
|
|
|
|
self.settings['subject'],
|
|
|
|
self.settings['body'],
|
|
|
|
internal=True)
|
|
|
|
)
|
2020-08-23 21:35:05 -04:00
|
|
|
except Exception as e:
|
|
|
|
return self._handleError(str(e))
|
2020-08-22 16:31:00 -04:00
|
|
|
|
|
|
|
def _convert_ebook_format(self):
|
|
|
|
error_message = None
|
2020-12-08 08:04:46 +01:00
|
|
|
local_db = db.CalibreDB(expire_on_commit=False)
|
2020-08-22 16:31:00 -04:00
|
|
|
file_path = self.file_path
|
|
|
|
book_id = self.bookid
|
|
|
|
format_old_ext = u'.' + self.settings['old_book_format'].lower()
|
|
|
|
format_new_ext = u'.' + self.settings['new_book_format'].lower()
|
|
|
|
|
2020-12-08 13:34:15 +01:00
|
|
|
# check to see if destination format already exists - or if book is in database
|
2020-08-22 16:31:00 -04:00
|
|
|
# if it does - mark the conversion task as complete and return a success
|
|
|
|
# this will allow send to kindle workflow to continue to work
|
2020-12-08 13:34:15 +01:00
|
|
|
if os.path.isfile(file_path + format_new_ext) or\
|
|
|
|
local_db.get_book_format(self.bookid, self.settings['new_book_format']):
|
2020-08-22 16:31:00 -04:00
|
|
|
log.info("Book id %d already converted to %s", book_id, format_new_ext)
|
2020-12-08 11:39:23 +01:00
|
|
|
cur_book = local_db.get_book(book_id)
|
2020-08-22 16:31:00 -04:00
|
|
|
self.results['path'] = file_path
|
|
|
|
self.results['title'] = cur_book.title
|
|
|
|
self._handleSuccess()
|
2020-12-08 08:01:42 +01:00
|
|
|
local_db.session.close()
|
2020-09-30 19:29:57 +02:00
|
|
|
return os.path.basename(file_path + format_new_ext)
|
2020-08-22 16:31:00 -04:00
|
|
|
else:
|
|
|
|
log.info("Book id %d - target format of %s does not exist. Moving forward with convert.",
|
|
|
|
book_id,
|
|
|
|
format_new_ext)
|
|
|
|
|
|
|
|
if config.config_kepubifypath and format_old_ext == '.epub' and format_new_ext == '.kepub':
|
|
|
|
check, error_message = self._convert_kepubify(file_path,
|
|
|
|
format_old_ext,
|
|
|
|
format_new_ext)
|
|
|
|
else:
|
|
|
|
# check if calibre converter-executable is existing
|
|
|
|
if not os.path.exists(config.config_converterpath):
|
|
|
|
# ToDo Text is not translated
|
|
|
|
self._handleError(_(u"Calibre ebook-convert %(tool)s not found", tool=config.config_converterpath))
|
|
|
|
return
|
|
|
|
check, error_message = self._convert_calibre(file_path, format_old_ext, format_new_ext)
|
|
|
|
|
|
|
|
if check == 0:
|
2020-12-08 08:01:42 +01:00
|
|
|
cur_book = local_db.get_book(book_id)
|
2020-08-22 16:31:00 -04:00
|
|
|
if os.path.isfile(file_path + format_new_ext):
|
|
|
|
new_format = db.Data(name=cur_book.data[0].name,
|
|
|
|
book_format=self.settings['new_book_format'].upper(),
|
|
|
|
book=book_id, uncompressed_size=os.path.getsize(file_path + format_new_ext))
|
2020-08-23 13:07:24 -04:00
|
|
|
try:
|
2020-12-08 08:01:42 +01:00
|
|
|
local_db.session.merge(new_format)
|
2021-01-10 11:01:54 +01:00
|
|
|
local_db.session.flush()
|
2020-12-08 08:01:42 +01:00
|
|
|
local_db.session.commit()
|
2020-08-23 13:07:24 -04:00
|
|
|
except SQLAlchemyError as e:
|
2020-12-08 13:34:15 +01:00
|
|
|
local_db.session.rollback()
|
2020-08-23 13:07:24 -04:00
|
|
|
log.error("Database error: %s", e)
|
2020-12-08 08:01:42 +01:00
|
|
|
local_db.session.close()
|
2020-08-23 13:33:57 -04:00
|
|
|
return
|
2020-08-22 16:31:00 -04:00
|
|
|
self.results['path'] = cur_book.path
|
|
|
|
self.results['title'] = cur_book.title
|
2020-10-04 13:59:33 +02:00
|
|
|
if not config.config_use_google_drive:
|
2020-10-04 12:09:52 +02:00
|
|
|
self._handleSuccess()
|
2020-09-30 19:29:57 +02:00
|
|
|
return os.path.basename(file_path + format_new_ext)
|
2020-08-22 16:31:00 -04:00
|
|
|
else:
|
2020-08-29 11:14:52 +02:00
|
|
|
error_message = _('%(format)s format not found on disk', format=format_new_ext.upper())
|
2020-12-08 08:01:42 +01:00
|
|
|
local_db.session.close()
|
2020-08-22 16:31:00 -04:00
|
|
|
log.info("ebook converter failed with error while converting book")
|
|
|
|
if not error_message:
|
2020-08-29 11:14:52 +02:00
|
|
|
error_message = _('Ebook converter failed with unknown error')
|
2020-08-22 16:31:00 -04:00
|
|
|
self._handleError(error_message)
|
|
|
|
return
|
|
|
|
|
|
|
|
def _convert_kepubify(self, file_path, format_old_ext, format_new_ext):
|
|
|
|
quotes = [1, 3]
|
|
|
|
command = [config.config_kepubifypath, (file_path + format_old_ext), '-o', os.path.dirname(file_path)]
|
|
|
|
try:
|
|
|
|
p = process_open(command, quotes)
|
|
|
|
except OSError as e:
|
|
|
|
return 1, _(u"Kepubify-converter failed: %(error)s", error=e)
|
|
|
|
self.progress = 0.01
|
|
|
|
while True:
|
|
|
|
nextline = p.stdout.readlines()
|
|
|
|
nextline = [x.strip('\n') for x in nextline if x != '\n']
|
|
|
|
if sys.version_info < (3, 0):
|
|
|
|
nextline = [x.decode('utf-8') for x in nextline]
|
|
|
|
for line in nextline:
|
|
|
|
log.debug(line)
|
|
|
|
if p.poll() is not None:
|
|
|
|
break
|
|
|
|
|
|
|
|
# ToD Handle
|
|
|
|
# process returncode
|
|
|
|
check = p.returncode
|
|
|
|
|
|
|
|
# move file
|
|
|
|
if check == 0:
|
|
|
|
converted_file = glob(os.path.join(os.path.dirname(file_path), "*.kepub.epub"))
|
|
|
|
if len(converted_file) == 1:
|
|
|
|
copyfile(converted_file[0], (file_path + format_new_ext))
|
|
|
|
os.unlink(converted_file[0])
|
|
|
|
else:
|
|
|
|
return 1, _(u"Converted file not found or more than one file in folder %(folder)s",
|
|
|
|
folder=os.path.dirname(file_path))
|
|
|
|
return check, None
|
|
|
|
|
|
|
|
def _convert_calibre(self, file_path, format_old_ext, format_new_ext):
|
|
|
|
try:
|
|
|
|
# Linux py2.7 encode as list without quotes no empty element for parameters
|
|
|
|
# linux py3.x no encode and as list without quotes no empty element for parameters
|
|
|
|
# windows py2.7 encode as string with quotes empty element for parameters is okay
|
|
|
|
# windows py 3.x no encode and as string with quotes empty element for parameters is okay
|
|
|
|
# separate handling for windows and linux
|
|
|
|
quotes = [1, 2]
|
|
|
|
command = [config.config_converterpath, (file_path + format_old_ext),
|
|
|
|
(file_path + format_new_ext)]
|
|
|
|
quotes_index = 3
|
|
|
|
if config.config_calibre:
|
|
|
|
parameters = config.config_calibre.split(" ")
|
|
|
|
for param in parameters:
|
|
|
|
command.append(param)
|
|
|
|
quotes.append(quotes_index)
|
|
|
|
quotes_index += 1
|
|
|
|
|
|
|
|
p = process_open(command, quotes)
|
|
|
|
except OSError as e:
|
|
|
|
return 1, _(u"Ebook-converter failed: %(error)s", error=e)
|
|
|
|
|
|
|
|
while p.poll() is None:
|
|
|
|
nextline = p.stdout.readline()
|
|
|
|
if os.name == 'nt' and sys.version_info < (3, 0):
|
|
|
|
nextline = nextline.decode('windows-1252')
|
|
|
|
elif os.name == 'posix' and sys.version_info < (3, 0):
|
|
|
|
nextline = nextline.decode('utf-8')
|
|
|
|
log.debug(nextline.strip('\r\n'))
|
|
|
|
# parse progress string from calibre-converter
|
|
|
|
progress = re.search(r"(\d+)%\s.*", nextline)
|
|
|
|
if progress:
|
|
|
|
self.progress = int(progress.group(1)) / 100
|
2020-10-04 12:09:52 +02:00
|
|
|
if config.config_use_google_drive:
|
|
|
|
self.progress *= 0.9
|
2020-08-22 16:31:00 -04:00
|
|
|
|
|
|
|
# process returncode
|
|
|
|
check = p.returncode
|
|
|
|
calibre_traceback = p.stderr.readlines()
|
|
|
|
error_message = ""
|
|
|
|
for ele in calibre_traceback:
|
|
|
|
if sys.version_info < (3, 0):
|
|
|
|
ele = ele.decode('utf-8')
|
|
|
|
log.debug(ele.strip('\n'))
|
|
|
|
if not ele.startswith('Traceback') and not ele.startswith(' File'):
|
2020-08-30 13:43:08 +02:00
|
|
|
error_message = _("Calibre failed with error: %(error)s", error=ele.strip('\n'))
|
2020-08-22 16:31:00 -04:00
|
|
|
return check, error_message
|
|
|
|
|
|
|
|
@property
|
|
|
|
def name(self):
|
|
|
|
return "Convert"
|