From a784c6bd5292a261277ed69ca36f8d7a0105d8cb Mon Sep 17 00:00:00 2001 From: Ozzieisaacs Date: Wed, 15 Apr 2020 19:57:33 +0200 Subject: [PATCH] Fixes for oauth login after programming basic tests --- cps/about.py | 14 +- cps/admin.py | 21 +- cps/oauth.py | 4 +- cps/oauth_bb.py | 114 ++++---- cps/templates/login.html | 26 +- cps/templates/stats.html | 2 + cps/templates/user_edit.html | 2 +- cps/web.py | 199 +++++++------ optional-requirements.txt | 2 +- test/Calibre-Web TestSummary.html | 451 ++++++++++++++++-------------- 10 files changed, 450 insertions(+), 385 deletions(-) diff --git a/cps/about.py b/cps/about.py index dfc6c502..2f1d4b43 100644 --- a/cps/about.py +++ b/cps/about.py @@ -43,6 +43,11 @@ try: except ImportError: unidecode_version = _(u'not installed') +try: + from flask_dance import __version__ as flask_danceVersion +except ImportError: + flask_danceVersion = None + from . import services about = flask.Blueprint('about', __name__) @@ -68,10 +73,11 @@ _VERSIONS = OrderedDict( iso639=isoLanguages.__version__, pytz=pytz.__version__, Unidecode = unidecode_version, - Flask_SimpleLDAP = u'installed' if bool(services.ldap) else u'not installed', - python_LDAP = services.ldapVersion if bool(services.ldapVersion) else u'not installed', - Goodreads = u'installed' if bool(services.goodreads_support) else u'not installed', - jsonschema = services.SyncToken.__version__ if bool(services.SyncToken) else u'not installed', + Flask_SimpleLDAP = u'installed' if bool(services.ldap) else None, + python_LDAP = services.ldapVersion if bool(services.ldapVersion) else None, + Goodreads = u'installed' if bool(services.goodreads_support) else None, + jsonschema = services.SyncToken.__version__ if bool(services.SyncToken) else None, + flask_dance = flask_danceVersion ) _VERSIONS.update(uploader.get_versions()) diff --git a/cps/admin.py b/cps/admin.py index 86f94a02..69316e8f 100644 --- a/cps/admin.py +++ b/cps/admin.py @@ -622,23 +622,22 @@ def _configuration_update_helper(): active_oauths = 0 for element in oauthblueprints: + if to_save["config_" + str(element['id']) + "_oauth_client_id"] != element['oauth_client_id'] \ + or to_save["config_" + str(element['id']) + "_oauth_client_secret"] != element['oauth_client_secret']: + reboot_required = True + element['oauth_client_id'] = to_save["config_" + str(element['id']) + "_oauth_client_id"] + element['oauth_client_secret'] = to_save["config_" + str(element['id']) + "_oauth_client_secret"] if to_save["config_"+str(element['id'])+"_oauth_client_id"] \ and to_save["config_"+str(element['id'])+"_oauth_client_secret"]: active_oauths += 1 element["active"] = 1 - ub.session.query(ub.OAuthProvider).filter(ub.OAuthProvider.id == element['id']).update( - {"oauth_client_id":to_save["config_"+str(element['id'])+"_oauth_client_id"], - "oauth_client_secret":to_save["config_"+str(element['id'])+"_oauth_client_secret"], - "active":1}) - if to_save["config_" + str(element['id']) + "_oauth_client_id"] != element['oauth_client_id'] \ - or to_save["config_" + str(element['id']) + "_oauth_client_secret"] != element['oauth_client_secret']: - reboot_required = True - element['oauth_client_id'] = to_save["config_"+str(element['id'])+"_oauth_client_id"] - element['oauth_client_secret'] = to_save["config_"+str(element['id'])+"_oauth_client_secret"] else: - ub.session.query(ub.OAuthProvider).filter(ub.OAuthProvider.id == element['id']).update( - {"active":0}) element["active"] = 0 + ub.session.query(ub.OAuthProvider).filter(ub.OAuthProvider.id == element['id']).update( + {"oauth_client_id":to_save["config_"+str(element['id'])+"_oauth_client_id"], + "oauth_client_secret":to_save["config_"+str(element['id'])+"_oauth_client_secret"], + "active":element["active"]}) + reboot_required |= _config_int("config_log_level") reboot_required |= _config_string("config_logfile") diff --git a/cps/oauth.py b/cps/oauth.py index c2f98bbb..7846a47b 100644 --- a/cps/oauth.py +++ b/cps/oauth.py @@ -70,7 +70,7 @@ try: use_provider_user_id = True if self.user_required and not u and not uid and not use_provider_user_id: - #raise ValueError("Cannot get OAuth token without an associated user") + # raise ValueError("Cannot get OAuth token without an associated user") return None # check for user ID if hasattr(self.model, "user_id") and uid: @@ -95,7 +95,7 @@ try: def set(self, blueprint, token, user=None, user_id=None): uid = first([user_id, self.user_id, blueprint.config.get("user_id")]) u = first(_get_real_user(ref, self.anon_user) - for ref in (user, self.user, blueprint.config.get("user"))) + for ref in (user, self.user, blueprint.config.get("user"))) if self.user_required and not u and not uid: raise ValueError("Cannot set OAuth token without an associated user") diff --git a/cps/oauth_bb.py b/cps/oauth_bb.py index 70dc9aa6..2dbcc875 100644 --- a/cps/oauth_bb.py +++ b/cps/oauth_bb.py @@ -36,7 +36,6 @@ from sqlalchemy.orm.exc import NoResultFound from . import constants, logger, config, app, ub from .web import login_required from .oauth import OAuthBackend -# from .web import github_oauth_required oauth_check = {} @@ -59,29 +58,29 @@ def oauth_required(f): return inner -def register_oauth_blueprint(id, show_name): - oauth_check[id] = show_name +def register_oauth_blueprint(cid, show_name): + oauth_check[cid] = show_name def register_user_with_oauth(user=None): all_oauth = {} - for oauth in oauth_check.keys(): - if str(oauth) + '_oauth_user_id' in session and session[str(oauth) + '_oauth_user_id'] != '': - all_oauth[oauth] = oauth_check[oauth] + for oauth_key in oauth_check.keys(): + if str(oauth_key) + '_oauth_user_id' in session and session[str(oauth_key) + '_oauth_user_id'] != '': + all_oauth[oauth_key] = oauth_check[oauth_key] if len(all_oauth.keys()) == 0: return if user is None: flash(_(u"Register with %(provider)s", provider=", ".join(list(all_oauth.values()))), category="success") else: - for oauth in all_oauth.keys(): + for oauth_key in all_oauth.keys(): # Find this OAuth token in the database, or create it query = ub.session.query(ub.OAuth).filter_by( - provider=oauth, - provider_user_id=session[str(oauth) + "_oauth_user_id"], + provider=oauth_key, + provider_user_id=session[str(oauth_key) + "_oauth_user_id"], ) try: - oauth = query.one() - oauth.user_id = user.id + oauth_key = query.one() + oauth_key.user_id = user.id except NoResultFound: # no found, return error return @@ -93,39 +92,40 @@ def register_user_with_oauth(user=None): def logout_oauth_user(): - for oauth in oauth_check.keys(): - if str(oauth) + '_oauth_user_id' in session: - session.pop(str(oauth) + '_oauth_user_id') + for oauth_key in oauth_check.keys(): + if str(oauth_key) + '_oauth_user_id' in session: + session.pop(str(oauth_key) + '_oauth_user_id') + if ub.oauth_support: - oauthblueprints =[] + oauthblueprints = [] if not ub.session.query(ub.OAuthProvider).count(): - oauth = ub.OAuthProvider() - oauth.provider_name = "github" - oauth.active = False - ub.session.add(oauth) + oauthProvider = ub.OAuthProvider() + oauthProvider.provider_name = "github" + oauthProvider.active = False + ub.session.add(oauthProvider) ub.session.commit() - oauth = ub.OAuthProvider() - oauth.provider_name = "google" - oauth.active = False - ub.session.add(oauth) + oauthProvider = ub.OAuthProvider() + oauthProvider.provider_name = "google" + oauthProvider.active = False + ub.session.add(oauthProvider) ub.session.commit() oauth_ids = ub.session.query(ub.OAuthProvider).all() - ele1=dict(provider_name='github', - id=oauth_ids[0].id, - active=oauth_ids[0].active, - oauth_client_id=oauth_ids[0].oauth_client_id, - scope=None, - oauth_client_secret=oauth_ids[0].oauth_client_secret, - obtain_link='https://github.com/settings/developers') - ele2=dict(provider_name='google', - id=oauth_ids[1].id, - active=oauth_ids[1].active, - scope=["https://www.googleapis.com/auth/plus.me", "https://www.googleapis.com/auth/userinfo.email"], - oauth_client_id=oauth_ids[1].oauth_client_id, - oauth_client_secret=oauth_ids[1].oauth_client_secret, - obtain_link='https://github.com/settings/developers') + ele1 = dict(provider_name='github', + id=oauth_ids[0].id, + active=oauth_ids[0].active, + oauth_client_id=oauth_ids[0].oauth_client_id, + scope=None, + oauth_client_secret=oauth_ids[0].oauth_client_secret, + obtain_link='https://github.com/settings/developers') + ele2 = dict(provider_name='google', + id=oauth_ids[1].id, + active=oauth_ids[1].active, + scope=["https://www.googleapis.com/auth/plus.me", "https://www.googleapis.com/auth/userinfo.email"], + oauth_client_id=oauth_ids[1].oauth_client_id, + oauth_client_secret=oauth_ids[1].oauth_client_secret, + obtain_link='https://github.com/settings/developers') oauthblueprints.append(ele1) oauthblueprints.append(ele2) @@ -138,9 +138,9 @@ if ub.oauth_support: client_id=element['oauth_client_id'], client_secret=element['oauth_client_secret'], redirect_to="oauth."+element['provider_name']+"_login", - scope = element['scope'] + scope=element['scope'] ) - element['blueprint']=blueprint + element['blueprint'] = blueprint app.register_blueprint(blueprint, url_prefix="/login") element['blueprint'].backend = OAuthBackend(ub.OAuth, ub.session, str(element['id']), user=current_user, user_required=True) @@ -190,17 +190,17 @@ if ub.oauth_support: provider_user_id=provider_user_id, ) try: - oauth = query.one() + oauth_entry = query.one() # update token - oauth.token = token + oauth_entry.token = token except NoResultFound: - oauth = ub.OAuth( + oauth_entry = ub.OAuth( provider=provider_id, provider_user_id=provider_user_id, token=token, ) try: - ub.session.add(oauth) + ub.session.add(oauth_entry) ub.session.commit() except Exception as e: log.exception(e) @@ -216,25 +216,25 @@ if ub.oauth_support: provider_user_id=provider_user_id, ) try: - oauth = query.one() + oauth_entry = query.one() # already bind with user, just login - if oauth.user: - login_user(oauth.user) + if oauth_entry.user: + login_user(oauth_entry.user) return redirect(url_for('web.index')) else: # bind to current user if current_user and current_user.is_authenticated: - oauth.user = current_user + oauth_entry.user = current_user try: - ub.session.add(oauth) + ub.session.add(oauth_entry) ub.session.commit() except Exception as e: log.exception(e) ub.session.rollback() return redirect(url_for('web.login')) - #if config.config_public_reg: + # if config.config_public_reg: # return redirect(url_for('web.register')) - #else: + # else: # flash(_(u"Public registration is not enabled"), category="error") # return redirect(url_for(redirect_url)) except NoResultFound: @@ -248,8 +248,8 @@ if ub.oauth_support: ) try: oauths = query.all() - for oauth in oauths: - status.append(int(oauth.provider)) + for oauth_entry in oauths: + status.append(int(oauth_entry.provider)) return status except NoResultFound: return None @@ -263,11 +263,11 @@ if ub.oauth_support: user_id=current_user.id, ) try: - oauth = query.one() + oauth_entry = query.one() if current_user and current_user.is_authenticated: - oauth.user = current_user + oauth_entry.user = current_user try: - ub.session.delete(oauth) + ub.session.delete(oauth_entry) ub.session.commit() logout_oauth_user() flash(_(u"Unlink to %(oauth)s success.", oauth=oauth_check[provider]), category="success") @@ -292,7 +292,7 @@ if ub.oauth_support: error=error, description=error_description, uri=error_uri, - ) # ToDo: Translate + ) # ToDo: Translate flash(msg, category="error") @@ -338,7 +338,7 @@ if ub.oauth_support: error=error, description=error_description, uri=error_uri, - ) # ToDo: Translate + ) # ToDo: Translate flash(msg, category="error") diff --git a/cps/templates/login.html b/cps/templates/login.html index 23d59f2e..a031d949 100644 --- a/cps/templates/login.html +++ b/cps/templates/login.html @@ -25,19 +25,21 @@ {{_('Log in with Magic Link')}} {% endif %} {% if config.config_login_type == 2 %} - - + + {% endif %} + {% if 2 in oauth_check %} + + + + {% endif %} {% endif %} diff --git a/cps/templates/stats.html b/cps/templates/stats.html index 0457e4b9..69712cc4 100644 --- a/cps/templates/stats.html +++ b/cps/templates/stats.html @@ -35,10 +35,12 @@ {% for library,version in versions.items() %} + {% if version %} {{library}} {{_(version)}} + {% endif %} {% endfor %} diff --git a/cps/templates/user_edit.html b/cps/templates/user_edit.html index dad0b773..d3b80a55 100644 --- a/cps/templates/user_edit.html +++ b/cps/templates/user_edit.html @@ -161,7 +161,7 @@ diff --git a/cps/web.py b/cps/web.py index 9a7966e7..e67ae9cc 100644 --- a/cps/web.py +++ b/cps/web.py @@ -47,20 +47,21 @@ from . import constants, logger, isoLanguages, services, worker from . import searched_ids, lm, babel, db, ub, config, get_locale, app from .gdriveutils import getFileFromEbooksFolder, do_gdrive_download from .helper import common_filters, get_search_results, fill_indexpage, speaking_language, check_valid_domain, \ - order_authors, get_typeahead, render_task_status, json_serial, get_cc_columns, \ - get_book_cover, get_download_link, send_mail, generate_random_password, send_registration_mail, \ - check_send_to_kindle, check_read_formats, lcase, tags_filters, reset_password + order_authors, get_typeahead, render_task_status, json_serial, get_cc_columns, \ + get_book_cover, get_download_link, send_mail, generate_random_password, send_registration_mail, \ + check_send_to_kindle, check_read_formats, lcase, tags_filters, reset_password from .pagination import Pagination from .redirect import redirect_back feature_support = { - 'ldap': bool(services.ldap), - 'goodreads': bool(services.goodreads_support), - 'kobo': bool(services.kobo) - } + 'ldap': bool(services.ldap), + 'goodreads': bool(services.goodreads_support), + 'kobo': bool(services.kobo) +} try: from .oauth_bb import oauth_check, register_user_with_oauth, logout_oauth_user, get_oauth_status + feature_support['oauth'] = True except ImportError: feature_support['oauth'] = False @@ -80,27 +81,27 @@ except ImportError: try: from natsort import natsorted as sort except ImportError: - sort = sorted # Just use regular sort then, may cause issues with badly named pages in cbz/cbr files + sort = sorted # Just use regular sort then, may cause issues with badly named pages in cbz/cbr files # custom error page def error_http(error): return render_template('http_error.html', - error_code="Error {0}".format(error.code), - error_name=error.name, - issue=False, - instance=config.config_calibre_web_title - ), error.code + error_code="Error {0}".format(error.code), + error_name=error.name, + issue=False, + instance=config.config_calibre_web_title + ), error.code def internal_error(error): return render_template('http_error.html', - error_code="Internal Server Error", - error_name=str(error), - issue=True, - error_stack=traceback.format_exc().split("\n"), - instance=config.config_calibre_web_title - ), 500 + error_code="Internal Server Error", + error_name=str(error), + issue=True, + error_stack=traceback.format_exc().split("\n"), + instance=config.config_calibre_web_title + ), 500 # http error handling @@ -108,16 +109,17 @@ for ex in default_exceptions: if ex < 500: app.register_error_handler(ex, error_http) elif ex == 500: - app.register_error_handler(ex, internal_error) - + app.register_error_handler(ex, internal_error) web = Blueprint('web', __name__) log = logger.create() + # ################################### Login logic and rights management ############################################### def _fetch_user_by_name(username): return ub.session.query(ub.User).filter(func.lower(ub.User.nickname) == username.lower()).first() + @lm.user_loader def load_user(user_id): return ub.session.query(ub.User).filter(ub.User.id == int(user_id)).first() @@ -165,6 +167,7 @@ def login_required_if_no_ano(func): if config.config_anonbrowse == 1: return func(*args, **kwargs) return login_required(func)(*args, **kwargs) + return decorated_view @@ -256,8 +259,9 @@ def edit_required(f): # Returns the template for rendering and includes the instance name def render_title_template(*args, **kwargs): - sidebar=ub.get_sidebar_config(kwargs) - return render_template(instance=config.config_calibre_web_title, sidebar=sidebar, accept=constants.EXTENSIONS_UPLOAD, + sidebar = ub.get_sidebar_config(kwargs) + return render_template(instance=config.config_calibre_web_title, sidebar=sidebar, + accept=constants.EXTENSIONS_UPLOAD, *args, **kwargs) @@ -269,8 +273,10 @@ def before_request(): g.allow_upload = config.config_uploading g.current_theme = config.config_theme g.config_authors_max = config.config_authors_max - g.shelves_access = ub.session.query(ub.Shelf).filter(or_(ub.Shelf.is_public == 1, ub.Shelf.user_id == current_user.id)).order_by(ub.Shelf.name).all() - if not config.db_configured and request.endpoint not in ('admin.basic_configuration', 'login') and '/static/' not in request.path: + g.shelves_access = ub.session.query(ub.Shelf).filter( + or_(ub.Shelf.is_public == 1, ub.Shelf.user_id == current_user.id)).order_by(ub.Shelf.name).all() + if not config.db_configured and request.endpoint not in ( + 'admin.basic_configuration', 'login') and '/static/' not in request.path: return redirect(url_for('admin.basic_configuration')) @@ -313,7 +319,7 @@ def import_ldap_users(): if user_data: content = ub.User() content.nickname = user - content.password = '' # dummy password which will be replaced by ldap one + content.password = '' # dummy password which will be replaced by ldap one if 'mail' in user_data: content.email = user_data['mail'][0].decode('utf-8') if (len(user_data['mail']) > 1): @@ -477,11 +483,10 @@ def get_publishers_json(): return get_typeahead(db.Publishers, request.args.get('q'), ('|', ',')) - @web.route("/get_tags_json", methods=['GET']) @login_required_if_no_ano def get_tags_json(): - return get_typeahead(db.Tags, request.args.get('q'),tag_filter=tags_filters()) + return get_typeahead(db.Tags, request.args.get('q'), tag_filter=tags_filters()) @web.route("/get_series_json", methods=['GET']) @@ -497,9 +502,8 @@ def get_languages_json(): language_names = isoLanguages.get_language_names(get_locale()) entries_start = [s for key, s in language_names.items() if s.lower().startswith(query.lower())] if len(entries_start) < 5: - entries = [s for key, s in language_names.items() if query in s.lower()] - entries_start.extend(entries[0:(5-len(entries_start))]) + entries_start.extend(entries[0:(5 - len(entries_start))]) entries_start = list(set(entries_start)) json_dumps = json.dumps([dict(name=r) for r in entries_start[0:5]]) return json_dumps @@ -604,7 +608,7 @@ def books_list(data, sort, book_id, page): else: entries, random, pagination = fill_indexpage(page, db.Books, True, order) return render_title_template('index.html', random=random, entries=entries, pagination=pagination, - title=_(u"Books"), page="newest") + title=_(u"Books"), page="newest") def render_hot_books(page): @@ -636,13 +640,13 @@ def render_hot_books(page): abort(404) - def render_author_books(page, author_id, order): entries, __, pagination = fill_indexpage(page, db.Books, db.Books.authors.any(db.Authors.id == author_id), [order[0], db.Series.name, db.Books.series_index], db.books_series_link, db.Series) if entries is None or not len(entries): - flash(_(u"Oops! Selected book title is unavailable. File does not exist or is not accessible"), category="error") + flash(_(u"Oops! Selected book title is unavailable. File does not exist or is not accessible"), + category="error") return redirect(url_for("web.index")) author = db.session.query(db.Authors).get(author_id) @@ -686,10 +690,10 @@ def render_series_books(page, book_id, order): def render_ratings_books(page, book_id, order): name = db.session.query(db.Ratings).filter(db.Ratings.id == book_id).first() entries, random, pagination = fill_indexpage(page, db.Books, db.Books.ratings.any(db.Ratings.id == book_id), - [db.Books.timestamp.desc(), order[0]]) + [db.Books.timestamp.desc(), order[0]]) if name and name.rating <= 10: return render_title_template('index.html', random=random, pagination=pagination, entries=entries, id=book_id, - title=_(u"Rating: %(rating)s stars", rating=int(name.rating/2)), page="ratings") + title=_(u"Rating: %(rating)s stars", rating=int(name.rating / 2)), page="ratings") else: abort(404) @@ -697,8 +701,9 @@ def render_ratings_books(page, book_id, order): def render_formats_books(page, book_id, order): name = db.session.query(db.Data).filter(db.Data.format == book_id.upper()).first() if name: - entries, random, pagination = fill_indexpage(page, db.Books, db.Books.data.any(db.Data.format == book_id.upper()), - [db.Books.timestamp.desc(), order[0]]) + entries, random, pagination = fill_indexpage(page, db.Books, + db.Books.data.any(db.Data.format == book_id.upper()), + [db.Books.timestamp.desc(), order[0]]) return render_title_template('index.html', random=random, pagination=pagination, entries=entries, id=book_id, title=_(u"File format: %(format)s", format=name.format), page="formats") else: @@ -712,7 +717,7 @@ def render_category_books(page, book_id, order): [db.Series.name, db.Books.series_index, order[0]], db.books_series_link, db.Series) return render_title_template('index.html', random=random, entries=entries, pagination=pagination, id=book_id, - title=_(u"Category: %(name)s", name=name.name), page="category") + title=_(u"Category: %(name)s", name=name.name), page="category") else: abort(404) @@ -736,12 +741,12 @@ def render_language_books(page, name, order): @login_required_if_no_ano def author_list(): if current_user.check_visibility(constants.SIDEBAR_AUTHOR): - entries = db.session.query(db.Authors, func.count('books_authors_link.book').label('count'))\ - .join(db.books_authors_link).join(db.Books).filter(common_filters())\ - .group_by(text('books_authors_link.author')).order_by(db.Authors.sort).all() - charlist = db.session.query(func.upper(func.substr(db.Authors.sort,1,1)).label('char')) \ + entries = db.session.query(db.Authors, func.count('books_authors_link.book').label('count')) \ .join(db.books_authors_link).join(db.Books).filter(common_filters()) \ - .group_by(func.upper(func.substr(db.Authors.sort,1,1))).all() + .group_by(text('books_authors_link.author')).order_by(db.Authors.sort).all() + charlist = db.session.query(func.upper(func.substr(db.Authors.sort, 1, 1)).label('char')) \ + .join(db.books_authors_link).join(db.Books).filter(common_filters()) \ + .group_by(func.upper(func.substr(db.Authors.sort, 1, 1))).all() for entry in entries: entry.Authors.name = entry.Authors.name.replace('|', ',') return render_title_template('list.html', entries=entries, folder='web.books_list', charlist=charlist, @@ -754,12 +759,12 @@ def author_list(): @login_required_if_no_ano def publisher_list(): if current_user.check_visibility(constants.SIDEBAR_PUBLISHER): - entries = db.session.query(db.Publishers, func.count('books_publishers_link.book').label('count'))\ - .join(db.books_publishers_link).join(db.Books).filter(common_filters())\ - .group_by(text('books_publishers_link.publisher')).order_by(db.Publishers.sort).all() - charlist = db.session.query(func.upper(func.substr(db.Publishers.name,1,1)).label('char')) \ + entries = db.session.query(db.Publishers, func.count('books_publishers_link.book').label('count')) \ .join(db.books_publishers_link).join(db.Books).filter(common_filters()) \ - .group_by(func.upper(func.substr(db.Publishers.name,1,1))).all() + .group_by(text('books_publishers_link.publisher')).order_by(db.Publishers.sort).all() + charlist = db.session.query(func.upper(func.substr(db.Publishers.name, 1, 1)).label('char')) \ + .join(db.books_publishers_link).join(db.Books).filter(common_filters()) \ + .group_by(func.upper(func.substr(db.Publishers.name, 1, 1))).all() return render_title_template('list.html', entries=entries, folder='web.books_list', charlist=charlist, title=_(u"Publishers"), page="publisherlist", data="publisher") else: @@ -770,12 +775,12 @@ def publisher_list(): @login_required_if_no_ano def series_list(): if current_user.check_visibility(constants.SIDEBAR_SERIES): - entries = db.session.query(db.Series, func.count('books_series_link.book').label('count'))\ - .join(db.books_series_link).join(db.Books).filter(common_filters())\ - .group_by(text('books_series_link.series')).order_by(db.Series.sort).all() - charlist = db.session.query(func.upper(func.substr(db.Series.sort,1,1)).label('char')) \ + entries = db.session.query(db.Series, func.count('books_series_link.book').label('count')) \ .join(db.books_series_link).join(db.Books).filter(common_filters()) \ - .group_by(func.upper(func.substr(db.Series.sort,1,1))).all() + .group_by(text('books_series_link.series')).order_by(db.Series.sort).all() + charlist = db.session.query(func.upper(func.substr(db.Series.sort, 1, 1)).label('char')) \ + .join(db.books_series_link).join(db.Books).filter(common_filters()) \ + .group_by(func.upper(func.substr(db.Series.sort, 1, 1))).all() return render_title_template('list.html', entries=entries, folder='web.books_list', charlist=charlist, title=_(u"Series"), page="serieslist", data="series") else: @@ -787,8 +792,8 @@ def series_list(): def ratings_list(): if current_user.check_visibility(constants.SIDEBAR_RATING): entries = db.session.query(db.Ratings, func.count('books_ratings_link.book').label('count'), - (db.Ratings.rating/2).label('name'))\ - .join(db.books_ratings_link).join(db.Books).filter(common_filters())\ + (db.Ratings.rating / 2).label('name')) \ + .join(db.books_ratings_link).join(db.Books).filter(common_filters()) \ .group_by(text('books_ratings_link.rating')).order_by(db.Ratings.rating).all() return render_title_template('list.html', entries=entries, folder='web.books_list', charlist=list(), title=_(u"Ratings list"), page="ratingslist", data="ratings") @@ -800,8 +805,8 @@ def ratings_list(): @login_required_if_no_ano def formats_list(): if current_user.check_visibility(constants.SIDEBAR_FORMAT): - entries = db.session.query(db.Data, func.count('data.book').label('count'),db.Data.format.label('format'))\ - .join(db.Books).filter(common_filters())\ + entries = db.session.query(db.Data, func.count('data.book').label('count'), db.Data.format.label('format')) \ + .join(db.Books).filter(common_filters()) \ .group_by(db.Data.format).order_by(db.Data.format).all() return render_title_template('list.html', entries=entries, folder='web.books_list', charlist=list(), title=_(u"File formats list"), page="formatslist", data="formats") @@ -842,12 +847,12 @@ def language_overview(): @login_required_if_no_ano def category_list(): if current_user.check_visibility(constants.SIDEBAR_CATEGORY): - entries = db.session.query(db.Tags, func.count('books_tags_link.book').label('count'))\ - .join(db.books_tags_link).join(db.Books).order_by(db.Tags.name).filter(common_filters())\ + entries = db.session.query(db.Tags, func.count('books_tags_link.book').label('count')) \ + .join(db.books_tags_link).join(db.Books).order_by(db.Tags.name).filter(common_filters()) \ .group_by(text('books_tags_link.tag')).all() - charlist = db.session.query(func.upper(func.substr(db.Tags.name,1,1)).label('char')) \ + charlist = db.session.query(func.upper(func.substr(db.Tags.name, 1, 1)).label('char')) \ .join(db.books_tags_link).join(db.Books).filter(common_filters()) \ - .group_by(func.upper(func.substr(db.Tags.name,1,1))).all() + .group_by(func.upper(func.substr(db.Tags.name, 1, 1))).all() return render_title_template('list.html', entries=entries, folder='web.books_list', charlist=charlist, title=_(u"Categories"), page="catlist", data="category") else: @@ -873,6 +878,7 @@ def reconnect(): db.reconnect_db(config) return json.dumps({}) + @web.route("/search", methods=["GET"]) @login_required_if_no_ano def search(): @@ -915,7 +921,7 @@ def advanced_search(): rating_high = request.args.get("ratinglow") description = request.args.get("comment") if author_name: - author_name = author_name.strip().lower().replace(',','|') + author_name = author_name.strip().lower().replace(',', '|') if book_title: book_title = book_title.strip().lower() if publisher: @@ -929,22 +935,22 @@ def advanced_search(): cc_present = True if include_tag_inputs or exclude_tag_inputs or include_series_inputs or exclude_series_inputs or \ - include_languages_inputs or exclude_languages_inputs or author_name or book_title or \ - publisher or pub_start or pub_end or rating_low or rating_high or description or cc_present or \ - include_extension_inputs or exclude_extension_inputs: + include_languages_inputs or exclude_languages_inputs or author_name or book_title or \ + publisher or pub_start or pub_end or rating_low or rating_high or description or cc_present or \ + include_extension_inputs or exclude_extension_inputs: searchterm = [] searchterm.extend((author_name.replace('|', ','), book_title, publisher)) if pub_start: try: searchterm.extend([_(u"Published after ") + - format_date(datetime.datetime.strptime(pub_start,"%Y-%m-%d"), + format_date(datetime.datetime.strptime(pub_start, "%Y-%m-%d"), format='medium', locale=get_locale())]) except ValueError: pub_start = u"" if pub_end: try: searchterm.extend([_(u"Published before ") + - format_date(datetime.datetime.strptime(pub_end,"%Y-%m-%d"), + format_date(datetime.datetime.strptime(pub_end, "%Y-%m-%d"), format='medium', locale=get_locale())]) except ValueError: pub_start = u"" @@ -1011,13 +1017,13 @@ def advanced_search(): custom_query = request.args.get('custom_column_' + str(c.id)) if custom_query: if c.datatype == 'bool': - q = q.filter(getattr(db.Books, 'custom_column_'+str(c.id)).any( + q = q.filter(getattr(db.Books, 'custom_column_' + str(c.id)).any( db.cc_classes[c.id].value == (custom_query == "True"))) elif c.datatype == 'int': - q = q.filter(getattr(db.Books, 'custom_column_'+str(c.id)).any( + q = q.filter(getattr(db.Books, 'custom_column_' + str(c.id)).any( db.cc_classes[c.id].value == custom_query)) else: - q = q.filter(getattr(db.Books, 'custom_column_'+str(c.id)).any( + 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() ids = list() @@ -1027,11 +1033,11 @@ def advanced_search(): return render_title_template('search.html', searchterm=searchterm, entries=q, title=_(u"search"), page="search") # prepare data for search-form - tags = db.session.query(db.Tags).join(db.books_tags_link).join(db.Books).filter(common_filters())\ + tags = db.session.query(db.Tags).join(db.books_tags_link).join(db.Books).filter(common_filters()) \ .group_by(text('books_tags_link.tag')).order_by(db.Tags.name).all() - series = db.session.query(db.Series).join(db.books_series_link).join(db.Books).filter(common_filters())\ + series = db.session.query(db.Series).join(db.books_series_link).join(db.Books).filter(common_filters()) \ .group_by(text('books_series_link.series')).order_by(db.Series.name).filter(common_filters()).all() - extensions = db.session.query(db.Data).join(db.Books).filter(common_filters())\ + extensions = db.session.query(db.Data).join(db.Books).filter(common_filters()) \ .group_by(db.Data.format).order_by(db.Data.format).all() if current_user.filter_language() == u"all": @@ -1045,12 +1051,12 @@ def advanced_search(): def render_read_books(page, are_read, as_xml=False, order=None, *args, **kwargs): order = order or [] if not config.config_read_column: - readBooks = ub.session.query(ub.ReadBook).filter(ub.ReadBook.user_id == int(current_user.id))\ + readBooks = ub.session.query(ub.ReadBook).filter(ub.ReadBook.user_id == int(current_user.id)) \ .filter(ub.ReadBook.is_read == True).all() readBookIds = [x.book_id for x in readBooks] else: try: - readBooks = db.session.query(db.cc_classes[config.config_read_column])\ + readBooks = db.session.query(db.cc_classes[config.config_read_column]) \ .filter(db.cc_classes[config.config_read_column].value == True).all() readBookIds = [x.book for x in readBooks] except KeyError: @@ -1094,7 +1100,7 @@ def get_cover(book_id): def serve_book(book_id, book_format, anyname): book_format = book_format.split(".")[0] book = db.session.query(db.Books).filter(db.Books.id == book_id).first() - data = db.session.query(db.Data).filter(db.Data.book == book.id).filter(db.Data.format == book_format.upper())\ + data = db.session.query(db.Data).filter(db.Data.book == book.id).filter(db.Data.format == book_format.upper()) \ .first() log.info('Serving book: %s', data.name) if config.config_use_google_drive: @@ -1105,6 +1111,7 @@ def serve_book(book_id, book_format, anyname): else: return send_from_directory(os.path.join(config.config_calibre_dir, book.path), data.name + "." + book_format) + @web.route("/download//", defaults={'anyname': 'None'}) @web.route("/download///") @login_required_if_no_ano @@ -1121,7 +1128,7 @@ def send_to_kindle(book_id, book_format, convert): flash(_(u"Please configure the SMTP mail settings first..."), category="error") elif current_user.kindle_mail: result = send_mail(book_id, book_format, convert, current_user.kindle_mail, config.config_calibre_dir, - current_user.nickname) + current_user.nickname) if result is None: flash(_(u"Book successfully queued for sending to %(kindlemail)s", kindlemail=current_user.kindle_mail), category="success") @@ -1168,7 +1175,7 @@ def register(): content.password = generate_password_hash(password) content.role = config.config_default_role content.sidebar_view = config.config_default_show - #content.mature_content = bool(config.config_default_show & constants.MATURE_CONTENT) + # content.mature_content = bool(config.config_default_show & constants.MATURE_CONTENT) try: ub.session.add(content) ub.session.commit() @@ -1206,25 +1213,23 @@ def login(): flash(_(u"Cannot activate LDAP authentication"), category="error") if request.method == "POST": form = request.form.to_dict() - user = ub.session.query(ub.User).filter(func.lower(ub.User.nickname) == form['username'].strip().lower())\ + user = ub.session.query(ub.User).filter(func.lower(ub.User.nickname) == form['username'].strip().lower()) \ .first() if config.config_login_type == constants.LOGIN_LDAP and services.ldap and user and form['password'] != "": login_result, error = services.ldap.bind_user(form['username'], form['password']) - # None if credentials are not matching - # -1 if LDAP Server error - # 0 if wrong passwort if login_result: login_user(user, remember=True) log.debug(u"You are now logged in as: '%s'", user.nickname) flash(_(u"you are now logged in as: '%(nickname)s'", nickname=user.nickname), category="success") return redirect_back(url_for("web.index")) - elif login_result is None and user and check_password_hash(str(user.password), form['password']) and user.nickname != "Guest": + elif login_result is None and user and check_password_hash(str(user.password), form['password']) \ + and user.nickname != "Guest": login_user(user, remember=True) log.info("Local Fallback Login as: '%s'", user.nickname) flash(_(u"Fallback Login as: '%(nickname)s', LDAP Server not reachable, or user not known", nickname=user.nickname), - category="warning") + category="warning") return redirect_back(url_for("web.index")) elif login_result is None: log.info(error) @@ -1258,9 +1263,18 @@ def login(): log.info('Login failed for user "%s" IP-adress: %s', form['username'], ipAdress) flash(_(u"Wrong Username or Password"), category="error") + if feature_support['oauth']: + oauth_status = get_oauth_status() + else: + oauth_status = None next_url = url_for('web.index') - return render_title_template('login.html', title=_(u"login"), next_url=next_url, config=config, - mail = config.get_mail_server_configured(), page="login") + return render_title_template('login.html', + title=_(u"login"), + next_url=next_url, + config=config, + # oauth_status=oauth_status, + oauth_check=oauth_check, + mail=config.get_mail_server_configured(), page="login") @web.route('/logout') @@ -1429,7 +1443,7 @@ def profile(): if "Show_detail_random" in to_save: current_user.sidebar_view += constants.DETAIL_RANDOM - #current_user.mature_content = "Show_mature_content" in to_save + # current_user.mature_content = "Show_mature_content" in to_save try: ub.session.commit() @@ -1440,12 +1454,12 @@ def profile(): return render_title_template("user_edit.html", content=current_user, downloads=downloads, translations=translations, kobo_support=kobo_support, title=_(u"%(name)s's profile", name=current_user.nickname), page="me", - registered_oauth=oauth_check, oauth_status=oauth_status) + registered_oauth=oauth_check, oauth_status=oauth_status) flash(_(u"Profile updated"), category="success") log.debug(u"Profile updated") return render_title_template("user_edit.html", translations=translations, profile=1, languages=languages, content=current_user, downloads=downloads, kobo_support=kobo_support, - title= _(u"%(name)s's profile", name=current_user.nickname), + title=_(u"%(name)s's profile", name=current_user.nickname), page="me", registered_oauth=oauth_check, oauth_status=oauth_status) @@ -1523,12 +1537,12 @@ def show_book(book_id): if not current_user.is_anonymous: if not config.config_read_column: - matching_have_read_book = ub.session.query(ub.ReadBook).\ + matching_have_read_book = ub.session.query(ub.ReadBook). \ filter(and_(ub.ReadBook.user_id == int(current_user.id), ub.ReadBook.book_id == book_id)).all() have_read = len(matching_have_read_book) > 0 and matching_have_read_book[0].is_read else: try: - matching_have_read_book = getattr(entries, 'custom_column_'+str(config.config_read_column)) + matching_have_read_book = getattr(entries, 'custom_column_' + str(config.config_read_column)) have_read = len(matching_have_read_book) > 0 and matching_have_read_book[0].value except KeyError: log.error("Custom Column No.%d is not existing in calibre database", config.config_read_column) @@ -1550,7 +1564,8 @@ def show_book(book_id): audioentries.append(media_format.format.lower()) return render_title_template('detail.html', entry=entries, audioentries=audioentries, cc=cc, - is_xhr=request.headers.get('X-Requested-With')=='XMLHttpRequest', title=entries.title, books_shelfs=book_in_shelfs, + is_xhr=request.headers.get('X-Requested-With') == 'XMLHttpRequest', + title=entries.title, books_shelfs=book_in_shelfs, have_read=have_read, kindle_list=kindle_list, reader_list=reader_list, page="book") else: log.debug(u"Error opening eBook. File does not exist or file is not accessible:") diff --git a/optional-requirements.txt b/optional-requirements.txt index fd40ca9d..3c10b52d 100644 --- a/optional-requirements.txt +++ b/optional-requirements.txt @@ -18,7 +18,7 @@ python-Levenshtein>=0.12.0,<0.13.0 # ldap login python_ldap>=3.0.0,<3.3.0 -flask-simpleldap>1.3.0,<1.5.0 +flask-simpleldap>=1.4.0,<1.5.0 #oauth flask-dance>=1.4.0,<3.1.0 diff --git a/test/Calibre-Web TestSummary.html b/test/Calibre-Web TestSummary.html index cb9772bf..4138e113 100755 --- a/test/Calibre-Web TestSummary.html +++ b/test/Calibre-Web TestSummary.html @@ -36,17 +36,17 @@
-

Start Time: 2020-04-13 20:58:27

+

Start Time: 2020-04-15 20:58:24

-

Stop Time: 2020-04-13 21:36:17

+

Stop Time: 2020-04-15 21:44:34

-

Duration: 2002.47 s

+

Duration: 2409.70 s

@@ -301,8 +301,8 @@ test_ebook_convert.test_ebook_convert 11 - 7 - 4 + 11 + 0 0 0 @@ -339,33 +339,11 @@ - +
test_convert_email
- -
- FAIL -
- - - - + PASS @@ -379,64 +357,20 @@ AssertionError: 'Failed' != 'Finished' - +
test_convert_only
- -
- FAIL -
- - - - + PASS - +
test_convert_parameter
- -
- FAIL -
- - - - + PASS @@ -459,33 +393,11 @@ AssertionError: 'Failed' != 'Finished' - +
test_email_only
- -
- FAIL -
- - - - + PASS @@ -502,13 +414,13 @@ AssertionError: 'Failed' != 'Finished' test_edit_books.test_edit_books - 30 - 27 + 31 + 28 0 0 3 - Detail + Detail @@ -720,7 +632,7 @@ AssertionError: 'Failed' != 'Finished' -
test_typeahead_language
+
test_typeahead_functions
PASS @@ -729,7 +641,7 @@ AssertionError: 'Failed' != 'Finished' -
test_typeahead_publisher
+
test_typeahead_language
PASS @@ -738,7 +650,7 @@ AssertionError: 'Failed' != 'Finished' -
test_typeahead_series
+
test_typeahead_publisher
PASS @@ -747,7 +659,7 @@ AssertionError: 'Failed' != 'Finished' -
test_typeahead_tag
+
test_typeahead_series
PASS @@ -756,7 +668,7 @@ AssertionError: 'Failed' != 'Finished' -
test_upload_book_cbr
+
test_typeahead_tag
PASS @@ -765,7 +677,7 @@ AssertionError: 'Failed' != 'Finished' -
test_upload_book_cbt
+
test_upload_book_cbr
PASS @@ -774,7 +686,7 @@ AssertionError: 'Failed' != 'Finished' -
test_upload_book_cbz
+
test_upload_book_cbt
PASS @@ -783,7 +695,7 @@ AssertionError: 'Failed' != 'Finished' -
test_upload_book_epub
+
test_upload_book_cbz
PASS @@ -792,7 +704,7 @@ AssertionError: 'Failed' != 'Finished' -
test_upload_book_fb2
+
test_upload_book_epub
PASS @@ -801,7 +713,7 @@ AssertionError: 'Failed' != 'Finished' -
test_upload_book_lit
+
test_upload_book_fb2
PASS @@ -810,7 +722,7 @@ AssertionError: 'Failed' != 'Finished' -
test_upload_book_mobi
+
test_upload_book_lit
PASS @@ -819,7 +731,7 @@ AssertionError: 'Failed' != 'Finished' -
test_upload_book_pdf
+
test_upload_book_mobi
PASS @@ -827,6 +739,15 @@ AssertionError: 'Failed' != 'Finished' + +
test_upload_book_pdf
+ + PASS + + + + +
test_upload_cover_hdd
@@ -921,46 +842,132 @@ AssertionError: 'Failed' != 'Finished' - unittest.suite._ErrorHolder - 1 + test_helper.calibre_helper + 13 + 13 0 0 - 1 0 - Detail + Detail - + -
setUpClass (test_helper)
+
test_author_sort
- -
- ERROR -
- - - + PASS + + + + + + +
test_author_sort_comma
+ PASS + + + + + + +
test_author_sort_junior
+ + PASS + + + + + + +
test_author_sort_oneword
+ + PASS + + + + + + +
test_author_sort_roman
+ + PASS + + + + + + +
test_check_Limit_Length
+ + PASS + + + + + + +
test_check_char_replacement
+ + PASS + + + + + + +
test_check_chinese_Characters
+ + PASS + + + + + + +
test_check_degEUR_replacement
+ + PASS + + + + + + +
test_check_doubleS
+ + PASS + + + + + + +
test_check_finish_Dot
+ + PASS + + + + + + +
test_check_high23
+ + PASS + + + + + + +
test_check_umlauts
+ + PASS @@ -968,13 +975,13 @@ ModuleNotFoundError: No module named 'babel' test_kobo_sync.test_kobo_sync + 3 2 1 - 1 0 0 - Detail + Detail @@ -997,11 +1004,11 @@ ModuleNotFoundError: No module named 'babel'
Traceback (most recent call last):
-  File "/home/matthias/Entwicklung/calibre-web-test/test/test_kobo_sync.py", line 87, in test_check_sync
+  File "/home/matthias/Entwicklung/calibre-web-test/test/test_kobo_sync.py", line 89, in test_check_sync
     self.assertEqual(r.json()['Resources']['image_url_quality_template'], self.kobo_adress+"/{ImageId}/{width}/{height}/image.jpg")
-AssertionError: 'http[35 chars]1d50f75da2a5578ea9baa6a77/{ImageId}/image.jpg' != 'http[35 chars]1d50f75da2a5578ea9baa6a77/{ImageId}/{width}/{height}/image.jpg'
-- http://192.168.188.33:8083/kobo/c97e71f1d50f75da2a5578ea9baa6a77/{ImageId}/image.jpg
-+ http://192.168.188.33:8083/kobo/c97e71f1d50f75da2a5578ea9baa6a77/{ImageId}/{width}/{height}/image.jpg
+AssertionError: 'http[35 chars]4304a11f244beb23c60f4b7b0/{ImageId}/image.jpg' != 'http[35 chars]4304a11f244beb23c60f4b7b0/{ImageId}/{width}/{height}/image.jpg'
+- http://192.168.188.33:8083/kobo/1b4c3e84304a11f244beb23c60f4b7b0/{ImageId}/image.jpg
++ http://192.168.188.33:8083/kobo/1b4c3e84304a11f244beb23c60f4b7b0/{ImageId}/{width}/{height}/image.jpg
 ?                                                                            +++++++++++++++++
@@ -1013,6 +1020,15 @@ AssertionError: 'http[35 chars]1d50f75da2a5578ea9baa6a77/{ImageId}/image.jpg' != + +
test_kobo_about
+ + PASS + + + + +
test_sync_invalid
@@ -1023,62 +1039,78 @@ AssertionError: 'http[35 chars]1d50f75da2a5578ea9baa6a77/{ImageId}/image.jpg' != - unittest.loader._FailedTest - 1 + test_ldap.test_ldap_login + 7 + 7 0 0 - 1 0 - Detail + Detail - + -
unittestloader_FailedTest)
+
test_LDAP_SSL
- -
- ERROR -
- - - + PASS + + + + + + +
test_LDAP_STARTTLS
+ PASS + + + + + + +
test_LDAP_fallback_Login
+ + PASS + + + + + + +
test_LDAP_import
+ + PASS + + + + + + +
test_LDAP_login
+ + PASS + + + + + + +
test_invalid_LDAP
+ + PASS + + + + + + +
test_ldap_about
+ + PASS @@ -1910,13 +1942,13 @@ AssertionError: False is not true : logfile config value is not empty after rese test_visiblilitys.calibre_web_visibilitys - 22 - 22 + 23 + 23 0 0 0 - Detail + Detail @@ -2104,7 +2136,7 @@ AssertionError: False is not true : logfile config value is not empty after rese -
test_user_email_available
+
test_search_functions
PASS @@ -2112,6 +2144,15 @@ AssertionError: False is not true : logfile config value is not empty after rese + +
test_user_email_available
+ + PASS + + + + +
test_user_visibility_sidebar
@@ -2122,10 +2163,10 @@ AssertionError: False is not true : logfile config value is not empty after rese Total - 162 - 147 - 6 + 183 + 174 2 + 0 7   @@ -2154,13 +2195,13 @@ AssertionError: False is not true : logfile config value is not empty after rese Platform - Linux 5.5.16-1-MANJARO #1 SMP PREEMPT Wed Apr 8 10:07:00 UTC 2020 x86_64 + Linux 5.3.0-46-generic #38~18.04.1-Ubuntu SMP Tue Mar 31 04:17:56 UTC 2020 x86_64 x86_64 Basic Python - 3.8.2 + 3.7.5 Basic @@ -2214,7 +2255,7 @@ AssertionError: False is not true : logfile config value is not empty after rese requests - 2.22.0 + 2.23.0 Basic @@ -2274,7 +2315,7 @@ AssertionError: False is not true : logfile config value is not empty after rese