From 23d66a0d68ece521b61e4633a3694dfcef765225 Mon Sep 17 00:00:00 2001 From: Ozzieisaacs Date: Sat, 3 Oct 2020 21:43:48 +0200 Subject: [PATCH 1/4] Fix --- cps/helper.py | 2 +- cps/tasks/convert.py | 12 ++++-------- cps/updater.py | 1 + 3 files changed, 6 insertions(+), 9 deletions(-) diff --git a/cps/helper.py b/cps/helper.py index 5178a522..b3fdba73 100644 --- a/cps/helper.py +++ b/cps/helper.py @@ -102,7 +102,7 @@ def convert_book_format(book_id, calibrepath, old_book_format, new_book_format, txt = (u"%s -> %s: %s" % (old_book_format, new_book_format, book.title)) settings['old_book_format'] = old_book_format settings['new_book_format'] = new_book_format - WorkerThread.add(user_id, TaskConvert(file_path, book.id, txt, settings, kindle_mail)) + WorkerThread.add(user_id, TaskConvert(file_path, book.id, txt, settings, kindle_mail, user_id)) return None else: error_message = _(u"%(format)s not found: %(fn)s", diff --git a/cps/tasks/convert.py b/cps/tasks/convert.py index b6093dbe..35aceddb 100644 --- a/cps/tasks/convert.py +++ b/cps/tasks/convert.py @@ -20,12 +20,13 @@ log = logger.create() class TaskConvert(CalibreTask): - def __init__(self, file_path, bookid, taskMessage, settings, kindle_mail): + def __init__(self, file_path, bookid, taskMessage, settings, kindle_mail, user=None): super(TaskConvert, self).__init__(taskMessage) self.file_path = file_path self.bookid = bookid self.settings = settings self.kindle_mail = kindle_mail + self.user = user self.results = dict() @@ -40,14 +41,9 @@ class TaskConvert(CalibreTask): # 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 try: - task = TaskEmail(self.settings['subject'], self.results["path"], + 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) - task.start(worker_thread) - - # even though the convert task might be finished, if this task fails, fail the whole thing - if task.stat != STAT_FINISH_SUCCESS: - raise Exception(task.error) + self.settings['subject'], self.settings['body'], internal=True)) except Exception as e: return self._handleError(str(e)) diff --git a/cps/updater.py b/cps/updater.py index e4883370..4a37f688 100644 --- a/cps/updater.py +++ b/cps/updater.py @@ -227,6 +227,7 @@ class Updater(threading.Thread): os.sep + 'vendor', os.sep + 'calibre-web.log', os.sep + '.git', os.sep + 'client_secrets.json', os.sep + 'gdrive_credentials', os.sep + 'settings.yaml', os.sep + 'venv', os.sep + 'virtualenv', os.sep + 'access.log', os.sep + 'access.log1', os.sep + 'access.log2', + os.sep + '.calibre-web.log.swp' ) additional_path = self.is_venv() if additional_path: From eb37e3a52b9cddb1af58712cf4d0d6a4ba26bcb8 Mon Sep 17 00:00:00 2001 From: Ozzieisaacs Date: Sun, 4 Oct 2020 12:09:52 +0200 Subject: [PATCH 2/4] Fix UI Deelete Buttons in Edit Books page Fix Conversion of multiple formats of one book in the queue with gdrive --- cps/helper.py | 43 +++++++++++++++--------------------- cps/tasks/convert.py | 25 ++++++++++++++++++++- cps/templates/book_edit.html | 4 +++- 3 files changed, 45 insertions(+), 27 deletions(-) diff --git a/cps/helper.py b/cps/helper.py index b3fdba73..b0ea7ff7 100644 --- a/cps/helper.py +++ b/cps/helper.py @@ -74,40 +74,33 @@ log = logger.create() def convert_book_format(book_id, calibrepath, old_book_format, new_book_format, user_id, kindle_mail=None): book = calibre_db.get_book(book_id) data = calibre_db.get_book_format(book.id, old_book_format) + file_path = os.path.join(calibrepath, book.path, data.name) if not data: error_message = _(u"%(format)s format not found for book id: %(book)d", format=old_book_format, book=book_id) log.error("convert_book_format: %s", error_message) return error_message if config.config_use_google_drive: - df = gd.getFileFromEbooksFolder(book.path, data.name + "." + old_book_format.lower()) - if df: - datafile = os.path.join(calibrepath, book.path, data.name + u"." + old_book_format.lower()) - if not os.path.exists(os.path.join(calibrepath, book.path)): - os.makedirs(os.path.join(calibrepath, book.path)) - df.GetContentFile(datafile) - else: + if not gd.getFileFromEbooksFolder(book.path, data.name + "." + old_book_format.lower()): error_message = _(u"%(format)s not found on Google Drive: %(fn)s", format=old_book_format, fn=data.name + "." + old_book_format.lower()) return error_message - file_path = os.path.join(calibrepath, book.path, data.name) - if os.path.exists(file_path + "." + old_book_format.lower()): - # read settings and append converter task to queue - if kindle_mail: - settings = config.get_mail_settings() - settings['subject'] = _('Send to Kindle') # pretranslate Subject for e-mail - settings['body'] = _(u'This e-mail has been sent via Calibre-Web.') - # text = _(u"%(format)s: %(book)s", format=new_book_format, book=book.title) - else: - settings = dict() - txt = (u"%s -> %s: %s" % (old_book_format, new_book_format, book.title)) - settings['old_book_format'] = old_book_format - settings['new_book_format'] = new_book_format - WorkerThread.add(user_id, TaskConvert(file_path, book.id, txt, settings, kindle_mail, user_id)) - return None else: - error_message = _(u"%(format)s not found: %(fn)s", - format=old_book_format, fn=data.name + "." + old_book_format.lower()) - return error_message + if not os.path.exists(file_path + "." + old_book_format.lower()): + error_message = _(u"%(format)s not found: %(fn)s", + format=old_book_format, fn=data.name + "." + old_book_format.lower()) + return error_message + # read settings and append converter task to queue + if kindle_mail: + settings = config.get_mail_settings() + settings['subject'] = _('Send to Kindle') # pretranslate Subject for e-mail + settings['body'] = _(u'This e-mail has been sent via Calibre-Web.') + else: + settings = dict() + txt = (u"%s -> %s: %s" % (old_book_format, new_book_format, book.title)) + settings['old_book_format'] = old_book_format + settings['new_book_format'] = new_book_format + WorkerThread.add(user_id, TaskConvert(file_path, book.id, txt, settings, kindle_mail, user_id)) + return None def send_test_mail(kindle_mail, user_name): diff --git a/cps/tasks/convert.py b/cps/tasks/convert.py index 35aceddb..a41c5ee1 100644 --- a/cps/tasks/convert.py +++ b/cps/tasks/convert.py @@ -32,11 +32,31 @@ class TaskConvert(CalibreTask): def run(self, worker_thread): self.worker_thread = worker_thread + if config.config_use_google_drive: + cur_book = calibre_db.get_book(self.bookid) + data = calibre_db.get_book_format(self.bookid, self.settings['old_book_format']) + 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) + 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()) + return error_message + filename = self._convert_ebook_format() if filename: if config.config_use_google_drive: + # Upload files to gdrive gdriveutils.updateGdriveCalibreFromLocal() + self._handleSuccess() 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 @@ -99,7 +119,8 @@ class TaskConvert(CalibreTask): self.results['title'] = cur_book.title if config.config_use_google_drive: os.remove(file_path + format_old_ext) - self._handleSuccess() + else: + self._handleSuccess() return os.path.basename(file_path + format_new_ext) else: error_message = _('%(format)s format not found on disk', format=format_new_ext.upper()) @@ -175,6 +196,8 @@ class TaskConvert(CalibreTask): progress = re.search(r"(\d+)%\s.*", nextline) if progress: self.progress = int(progress.group(1)) / 100 + if config.config_use_google_drive: + self.progress *= 0.9 # process returncode check = p.returncode diff --git a/cps/templates/book_edit.html b/cps/templates/book_edit.html index 01f6772d..16c02fae 100644 --- a/cps/templates/book_edit.html +++ b/cps/templates/book_edit.html @@ -12,7 +12,9 @@ {% if book.data|length > 1 %}

{{_('Delete formats:')}}

{% for file in book.data %} - +
+ +
{% endfor %}
{% endif %} From e3f4f24c3ef89ec8d0e2cb5313e80ccb3ad2898f Mon Sep 17 00:00:00 2001 From: Ozzieisaacs Date: Sun, 4 Oct 2020 13:59:33 +0200 Subject: [PATCH 3/4] Fixes for failed email and conversions with gdrive --- cps/services/worker.py | 2 +- cps/tasks/convert.py | 6 +++--- cps/tasks/mail.py | 7 ++++--- 3 files changed, 8 insertions(+), 7 deletions(-) diff --git a/cps/services/worker.py b/cps/services/worker.py index 369522d6..c2ea594c 100644 --- a/cps/services/worker.py +++ b/cps/services/worker.py @@ -210,7 +210,7 @@ class CalibreTask: self._progress = x def _handleError(self, error_message): - log.error(error_message) + log.exception(error_message) self.stat = STAT_FAIL self.progress = 1 self.error = error_message diff --git a/cps/tasks/convert.py b/cps/tasks/convert.py index a41c5ee1..e3857389 100644 --- a/cps/tasks/convert.py +++ b/cps/tasks/convert.py @@ -51,6 +51,8 @@ class TaskConvert(CalibreTask): return error_message filename = self._convert_ebook_format() + if config.config_use_google_drive: + os.remove(self.file_path + u'.' + self.settings['old_book_format'].lower()) if filename: if config.config_use_google_drive: @@ -117,9 +119,7 @@ class TaskConvert(CalibreTask): return self.results['path'] = cur_book.path self.results['title'] = cur_book.title - if config.config_use_google_drive: - os.remove(file_path + format_old_ext) - else: + if not config.config_use_google_drive: self._handleSuccess() return os.path.basename(file_path + format_new_ext) else: diff --git a/cps/tasks/mail.py b/cps/tasks/mail.py index f83cc4fa..ac3ec424 100644 --- a/cps/tasks/mail.py +++ b/cps/tasks/mail.py @@ -169,7 +169,7 @@ class TaskEmail(CalibreTask): except (MemoryError) as e: log.exception(e) self._handleError(u'MemoryError sending email: ' + str(e)) - return None + # return None except (smtplib.SMTPException, smtplib.SMTPAuthenticationError) as e: if hasattr(e, "smtp_error"): text = e.smtp_error.decode('utf-8').replace("\n", '. ') @@ -181,10 +181,11 @@ class TaskEmail(CalibreTask): log.exception(e) text = '' self._handleError(u'Smtplib Error sending email: ' + text) - return None + # return None except (socket.error) as e: self._handleError(u'Socket Error sending email: ' + e.strerror) - return None + # return None + @property def progress(self): From 6e6f144b7a5c88a37305754802d3baa438874f61 Mon Sep 17 00:00:00 2001 From: Ozzieisaacs Date: Sun, 4 Oct 2020 19:23:06 +0200 Subject: [PATCH 4/4] Paginated search and advanced search Wraparound on books list deactivated --- cps/db.py | 7 ++-- cps/jinjia.py | 2 ++ cps/static/js/table.js | 1 + cps/templates/book_table.html | 16 ++++----- cps/web.py | 68 +++++++++++++++++++++++------------ 5 files changed, 61 insertions(+), 33 deletions(-) diff --git a/cps/db.py b/cps/db.py index 32dd022b..28b21c45 100644 --- a/cps/db.py +++ b/cps/db.py @@ -648,9 +648,10 @@ class CalibreDB(): # read search results from calibre-database and return it (function is used for feed and simple search def get_search_results(self, term, offset=None, order=None, limit=None): order = order or [Books.sort] + pagination = None if offset != None and limit != None: offset = int(offset) - limit = offset + int(limit) + limit_all = offset + int(limit) term.strip().lower() self.session.connection().connection.connection.create_function("lower", 1, lcase) q = list() @@ -665,7 +666,9 @@ class CalibreDB(): func.lower(Books.title).ilike("%" + term + "%") )).order_by(*order).all() result_count = len(result) - return result[offset:limit], result_count + if offset != None and limit != None: + pagination = Pagination((offset / (int(limit)) + 1), limit, result_count) + return result[offset:limit_all], result_count, pagination # Creates for all stored languages a translated speaking name in the array for the UI def speaking_language(self, languages=None): diff --git a/cps/jinjia.py b/cps/jinjia.py index a6df156d..87ba4159 100644 --- a/cps/jinjia.py +++ b/cps/jinjia.py @@ -44,6 +44,8 @@ log = logger.create() def url_for_other_page(page): args = request.view_args.copy() args['page'] = page + for get, val in request.args.items(): + args[get] = val return url_for(request.endpoint, **args) diff --git a/cps/static/js/table.js b/cps/static/js/table.js index cbfe58e7..62f7e220 100644 --- a/cps/static/js/table.js +++ b/cps/static/js/table.js @@ -109,6 +109,7 @@ $(function() { $("#books-table").bootstrapTable({ sidePagination: "server", pagination: true, + paginationLoop: false, paginationDetailHAlign: " hidden", paginationHAlign: "left", idField: "id", diff --git a/cps/templates/book_table.html b/cps/templates/book_table.html index f93a3af6..067ff5e0 100644 --- a/cps/templates/book_table.html +++ b/cps/templates/book_table.html @@ -2,13 +2,13 @@ {% macro text_table_row(parameter, edit_text, show_text, validate) -%} {{ show_text }} {%- endmacro %} @@ -39,9 +39,9 @@ data-url="{{url_for('web.list_books')}}"> - {% if g.user.role_edit() %} + {% if g.user.role_edit() %} - {% endif %} + {% endif %} {{ text_table_row('title', _('Enter Title'),_('Title'), true) }} {{ text_table_row('sort', _('Enter Title Sort'),_('Title Sort'), false) }} @@ -49,13 +49,13 @@ {{ text_table_row('authors', _('Enter Authors'),_('Authors'), true) }} {{ text_table_row('tags', _('Enter Categories'),_('Categories'), false) }} {{ text_table_row('series', _('Enter Series'),_('Series'), false) }} - {{_('Series Index')}} + {{_('Series Index')}} {{ text_table_row('languages', _('Enter Languages'),_('Languages'), false) }} {{ text_table_row('publishers', _('Enter Publishers'),_('Publishers'), false) }} - {% if g.user.role_edit() %} + {% if g.user.role_edit() %} {{_('Delete')}} - {% endif %} + {% endif %} diff --git a/cps/web.py b/cps/web.py index 825772cd..e20759ec 100644 --- a/cps/web.py +++ b/cps/web.py @@ -35,6 +35,7 @@ from babel import Locale as LC from babel.core import UnknownLocaleError from flask import Blueprint, jsonify from flask import render_template, request, redirect, send_from_directory, make_response, g, flash, abort, url_for +from flask import session as flask_session from flask_babel import gettext as _ from flask_login import login_user, logout_user, login_required, current_user, confirm_login from sqlalchemy.exc import IntegrityError, InvalidRequestError, OperationalError @@ -669,10 +670,11 @@ def render_books_list(data, sort, book_id, page): elif data == "search": term = (request.args.get('query') or '') offset = int(int(config.config_books_per_page) * (page - 1)) - if '&' not in term: - return render_search_results(term, offset, order, config.config_books_per_page) - else: - return render_adv_search_results(term, offset, order, config.config_books_per_page) + return render_search_results(term, offset, order, config.config_books_per_page) + elif data == "advsearch": + term = json.loads(flask_session['query']) + offset = int(int(config.config_books_per_page) * (page - 1)) + return render_adv_search_results(term, offset, order, config.config_books_per_page) else: website = data or "newest" entries, random, pagination = calibre_db.fill_indexpage(page, 0, db.Books, True, order) @@ -953,13 +955,14 @@ def render_prepare_search_form(cc): def render_search_results(term, offset=None, order=None, limit=None): - entries, result_count = calibre_db.get_search_results(term, offset, order, limit) + entries, result_count, pagination = calibre_db.get_search_results(term, offset, order, limit) ids = list() for element in entries: ids.append(element.id) searched_ids[current_user.id] = ids return render_title_template('search.html', searchterm=term, + pagination=pagination, query=term, adv_searchterm=term, entries=entries, @@ -967,6 +970,7 @@ def render_search_results(term, offset=None, order=None, limit=None): title=_(u"Search"), page="search") + # ################################### View Books list ################################################################## @@ -1007,7 +1011,7 @@ def list_books(): search = request.args.get("search") total_count = calibre_db.session.query(db.Books).count() if search: - entries, filtered_count = calibre_db.get_search_results(search, off, order, limit) + entries, filtered_count, pagination = calibre_db.get_search_results(search, off, order, limit) else: entries, __, __ = calibre_db.fill_indexpage((int(off) / (int(limit)) + 1), limit, db.Books, True, order) filtered_count = total_count @@ -1227,7 +1231,7 @@ def reconnect(): def search(): term = request.args.get("query") if term: - return render_search_results(term) + return render_search_results(term, 0, None, config.config_books_per_page) else: return render_title_template('search.html', searchterm="", @@ -1238,9 +1242,19 @@ def search(): @web.route("/advanced_search", methods=['POST']) @login_required_if_no_ano def advanced_search(): + term = request.form + return render_adv_search_results(term, 0, None, config.config_books_per_page) + +def render_adv_search_results(term, offset=None, order=None, limit=None): + order = order or [db.Books.sort] + pagination = None + if offset != None and limit != None: + offset = int(offset) + limit_all = offset + int(limit) + cc = get_cc_columns(filter_config_custom_read=True) calibre_db.session.connection().connection.connection.create_function("lower", 1, db.lcase) - q = calibre_db.session.query(db.Books).filter(calibre_db.common_filters(True)).order_by(db.Books.sort) + q = calibre_db.session.query(db.Books).filter(calibre_db.common_filters(True)) include_tag_inputs = request.form.getlist('include_tag') exclude_tag_inputs = request.form.getlist('exclude_tag') @@ -1251,14 +1265,14 @@ def advanced_search(): include_extension_inputs = request.form.getlist('include_extension') exclude_extension_inputs = request.form.getlist('exclude_extension') - author_name = request.form.get("author_name") - book_title = request.form.get("book_title") - publisher = request.form.get("publisher") - pub_start = request.form.get("Publishstart") - pub_end = request.form.get("Publishend") - rating_low = request.form.get("ratinghigh") - rating_high = request.form.get("ratinglow") - description = request.form.get("comment") + author_name = term.get("author_name") + book_title = term.get("book_title") + publisher = term.get("publisher") + pub_start = term.get("Publishstart") + pub_end = term.get("Publishend") + rating_low = term.get("ratinghigh") + rating_high = term.get("ratinglow") + description = term.get("comment") if author_name: author_name = author_name.strip().lower().replace(',', '|') if book_title: @@ -1367,17 +1381,25 @@ def advanced_search(): else: q = q.filter(getattr(db.Books, 'custom_column_' + str(c.id)).any( func.lower(db.cc_classes[c.id].value).ilike("%" + custom_query + "%"))) - q = q.all() + q = q.order_by(*order).all() + flask_session['query'] = json.dumps(term) + # ToDo: Check saved ids mechanism ? ids = list() for element in q: ids.append(element.id) searched_ids[current_user.id] = ids - return render_title_template('search.html', - adv_searchterm=searchterm, - query=request.form, - entries=q, - result_count=len(q), - title=_(u"search"), page="search") + # entries, result_count, pagination = calibre_db.get_search_results(term, offset, order, limit) + result_count = len(q) + if offset != None and limit != None: + pagination = Pagination((offset / (int(limit)) + 1), limit, result_count) + return render_title_template('search.html', + adv_searchterm=searchterm, + pagination=pagination, + # query=request.form, + entries=q[offset:limit_all], + result_count=result_count, + title=_(u"search"), page="advsearch") + @web.route("/advanced_search", methods=['GET'])