mirror of
https://gitlab.ub.uni-bielefeld.de/sfb1288inf/nopaque.git
synced 2024-12-25 02:44:18 +00:00
Create and use a decorator for background functions
This commit is contained in:
parent
79c9ca97b2
commit
bc27744946
@ -5,7 +5,7 @@ from . import auth
|
|||||||
from .forms import (LoginForm, ResetPasswordForm, ResetPasswordRequestForm,
|
from .forms import (LoginForm, ResetPasswordForm, ResetPasswordRequestForm,
|
||||||
RegistrationForm)
|
RegistrationForm)
|
||||||
from .. import db
|
from .. import db
|
||||||
from ..email import create_message, send_async
|
from ..email import create_message, send
|
||||||
from ..models import User
|
from ..models import User
|
||||||
import os
|
import os
|
||||||
import shutil
|
import shutil
|
||||||
@ -70,7 +70,7 @@ def register():
|
|||||||
token = user.generate_confirmation_token()
|
token = user.generate_confirmation_token()
|
||||||
msg = create_message(user.email, 'Confirm Your Account',
|
msg = create_message(user.email, 'Confirm Your Account',
|
||||||
'auth/email/confirm', token=token, user=user)
|
'auth/email/confirm', token=token, user=user)
|
||||||
send_async(msg)
|
send(msg)
|
||||||
flash('A confirmation email has been sent to you by email.')
|
flash('A confirmation email has been sent to you by email.')
|
||||||
return redirect(url_for('auth.login'))
|
return redirect(url_for('auth.login'))
|
||||||
return render_template('auth/register.html.j2',
|
return render_template('auth/register.html.j2',
|
||||||
@ -107,7 +107,7 @@ def resend_confirmation():
|
|||||||
token = current_user.generate_confirmation_token()
|
token = current_user.generate_confirmation_token()
|
||||||
msg = create_message(current_user.email, 'Confirm Your Account',
|
msg = create_message(current_user.email, 'Confirm Your Account',
|
||||||
'auth/email/confirm', token=token, user=current_user)
|
'auth/email/confirm', token=token, user=current_user)
|
||||||
send_async(msg)
|
send(msg)
|
||||||
flash('A new confirmation email has been sent to you by email.')
|
flash('A new confirmation email has been sent to you by email.')
|
||||||
return redirect(url_for('auth.unconfirmed'))
|
return redirect(url_for('auth.unconfirmed'))
|
||||||
|
|
||||||
@ -126,7 +126,7 @@ def reset_password_request():
|
|||||||
msg = create_message(user.email, 'Reset Your Password',
|
msg = create_message(user.email, 'Reset Your Password',
|
||||||
'auth/email/reset_password', token=token,
|
'auth/email/reset_password', token=token,
|
||||||
user=user)
|
user=user)
|
||||||
send_async(msg)
|
send(msg)
|
||||||
flash('An email with instructions to reset your password has been '
|
flash('An email with instructions to reset your password has been '
|
||||||
'sent to you.')
|
'sent to you.')
|
||||||
return redirect(url_for('auth.login'))
|
return redirect(url_for('auth.login'))
|
||||||
|
@ -1,30 +0,0 @@
|
|||||||
from ..models import Corpus, CorpusFile
|
|
||||||
|
|
||||||
|
|
||||||
def delete_corpus_(app, corpus_id):
|
|
||||||
with app.app_context():
|
|
||||||
corpus = Corpus.query.get(corpus_id)
|
|
||||||
if corpus is None:
|
|
||||||
# raise Exception('Corpus {} not found!'.format(corpus_id))
|
|
||||||
pass
|
|
||||||
else:
|
|
||||||
corpus.delete()
|
|
||||||
|
|
||||||
|
|
||||||
def delete_corpus_file_(app, corpus_file_id):
|
|
||||||
with app.app_context():
|
|
||||||
corpus_file = CorpusFile.query.get(corpus_file_id)
|
|
||||||
if corpus_file is None:
|
|
||||||
# raise Exception('Corpus file {} not found!'.format(corpus_file_id))
|
|
||||||
pass
|
|
||||||
else:
|
|
||||||
corpus_file.delete()
|
|
||||||
|
|
||||||
|
|
||||||
def edit_corpus_file_(app, corpus_file_id):
|
|
||||||
with app.app_context():
|
|
||||||
corpus_file = CorpusFile.query.get(corpus_file_id)
|
|
||||||
if corpus_file is None:
|
|
||||||
raise Exception('Corpus file {} not found!'.format(corpus_file_id))
|
|
||||||
else:
|
|
||||||
corpus_file.insert_metadata()
|
|
41
app/corpora/tasks.py
Normal file
41
app/corpora/tasks.py
Normal file
@ -0,0 +1,41 @@
|
|||||||
|
from ..decorators import background
|
||||||
|
from ..models import Corpus, CorpusFile
|
||||||
|
import os
|
||||||
|
import shutil
|
||||||
|
|
||||||
|
|
||||||
|
@background
|
||||||
|
def delete_corpus(app, corpus_id):
|
||||||
|
with app.app_context():
|
||||||
|
corpus = Corpus.query.get(corpus_id)
|
||||||
|
if corpus is None:
|
||||||
|
return
|
||||||
|
path = os.path.join(app.config['NOPAQUE_STORAGE'], str(corpus.user_id),
|
||||||
|
'corpora', str(corpus.id))
|
||||||
|
shutil.rmtree(path, ignore_errors=True)
|
||||||
|
corpus.delete()
|
||||||
|
|
||||||
|
|
||||||
|
@background
|
||||||
|
def delete_corpus_file(app, corpus_file_id):
|
||||||
|
with app.app_context():
|
||||||
|
corpus_file = CorpusFile.query.get(corpus_file_id)
|
||||||
|
if corpus_file is None:
|
||||||
|
return
|
||||||
|
path = os.path.join(app.config['NOPAQUE_STORAGE'], corpus_file.dir,
|
||||||
|
corpus_file.filename)
|
||||||
|
try:
|
||||||
|
os.remove(path)
|
||||||
|
except Exception:
|
||||||
|
pass
|
||||||
|
else:
|
||||||
|
corpus_file.delete()
|
||||||
|
|
||||||
|
|
||||||
|
@background
|
||||||
|
def edit_corpus_file(app, corpus_file_id):
|
||||||
|
with app.app_context():
|
||||||
|
corpus_file = CorpusFile.query.get(corpus_file_id)
|
||||||
|
if corpus_file is None:
|
||||||
|
raise Exception('Corpus file {} not found!'.format(corpus_file_id))
|
||||||
|
corpus_file.insert_metadata()
|
@ -1,10 +1,8 @@
|
|||||||
from flask import (abort, current_app, flash, make_response, redirect, request,
|
from flask import (abort, current_app, flash, make_response, redirect, request,
|
||||||
render_template, url_for, send_from_directory)
|
render_template, url_for, send_from_directory)
|
||||||
from flask_login import current_user, login_required
|
from flask_login import current_user, login_required
|
||||||
from threading import Thread
|
|
||||||
from . import corpora
|
from . import corpora
|
||||||
from .background_functions import (delete_corpus_, delete_corpus_file_,
|
from . import tasks
|
||||||
edit_corpus_file_)
|
|
||||||
from .forms import (AddCorpusFileForm, AddCorpusForm, EditCorpusFileForm,
|
from .forms import (AddCorpusFileForm, AddCorpusForm, EditCorpusFileForm,
|
||||||
QueryDownloadForm, QueryForm, DisplayOptionsForm,
|
QueryDownloadForm, QueryForm, DisplayOptionsForm,
|
||||||
InspectDisplayOptionsForm)
|
InspectDisplayOptionsForm)
|
||||||
@ -78,9 +76,7 @@ def delete_corpus(corpus_id):
|
|||||||
corpus = Corpus.query.get_or_404(corpus_id)
|
corpus = Corpus.query.get_or_404(corpus_id)
|
||||||
if not (corpus.creator == current_user or current_user.is_administrator()):
|
if not (corpus.creator == current_user or current_user.is_administrator()):
|
||||||
abort(403)
|
abort(403)
|
||||||
thread = Thread(target=delete_corpus_,
|
tasks.delete_corpus(corpus_id)
|
||||||
args=(current_app._get_current_object(), corpus.id))
|
|
||||||
thread.start()
|
|
||||||
flash('Corpus deleted!')
|
flash('Corpus deleted!')
|
||||||
return redirect(url_for('main.dashboard'))
|
return redirect(url_for('main.dashboard'))
|
||||||
|
|
||||||
@ -119,10 +115,7 @@ def add_corpus_file(corpus_id):
|
|||||||
title=add_corpus_file_form.title.data)
|
title=add_corpus_file_form.title.data)
|
||||||
db.session.add(corpus_file)
|
db.session.add(corpus_file)
|
||||||
db.session.commit()
|
db.session.commit()
|
||||||
thread = Thread(target=edit_corpus_file_,
|
tasks.edit_corpus_file(corpus_file.id)
|
||||||
args=(current_app._get_current_object(),
|
|
||||||
corpus_file.id))
|
|
||||||
thread.start()
|
|
||||||
flash('Corpus file added!')
|
flash('Corpus file added!')
|
||||||
return make_response(
|
return make_response(
|
||||||
{'redirect_url': url_for('corpora.corpus', corpus_id=corpus.id)},
|
{'redirect_url': url_for('corpora.corpus', corpus_id=corpus.id)},
|
||||||
@ -142,9 +135,7 @@ def delete_corpus_file(corpus_id, corpus_file_id):
|
|||||||
if not (corpus_file.corpus.creator == current_user
|
if not (corpus_file.corpus.creator == current_user
|
||||||
or current_user.is_administrator()):
|
or current_user.is_administrator()):
|
||||||
abort(403)
|
abort(403)
|
||||||
thread = Thread(target=delete_corpus_file_,
|
tasks.delete_corpus_file(corpus_file_id)
|
||||||
args=(current_app._get_current_object(), corpus_file.id))
|
|
||||||
thread.start()
|
|
||||||
flash('Corpus file deleted!')
|
flash('Corpus file deleted!')
|
||||||
return redirect(url_for('corpora.corpus', corpus_id=corpus_id))
|
return redirect(url_for('corpora.corpus', corpus_id=corpus_id))
|
||||||
|
|
||||||
@ -191,10 +182,7 @@ def edit_corpus_file(corpus_id, corpus_file_id):
|
|||||||
corpus_file.school = edit_corpus_file_form.school.data
|
corpus_file.school = edit_corpus_file_form.school.data
|
||||||
corpus_file.title = edit_corpus_file_form.title.data
|
corpus_file.title = edit_corpus_file_form.title.data
|
||||||
db.session.commit()
|
db.session.commit()
|
||||||
thread = Thread(target=edit_corpus_file_,
|
tasks.edit_corpus_file(corpus_file_id)
|
||||||
args=(current_app._get_current_object(),
|
|
||||||
corpus_file.id))
|
|
||||||
thread.start()
|
|
||||||
flash('Corpus file edited!')
|
flash('Corpus file edited!')
|
||||||
return redirect(url_for('corpora.corpus', corpus_id=corpus_id))
|
return redirect(url_for('corpora.corpus', corpus_id=corpus_id))
|
||||||
# If no form is submitted or valid, fill out fields with current values
|
# If no form is submitted or valid, fill out fields with current values
|
||||||
|
@ -1,16 +1,38 @@
|
|||||||
from flask import abort
|
from flask import abort, current_app
|
||||||
from flask_login import current_user
|
from flask_login import current_user
|
||||||
from flask_socketio import disconnect
|
from flask_socketio import disconnect
|
||||||
from functools import wraps
|
from functools import wraps
|
||||||
from .models import Permission
|
from threading import Thread
|
||||||
|
|
||||||
|
|
||||||
def admin_required(f):
|
def admin_required(f):
|
||||||
@wraps(f)
|
@wraps(f)
|
||||||
def wrapped(*args, **kwargs):
|
def wrapped(*args, **kwargs):
|
||||||
if not current_user.can(Permission.ADMIN):
|
if current_user.is_administrator:
|
||||||
abort(403)
|
|
||||||
return f(*args, **kwargs)
|
return f(*args, **kwargs)
|
||||||
|
else:
|
||||||
|
abort(403)
|
||||||
|
return wrapped
|
||||||
|
|
||||||
|
|
||||||
|
def background(f):
|
||||||
|
''' This decorator executes a function in a Thread '''
|
||||||
|
@wraps(f)
|
||||||
|
def wrapped(*args, **kwargs):
|
||||||
|
app = current_app._get_current_object()
|
||||||
|
thread = Thread(target=f, args=(app, *args), kwargs=kwargs)
|
||||||
|
thread.start()
|
||||||
|
return thread
|
||||||
|
return wrapped
|
||||||
|
|
||||||
|
|
||||||
|
def socketio_admin_required(f):
|
||||||
|
@wraps(f)
|
||||||
|
def wrapped(*args, **kwargs):
|
||||||
|
if current_user.is_administrator:
|
||||||
|
return f(*args, **kwargs)
|
||||||
|
else:
|
||||||
|
disconnect()
|
||||||
return wrapped
|
return wrapped
|
||||||
|
|
||||||
|
|
||||||
@ -22,13 +44,3 @@ def socketio_login_required(f):
|
|||||||
else:
|
else:
|
||||||
return f(*args, **kwargs)
|
return f(*args, **kwargs)
|
||||||
return wrapped
|
return wrapped
|
||||||
|
|
||||||
|
|
||||||
def socketio_admin_required(f):
|
|
||||||
@wraps(f)
|
|
||||||
def wrapped(*args, **kwargs):
|
|
||||||
if not current_user.can(Permission.ADMIN):
|
|
||||||
disconnect()
|
|
||||||
else:
|
|
||||||
return f(*args, **kwargs)
|
|
||||||
return wrapped
|
|
||||||
|
10
app/email.py
10
app/email.py
@ -1,7 +1,7 @@
|
|||||||
from flask import current_app, render_template
|
from flask import current_app, render_template
|
||||||
from flask_mail import Message
|
from flask_mail import Message
|
||||||
from threading import Thread
|
|
||||||
from . import mail
|
from . import mail
|
||||||
|
from .decorators import background
|
||||||
|
|
||||||
|
|
||||||
def create_message(recipient, subject, template, **kwargs):
|
def create_message(recipient, subject, template, **kwargs):
|
||||||
@ -15,13 +15,7 @@ def create_message(recipient, subject, template, **kwargs):
|
|||||||
return msg
|
return msg
|
||||||
|
|
||||||
|
|
||||||
|
@background
|
||||||
def send(app, msg):
|
def send(app, msg):
|
||||||
with app.app_context():
|
with app.app_context():
|
||||||
mail.send(msg)
|
mail.send(msg)
|
||||||
|
|
||||||
|
|
||||||
def send_async(msg):
|
|
||||||
app = current_app._get_current_object()
|
|
||||||
thread = Thread(target=send, args=(app, msg))
|
|
||||||
thread.start()
|
|
||||||
return thread
|
|
||||||
|
@ -1,9 +0,0 @@
|
|||||||
from ..models import Job
|
|
||||||
|
|
||||||
|
|
||||||
def delete_job_(app, job_id):
|
|
||||||
with app.app_context():
|
|
||||||
job = Job.query.get(job_id)
|
|
||||||
if job is None:
|
|
||||||
raise Exception('Job {} not found!'.format(job_id))
|
|
||||||
job.delete()
|
|
28
app/jobs/tasks.py
Normal file
28
app/jobs/tasks.py
Normal file
@ -0,0 +1,28 @@
|
|||||||
|
from time import sleep
|
||||||
|
from .. import db
|
||||||
|
from ..decorators import background
|
||||||
|
from ..models import Job
|
||||||
|
import os
|
||||||
|
import shutil
|
||||||
|
|
||||||
|
|
||||||
|
@background
|
||||||
|
def delete_job(app, job_id):
|
||||||
|
with app.app_context():
|
||||||
|
job = Job.query.get(job_id)
|
||||||
|
if job is None:
|
||||||
|
return
|
||||||
|
if job.status not in ['complete', 'failed']:
|
||||||
|
job.status = 'canceling'
|
||||||
|
db.session.commit()
|
||||||
|
while job.status != 'canceled':
|
||||||
|
# In case the daemon handled a job in any way
|
||||||
|
if job.status != 'canceling':
|
||||||
|
job.status = 'canceling'
|
||||||
|
db.session.commit()
|
||||||
|
sleep(1)
|
||||||
|
db.session.refresh(job)
|
||||||
|
path = os.path.join(app.config['NOPAQUE_STORAGE'], str(job.user_id),
|
||||||
|
'jobs', str(job.id))
|
||||||
|
shutil.rmtree(path, ignore_errors=True)
|
||||||
|
job.delete()
|
@ -1,9 +1,8 @@
|
|||||||
from flask import (abort, current_app, flash, redirect, render_template,
|
from flask import (abort, current_app, flash, redirect, render_template,
|
||||||
send_from_directory, url_for)
|
send_from_directory, url_for)
|
||||||
from flask_login import current_user, login_required
|
from flask_login import current_user, login_required
|
||||||
from threading import Thread
|
|
||||||
from . import jobs
|
from . import jobs
|
||||||
from .background_functions import delete_job_
|
from . import tasks
|
||||||
from ..models import Job, JobInput, JobResult
|
from ..models import Job, JobInput, JobResult
|
||||||
import os
|
import os
|
||||||
|
|
||||||
@ -23,9 +22,7 @@ def delete_job(job_id):
|
|||||||
job = Job.query.get_or_404(job_id)
|
job = Job.query.get_or_404(job_id)
|
||||||
if not (job.creator == current_user or current_user.is_administrator()):
|
if not (job.creator == current_user or current_user.is_administrator()):
|
||||||
abort(403)
|
abort(403)
|
||||||
thread = Thread(target=delete_job_,
|
tasks.delete_job(job_id)
|
||||||
args=(current_app._get_current_object(), job_id))
|
|
||||||
thread.start()
|
|
||||||
flash('Job has been deleted!')
|
flash('Job has been deleted!')
|
||||||
return redirect(url_for('main.dashboard'))
|
return redirect(url_for('main.dashboard'))
|
||||||
|
|
||||||
|
@ -2,7 +2,6 @@ from datetime import datetime
|
|||||||
from flask import current_app
|
from flask import current_app
|
||||||
from flask_login import UserMixin, AnonymousUserMixin
|
from flask_login import UserMixin, AnonymousUserMixin
|
||||||
from itsdangerous import BadSignature, TimedJSONWebSignatureSerializer
|
from itsdangerous import BadSignature, TimedJSONWebSignatureSerializer
|
||||||
from time import sleep
|
|
||||||
from werkzeug.security import generate_password_hash, check_password_hash
|
from werkzeug.security import generate_password_hash, check_password_hash
|
||||||
from werkzeug.utils import secure_filename
|
from werkzeug.utils import secure_filename
|
||||||
from . import db, logger, login_manager
|
from . import db, logger, login_manager
|
||||||
@ -326,26 +325,12 @@ class Job(db.Model):
|
|||||||
|
|
||||||
def delete(self):
|
def delete(self):
|
||||||
"""
|
"""
|
||||||
Delete the job and its inputs and outputs from database and filesystem.
|
Delete the job and its inputs and results from the database.
|
||||||
"""
|
"""
|
||||||
if self.status != 'complete' and self.status != 'failed':
|
for input in self.inputs:
|
||||||
self.status = 'canceling'
|
db.session.delete(input)
|
||||||
db.session.commit()
|
for result in self.results:
|
||||||
while self.status != 'canceled':
|
db.session.delete(result)
|
||||||
# In case the daemon handled a job in any way
|
|
||||||
if self.status != 'canceling':
|
|
||||||
self.status = 'canceling'
|
|
||||||
db.session.commit()
|
|
||||||
sleep(1)
|
|
||||||
db.session.refresh(self)
|
|
||||||
path = os.path.join(current_app.config['NOPAQUE_STORAGE'],
|
|
||||||
str(self.user_id), 'jobs', str(self.id))
|
|
||||||
try:
|
|
||||||
shutil.rmtree(path)
|
|
||||||
except Exception as e:
|
|
||||||
''' TODO: Proper exception handling '''
|
|
||||||
logger.warning(e)
|
|
||||||
pass
|
|
||||||
db.session.delete(self)
|
db.session.delete(self)
|
||||||
db.session.commit()
|
db.session.commit()
|
||||||
|
|
||||||
@ -391,14 +376,6 @@ class CorpusFile(db.Model):
|
|||||||
corpus_id = db.Column(db.Integer, db.ForeignKey('corpora.id'))
|
corpus_id = db.Column(db.Integer, db.ForeignKey('corpora.id'))
|
||||||
|
|
||||||
def delete(self):
|
def delete(self):
|
||||||
path = os.path.join(current_app.config['NOPAQUE_STORAGE'],
|
|
||||||
self.dir, self.filename)
|
|
||||||
try:
|
|
||||||
os.remove(path)
|
|
||||||
except Exception as e:
|
|
||||||
''' TODO: Proper exception handling '''
|
|
||||||
logger.warning(e)
|
|
||||||
pass
|
|
||||||
self.corpus.status = 'unprepared'
|
self.corpus.status = 'unprepared'
|
||||||
db.session.delete(self)
|
db.session.delete(self)
|
||||||
db.session.commit()
|
db.session.commit()
|
||||||
@ -460,12 +437,6 @@ class Corpus(db.Model):
|
|||||||
files = db.relationship('CorpusFile', backref='corpus', lazy='dynamic',
|
files = db.relationship('CorpusFile', backref='corpus', lazy='dynamic',
|
||||||
cascade='save-update, merge, delete')
|
cascade='save-update, merge, delete')
|
||||||
|
|
||||||
def __repr__(self):
|
|
||||||
"""
|
|
||||||
String representation of the corpus. For human readability.
|
|
||||||
"""
|
|
||||||
return '<Corpus %r>' % self.title
|
|
||||||
|
|
||||||
def to_dict(self):
|
def to_dict(self):
|
||||||
return {'id': self.id,
|
return {'id': self.id,
|
||||||
'creation_date': self.creation_date.timestamp(),
|
'creation_date': self.creation_date.timestamp(),
|
||||||
@ -475,22 +446,20 @@ class Corpus(db.Model):
|
|||||||
'title': self.title,
|
'title': self.title,
|
||||||
'user_id': self.user_id}
|
'user_id': self.user_id}
|
||||||
|
|
||||||
|
def build(self):
|
||||||
|
pass
|
||||||
|
|
||||||
def delete(self):
|
def delete(self):
|
||||||
for corpus_file in self.files:
|
for corpus_file in self.files:
|
||||||
corpus_file.delete()
|
db.session.delete(corpus_file)
|
||||||
path = os.path.join(current_app.config['NOPAQUE_STORAGE'],
|
|
||||||
str(self.user_id), 'corpora', str(self.id))
|
|
||||||
try:
|
|
||||||
shutil.rmtree(path)
|
|
||||||
except Exception as e:
|
|
||||||
''' TODO: Proper exception handling '''
|
|
||||||
logger.warning(e)
|
|
||||||
pass
|
|
||||||
db.session.delete(self)
|
db.session.delete(self)
|
||||||
db.session.commit()
|
db.session.commit()
|
||||||
|
|
||||||
def prepare(self):
|
def __repr__(self):
|
||||||
pass
|
"""
|
||||||
|
String representation of the corpus. For human readability.
|
||||||
|
"""
|
||||||
|
return '<Corpus %r>' % self.title
|
||||||
|
|
||||||
|
|
||||||
'''
|
'''
|
||||||
|
@ -9,6 +9,8 @@
|
|||||||
|
|
||||||
{% if field.type == 'BooleanField' %}
|
{% if field.type == 'BooleanField' %}
|
||||||
{{ render_boolean_field(field, *args, **kwargs) }}
|
{{ render_boolean_field(field, *args, **kwargs) }}
|
||||||
|
{% elif field.type == 'DecimalRangeField' %}
|
||||||
|
{{ render_decimal_range_field(field, *args, **kwargs) }}
|
||||||
{% elif field.type == 'IntegerField' %}
|
{% elif field.type == 'IntegerField' %}
|
||||||
{% set tmp = kwargs.update({'type': 'number'}) %}
|
{% set tmp = kwargs.update({'type': 'number'}) %}
|
||||||
{% if 'class_' in kwargs and 'validate' not in kwargs['class_'] %}
|
{% if 'class_' in kwargs and 'validate' not in kwargs['class_'] %}
|
||||||
@ -42,6 +44,12 @@
|
|||||||
</div>
|
</div>
|
||||||
{% endmacro %}
|
{% endmacro %}
|
||||||
|
|
||||||
|
{% macro render_decimal_range_field(field) %}
|
||||||
|
<p class="range-field">
|
||||||
|
{{ field(**kwargs) }}
|
||||||
|
</p>
|
||||||
|
{% endmacro %}
|
||||||
|
|
||||||
{% macro render_file_field(field) %}
|
{% macro render_file_field(field) %}
|
||||||
<div class="file-field input-field">
|
<div class="file-field input-field">
|
||||||
<div class="btn">
|
<div class="btn">
|
||||||
|
@ -9,4 +9,4 @@ GUNICORN_WORKERS="${GUNICORN_WORKERS:-1}"
|
|||||||
|
|
||||||
source venv/bin/activate
|
source venv/bin/activate
|
||||||
flask deploy
|
flask deploy
|
||||||
gunicorn --bind :5000 --workers "${GUNICORN_WORKERS}" --worker-class eventlet nopaque:app
|
gunicorn --access-logfile - --bind :5000 --error-logfile - --workers "${GUNICORN_WORKERS}" --worker-class eventlet nopaque:app
|
||||||
|
Loading…
Reference in New Issue
Block a user