from datetime import datetime from flask import ( abort, current_app, flash, jsonify, Markup, redirect, render_template, request, send_from_directory, url_for ) from flask_login import current_user, login_required from threading import Thread import os from .decorators import corpus_follower_permission_required, corpus_owner_or_admin_required from app import db, hashids from app.decorators import content_negotiation from app.models import ( Corpus, CorpusFile, CorpusFollowerAssociation, CorpusFollowerRole, CorpusStatus, User ) from . import bp from .forms import ( CreateCorpusFileForm, CreateCorpusForm, UpdateCorpusFileForm ) @bp.route('/create', methods=['GET', 'POST']) @login_required def create_corpus(): form = CreateCorpusForm() if form.validate_on_submit(): try: corpus = Corpus.create( title=form.title.data, description=form.description.data, user=current_user ) except OSError: abort(500) db.session.commit() message = Markup( f'Corpus "{corpus.title}" created' ) flash(message, 'corpus') return redirect(corpus.url) return render_template( 'corpora/create_corpus.html.j2', form=form, title='Create corpus' ) @bp.route('/public') @login_required def public_corpora(): corpora = [ c.to_json_serializeable() for c in Corpus.query.filter(Corpus.is_public == True).all() ] return render_template( 'corpora/public_corpora.html.j2', corpora=corpora, title='Corpora' ) ############################################################################## # Corpus # ############################################################################## #region corpus @bp.route('/') @login_required def corpus(corpus_id): corpus = Corpus.query.get_or_404(corpus_id) corpus_follower_roles = CorpusFollowerRole.query.all() if corpus.user == current_user or current_user.is_administrator(): return render_template( 'corpora/corpus.html.j2', corpus=corpus, corpus_follower_roles=corpus_follower_roles, title='Corpus' ) if current_user.is_following_corpus(corpus) or corpus.is_public: cfa = CorpusFollowerAssociation.query.filter_by(corpus_id=corpus_id, follower_id=current_user.id).first_or_404() corpus_files = [x.to_json_serializeable() for x in corpus.files] owner = corpus.user.to_json_serializeable() return render_template( 'corpora/public_corpus.html.j2', corpus=corpus, corpus_files=corpus_files, cfa=cfa, owner=owner, title='Corpus', ) abort(403) @bp.route('//analyse') @login_required @corpus_follower_permission_required('VIEW') def analyse_corpus(corpus_id): corpus = Corpus.query.get_or_404(corpus_id) return render_template( 'corpora/analyse_corpus.html.j2', corpus=corpus, title=f'Analyse Corpus {corpus.title}' ) @bp.route('//follow/') @login_required def follow_corpus(corpus_id, token): corpus = Corpus.query.get_or_404(corpus_id) if current_user.follow_corpus_by_token(token): db.session.commit() flash(f'You are following {corpus.title} now', category='corpus') return redirect(url_for('corpora.corpus', corpus_id=corpus_id)) abort(403) @bp.route('/import', methods=['GET', 'POST']) @login_required def import_corpus(): abort(503) @bp.route('//export') @login_required def export_corpus(corpus_id): abort(503) #region json-routes @bp.route('/', methods=['DELETE']) @login_required @corpus_owner_or_admin_required @content_negotiation(produces='application/json') def delete_corpus(corpus_id): def _delete_corpus(app, corpus_id): with app.app_context(): corpus = Corpus.query.get(corpus_id) corpus.delete() db.session.commit() corpus = Corpus.query.get_or_404(corpus_id) thread = Thread( target=_delete_corpus, args=(current_app._get_current_object(), corpus.id) ) thread.start() response_data = { 'message': f'Corpus "{corpus.title}" marked for deletion', 'category': 'corpus' } response = jsonify(response_data) response.status_code = 200 return response @bp.route('//build', methods=['POST']) @login_required @corpus_owner_or_admin_required @content_negotiation(produces='application/json') def build_corpus(corpus_id): def _build_corpus(app, corpus_id): with app.app_context(): corpus = Corpus.query.get(corpus_id) corpus.build() db.session.commit() corpus = Corpus.query.get_or_404(corpus_id) if not (corpus.user == current_user or current_user.is_administrator()): abort(403) if len(corpus.files.all()) == 0: abort(409) thread = Thread( target=_build_corpus, args=(current_app._get_current_object(), corpus_id) ) thread.start() response_data = { 'message': f'Corpus "{corpus.title}" marked for building', 'category': 'corpus' } response = jsonify(response_data) response.status_code = 202 return response @bp.route('//generate-corpus-share-link', methods=['POST']) @login_required @corpus_follower_permission_required('GENERATE_SHARE_LINK') @content_negotiation(consumes='application/json', produces='application/json') def generate_corpus_share_link(corpus_id): corpus_hashid = hashids.encode(corpus_id) data = request.json if not isinstance(data, dict): abort(400) expiration = data.get('expiration') if not isinstance(expiration, str): abort(400) role_name = data.get('role') if not isinstance(role_name, str): abort(400) expiration_date = datetime.strptime(expiration, '%b %d, %Y') cfr = CorpusFollowerRole.query.filter_by(name=role_name).first() if cfr is None: abort(400) token = current_user.generate_follow_corpus_token(corpus_hashid, role_name, expiration_date) corpus_share_link = url_for( 'corpora.follow_corpus', corpus_id=corpus_id, token=token, _external=True ) response_data = { 'message': 'Corpus share link generated', 'category': 'corpus', 'corpusShareLink': corpus_share_link } response = jsonify(response_data) response.status_code = 200 return response @bp.route('//is_public', methods=['PUT']) @login_required @corpus_owner_or_admin_required @content_negotiation(consumes='application/json', produces='application/json') def update_corpus_is_public(corpus_id): is_public = request.json if not isinstance(is_public, bool): abort(400) corpus = Corpus.query.get_or_404(corpus_id) corpus.is_public = is_public db.session.commit() response_data = { 'message': ( f'Corpus "{corpus.title}" is now' f' {"public" if is_public else "private"}' ), 'category': 'corpus' } response = jsonify(response_data) response.status_code = 200 return response #endregion json-routes #endregion corpus ############################################################################## # Corpus/Files # ############################################################################## #region files @bp.route('//files/create', methods=['GET', 'POST']) @login_required @corpus_follower_permission_required('ADD_CORPUS_FILE') def create_corpus_file(corpus_id): corpus = Corpus.query.get_or_404(corpus_id) if not (corpus.user == current_user or current_user.is_administrator()): abort(403) form = CreateCorpusFileForm() if form.is_submitted(): if not form.validate(): response = {'errors': form.errors} return response, 400 try: corpus_file = CorpusFile.create( form.vrt.data, address=form.address.data, author=form.author.data, booktitle=form.booktitle.data, chapter=form.chapter.data, editor=form.editor.data, institution=form.institution.data, journal=form.journal.data, pages=form.pages.data, publisher=form.publisher.data, publishing_year=form.publishing_year.data, school=form.school.data, title=form.title.data, mimetype='application/vrt+xml', corpus=corpus ) except (AttributeError, OSError): abort(500) corpus.status = CorpusStatus.UNPREPARED db.session.commit() message = Markup( 'Corpus file' f'"{corpus_file.filename}" added' ) flash(message, category='corpus') return {}, 201, {'Location': corpus.url} return render_template( 'corpora/create_corpus_file.html.j2', corpus=corpus, form=form, title='Add corpus file' ) @bp.route('//files/', methods=['GET', 'POST']) @login_required @corpus_follower_permission_required('UPDATE_CORPUS_FILE') def corpus_file(corpus_id, corpus_file_id): corpus_file = CorpusFile.query.filter_by(corpus_id = corpus_id, id=corpus_file_id).first_or_404() form = UpdateCorpusFileForm(data=corpus_file.to_json_serializeable()) if form.validate_on_submit(): form.populate_obj(corpus_file) if db.session.is_modified(corpus_file): corpus_file.corpus.status = CorpusStatus.UNPREPARED db.session.commit() message = Markup(f'Corpus file "{corpus_file.filename}" updated') flash(message, category='corpus') return redirect(corpus_file.corpus.url) return render_template( 'corpora/corpus_file.html.j2', corpus=corpus_file.corpus, corpus_file=corpus_file, form=form, title='Edit corpus file' ) @bp.route('//files//download') @login_required @corpus_follower_permission_required('VIEW') def download_corpus_file(corpus_id, corpus_file_id): corpus_file = CorpusFile.query.filter_by(corpus_id = corpus_id, id=corpus_file_id).first_or_404() if not (corpus_file.corpus.user == current_user or current_user.is_administrator()): abort(403) return send_from_directory( os.path.dirname(corpus_file.path), os.path.basename(corpus_file.path), as_attachment=True, attachment_filename=corpus_file.filename, mimetype=corpus_file.mimetype ) #region json-routes @bp.route('//files/', methods=['DELETE']) @login_required @corpus_follower_permission_required('REMOVE_CORPUS_FILE') @content_negotiation(produces='application/json') def delete_corpus_file(corpus_id, corpus_file_id): def _delete_corpus_file(app, corpus_file_id): with app.app_context(): corpus_file = CorpusFile.query.get(corpus_file_id) corpus_file.delete() db.session.commit() corpus_file = CorpusFile.query.filter_by(corpus_id=corpus_id, id=corpus_file_id).first_or_404() thread = Thread( target=_delete_corpus_file, args=(current_app._get_current_object(), corpus_file.id) ) thread.start() return {}, 202 #endregion json-routes #endregion files ############################################################################## # Corpus/Followers # ############################################################################## #region followers #region json-routes @bp.route('//followers', methods=['POST']) @login_required @corpus_owner_or_admin_required @content_negotiation(consumes='application/json', produces='application/json') def add_corpus_followers(corpus_id): usernames = request.json if not (isinstance(usernames, list) or all(isinstance(u, str) for u in usernames)): abort(400) corpus = Corpus.query.get_or_404(corpus_id) for username in usernames: user = User.query.filter_by(username=username, is_public=True).first_or_404() user.follow_corpus(corpus) db.session.commit() resonse_data = { 'message': f'Users are now following "{corpus.title}"', 'category': 'corpus' } response = jsonify(resonse_data) response.status_code = 200 return response @bp.route('//followers/', methods=['DELETE']) @login_required @content_negotiation(produces='application/json') def unfollow_corpus(corpus_id, follower_id): corpus = Corpus.query.get_or_404(corpus_id) follower = User.query.get_or_404(follower_id) if not (corpus.user == current_user or follower == current_user or current_user.is_administrator()): abort(403) if not follower.is_following_corpus(corpus): abort(409) # 'User is not following the corpus' follower.unfollow_corpus(corpus) db.session.commit() response_data = { 'message': \ f'"{follower.username}" is not following "{corpus.title}" anymore', 'category': 'corpus' } response = jsonify(response_data) response.status_code = 200 return response @bp.route('//followers//role', methods=['PUT']) @login_required @corpus_owner_or_admin_required @content_negotiation(consumes='application/json', produces='application/json') def add_permission(corpus_id, follower_id): role_name = request.json if not isinstance(role_name, str): abort(400) cfr = CorpusFollowerRole.query.filter_by(name=role_name).first() if cfr is None: abort(400) cfa = CorpusFollowerAssociation.query.filter_by(corpus_id=corpus_id, follower_id=follower_id).first_or_404() cfa.role = cfr db.session.commit() resonse_data = { 'message': f'User "{cfa.follower.username}" is now {cfa.role.name}', 'category': 'corpus' } response = jsonify(resonse_data) response.status_code = 200 return response #endregion json-routes #endregion followers