mirror of
https://gitlab.ub.uni-bielefeld.de/sfb1288inf/nopaque.git
synced 2025-01-13 11:40:35 +00:00
Compare commits
3 Commits
492fdc9d28
...
5fc3015bf1
Author | SHA1 | Date | |
---|---|---|---|
|
5fc3015bf1 | ||
|
5f05cedf5e | ||
|
aabea234fe |
150
app/__init__.py
150
app/__init__.py
@ -2,6 +2,7 @@ from apifairy import APIFairy
|
|||||||
from config import Config
|
from config import Config
|
||||||
from docker import DockerClient
|
from docker import DockerClient
|
||||||
from flask import Flask
|
from flask import Flask
|
||||||
|
from flask.logging import default_handler
|
||||||
from flask_apscheduler import APScheduler
|
from flask_apscheduler import APScheduler
|
||||||
from flask_assets import Environment
|
from flask_assets import Environment
|
||||||
from flask_login import LoginManager
|
from flask_login import LoginManager
|
||||||
@ -12,11 +13,12 @@ from flask_paranoid import Paranoid
|
|||||||
from flask_socketio import SocketIO
|
from flask_socketio import SocketIO
|
||||||
from flask_sqlalchemy import SQLAlchemy
|
from flask_sqlalchemy import SQLAlchemy
|
||||||
from flask_hashids import Hashids
|
from flask_hashids import Hashids
|
||||||
|
from logging import Formatter, StreamHandler
|
||||||
|
from werkzeug.middleware.proxy_fix import ProxyFix
|
||||||
|
|
||||||
|
|
||||||
docker_client = DockerClient.from_env()
|
docker_client = DockerClient.from_env()
|
||||||
|
|
||||||
|
|
||||||
apifairy = APIFairy()
|
apifairy = APIFairy()
|
||||||
assets = Environment()
|
assets = Environment()
|
||||||
db = SQLAlchemy()
|
db = SQLAlchemy()
|
||||||
@ -36,82 +38,42 @@ def create_app(config: Config = Config) -> Flask:
|
|||||||
app = Flask(__name__)
|
app = Flask(__name__)
|
||||||
app.config.from_object(config)
|
app.config.from_object(config)
|
||||||
|
|
||||||
_configure_logging(app)
|
# region Logging
|
||||||
_configure_middlewares(app)
|
log_formatter = Formatter(
|
||||||
_init_docker_client(app)
|
fmt=app.config['NOPAQUE_LOG_FORMAT'],
|
||||||
_init_extensions(app)
|
datefmt=app.config['NOPAQUE_LOG_DATE_FORMAT']
|
||||||
_register_blueprints(app)
|
|
||||||
_register_socketio_namespaces(app)
|
|
||||||
_register_db_event_listeners(app)
|
|
||||||
|
|
||||||
@app.before_request
|
|
||||||
def log_headers():
|
|
||||||
from flask import request
|
|
||||||
print(request.__dict__)
|
|
||||||
|
|
||||||
return app
|
|
||||||
|
|
||||||
|
|
||||||
def _configure_logging(app: Flask):
|
|
||||||
from flask.logging import default_handler
|
|
||||||
from logging import Formatter, StreamHandler
|
|
||||||
|
|
||||||
log_date_format: str = app.config['NOPAQUE_LOG_DATE_FORMAT']
|
|
||||||
log_format: str = app.config['NOPAQUE_LOG_FORMAT']
|
|
||||||
log_level: str = app.config['NOPAQUE_LOG_LEVEL']
|
|
||||||
|
|
||||||
formatter = Formatter(fmt=log_format, datefmt=log_date_format)
|
|
||||||
handler = StreamHandler()
|
|
||||||
handler.setFormatter(formatter)
|
|
||||||
handler.setLevel(log_level)
|
|
||||||
|
|
||||||
app.logger.removeHandler(default_handler)
|
|
||||||
app.logger.addHandler(handler)
|
|
||||||
|
|
||||||
|
|
||||||
def _configure_middlewares(app: Flask):
|
|
||||||
proxy_fix_enabled: bool = app.config['NOPAQUE_PROXY_FIX_ENABLED']
|
|
||||||
|
|
||||||
if proxy_fix_enabled:
|
|
||||||
from werkzeug.middleware.proxy_fix import ProxyFix
|
|
||||||
|
|
||||||
proxy_fix_x_for: int = app.config['NOPAQUE_PROXY_FIX_X_FOR']
|
|
||||||
proxy_fix_x_host: int = app.config['NOPAQUE_PROXY_FIX_X_HOST']
|
|
||||||
proxy_fix_x_port: int = app.config['NOPAQUE_PROXY_FIX_X_PORT']
|
|
||||||
proxy_fix_x_prefix: int = app.config['NOPAQUE_PROXY_FIX_X_PREFIX']
|
|
||||||
proxy_fix_x_proto: int = app.config['NOPAQUE_PROXY_FIX_X_PROTO']
|
|
||||||
|
|
||||||
app.wsgi_app = ProxyFix(
|
|
||||||
app.wsgi_app,
|
|
||||||
x_for=proxy_fix_x_for,
|
|
||||||
x_host=proxy_fix_x_host,
|
|
||||||
x_port=proxy_fix_x_port,
|
|
||||||
x_prefix=proxy_fix_x_prefix,
|
|
||||||
x_proto=proxy_fix_x_proto
|
|
||||||
)
|
|
||||||
|
|
||||||
|
|
||||||
def _init_docker_client(app: Flask):
|
|
||||||
registry: str = app.config['NOPAQUE_DOCKER_REGISTRY']
|
|
||||||
username: str = app.config['NOPAQUE_DOCKER_REGISTRY_USERNAME']
|
|
||||||
password: str = app.config['NOPAQUE_DOCKER_REGISTRY_PASSWORD']
|
|
||||||
|
|
||||||
docker_client.login(
|
|
||||||
username,
|
|
||||||
password=password,
|
|
||||||
registry=registry
|
|
||||||
)
|
)
|
||||||
|
|
||||||
|
log_handler = StreamHandler()
|
||||||
|
log_handler.setFormatter(log_formatter)
|
||||||
|
log_handler.setLevel(app.config['NOPAQUE_LOG_LEVEL'])
|
||||||
|
|
||||||
|
app.logger.setLevel('DEBUG')
|
||||||
|
app.logger.removeHandler(default_handler)
|
||||||
|
app.logger.addHandler(log_handler)
|
||||||
|
# endregion Logging
|
||||||
|
|
||||||
|
# region Middlewares
|
||||||
|
if app.config['NOPAQUE_PROXY_FIX_ENABLED']:
|
||||||
|
app.wsgi_app = ProxyFix(
|
||||||
|
app.wsgi_app,
|
||||||
|
x_for=app.config['NOPAQUE_PROXY_FIX_X_FOR'],
|
||||||
|
x_host=app.config['NOPAQUE_PROXY_FIX_X_HOST'],
|
||||||
|
x_port=app.config['NOPAQUE_PROXY_FIX_X_PORT'],
|
||||||
|
x_prefix=app.config['NOPAQUE_PROXY_FIX_X_PREFIX'],
|
||||||
|
x_proto=app.config['NOPAQUE_PROXY_FIX_X_PROTO']
|
||||||
|
)
|
||||||
|
# endregion Middlewares
|
||||||
|
|
||||||
|
# region Extensions
|
||||||
|
docker_client.login(
|
||||||
|
app.config['NOPAQUE_DOCKER_REGISTRY_USERNAME'],
|
||||||
|
password=app.config['NOPAQUE_DOCKER_REGISTRY_PASSWORD'],
|
||||||
|
registry=app.config['NOPAQUE_DOCKER_REGISTRY']
|
||||||
|
)
|
||||||
|
|
||||||
def _init_extensions(app: Flask):
|
|
||||||
from typing import Callable
|
|
||||||
from .daemon import daemon
|
|
||||||
from .models import AnonymousUser, User
|
from .models import AnonymousUser, User
|
||||||
|
|
||||||
is_primary_instance: bool = app.config['NOPAQUE_IS_PRIMARY_INSTANCE']
|
|
||||||
socketio_message_queue_uri: str = app.config['NOPAQUE_SOCKETIO_MESSAGE_QUEUE_URI']
|
|
||||||
login_user_loader_callback: Callable[[int], User | None] = lambda user_id: User.query.get(int(user_id))
|
|
||||||
|
|
||||||
apifairy.init_app(app)
|
apifairy.init_app(app)
|
||||||
assets.init_app(app)
|
assets.init_app(app)
|
||||||
db.init_app(app)
|
db.init_app(app)
|
||||||
@ -119,19 +81,17 @@ def _init_extensions(app: Flask):
|
|||||||
login.init_app(app)
|
login.init_app(app)
|
||||||
login.anonymous_user = AnonymousUser
|
login.anonymous_user = AnonymousUser
|
||||||
login.login_view = 'auth.login'
|
login.login_view = 'auth.login'
|
||||||
login.user_loader(login_user_loader_callback)
|
login.user_loader(lambda user_id: User.query.get(int(user_id)))
|
||||||
ma.init_app(app)
|
ma.init_app(app)
|
||||||
mail.init_app(app)
|
mail.init_app(app)
|
||||||
migrate.init_app(app, db)
|
migrate.init_app(app, db)
|
||||||
paranoid.init_app(app)
|
paranoid.init_app(app)
|
||||||
paranoid.redirect_view = '/'
|
paranoid.redirect_view = '/'
|
||||||
scheduler.init_app(app)
|
scheduler.init_app(app)
|
||||||
if is_primary_instance:
|
socketio.init_app(app, message_queue=app.config['NOPAQUE_SOCKETIO_MESSAGE_QUEUE_URI'])
|
||||||
scheduler.add_job('daemon', daemon, args=(app,), seconds=3, trigger='interval')
|
# endregion Extensions
|
||||||
socketio.init_app(app, message_queue=socketio_message_queue_uri)
|
|
||||||
|
|
||||||
|
# region Blueprints
|
||||||
def _register_blueprints(app: Flask):
|
|
||||||
from .admin import bp as admin_blueprint
|
from .admin import bp as admin_blueprint
|
||||||
app.register_blueprint(admin_blueprint, url_prefix='/admin')
|
app.register_blueprint(admin_blueprint, url_prefix='/admin')
|
||||||
|
|
||||||
@ -167,6 +127,28 @@ def _register_blueprints(app: Flask):
|
|||||||
|
|
||||||
from .workshops import bp as workshops_blueprint
|
from .workshops import bp as workshops_blueprint
|
||||||
app.register_blueprint(workshops_blueprint, url_prefix='/workshops')
|
app.register_blueprint(workshops_blueprint, url_prefix='/workshops')
|
||||||
|
# endregion Blueprints
|
||||||
|
|
||||||
|
# region SocketIO Namespaces
|
||||||
|
from .corpora.cqi_over_sio import CQiOverSocketIO
|
||||||
|
socketio.on_namespace(CQiOverSocketIO('/cqi_over_sio'))
|
||||||
|
# endregion SocketIO Namespaces
|
||||||
|
|
||||||
|
# region Database event Listeners
|
||||||
|
from .models.event_listeners import register_event_listeners
|
||||||
|
register_event_listeners()
|
||||||
|
# endregion Database event Listeners
|
||||||
|
|
||||||
|
# region Add scheduler jobs
|
||||||
|
if app.config['NOPAQUE_IS_PRIMARY_INSTANCE']:
|
||||||
|
from .tasks import handle_corpora
|
||||||
|
scheduler.add_job('handle_corpora', handle_corpora, seconds=3, trigger='interval')
|
||||||
|
|
||||||
|
from .tasks import handle_jobs
|
||||||
|
scheduler.add_job('handle_jobs', handle_jobs, seconds=3, trigger='interval')
|
||||||
|
# endregion Add scheduler jobs
|
||||||
|
|
||||||
|
return app
|
||||||
|
|
||||||
|
|
||||||
# def _add_admin_views():
|
# def _add_admin_views():
|
||||||
@ -181,15 +163,3 @@ def _register_blueprints(app: Flask):
|
|||||||
# if not issubclass(v, db.Model):
|
# if not issubclass(v, db.Model):
|
||||||
# continue
|
# continue
|
||||||
# admin.add_view(ModelView(v, db.session, category='Database'))
|
# admin.add_view(ModelView(v, db.session, category='Database'))
|
||||||
|
|
||||||
|
|
||||||
def _register_socketio_namespaces(app: Flask):
|
|
||||||
from .corpora.cqi_over_sio import CQiOverSocketIO
|
|
||||||
|
|
||||||
socketio.on_namespace(CQiOverSocketIO('/cqi_over_sio'))
|
|
||||||
|
|
||||||
|
|
||||||
def _register_db_event_listeners(app: Flask):
|
|
||||||
from .models.event_listeners import register_event_listeners
|
|
||||||
|
|
||||||
register_event_listeners()
|
|
||||||
|
@ -1,11 +0,0 @@
|
|||||||
from flask import Flask
|
|
||||||
from app import db
|
|
||||||
from .corpus_utils import check_corpora
|
|
||||||
from .job_utils import check_jobs
|
|
||||||
|
|
||||||
|
|
||||||
def daemon(app: Flask):
|
|
||||||
with app.app_context():
|
|
||||||
check_corpora()
|
|
||||||
check_jobs()
|
|
||||||
db.session.commit()
|
|
2
app/tasks/__init__.py
Normal file
2
app/tasks/__init__.py
Normal file
@ -0,0 +1,2 @@
|
|||||||
|
from .handle_corpora import task as handle_corpora
|
||||||
|
from .handle_jobs import task as handle_jobs
|
@ -1,4 +1,4 @@
|
|||||||
from app import docker_client
|
from app import db, docker_client, scheduler
|
||||||
from app.models import Corpus, CorpusStatus
|
from app.models import Corpus, CorpusStatus
|
||||||
from flask import current_app
|
from flask import current_app
|
||||||
import docker
|
import docker
|
||||||
@ -6,7 +6,11 @@ import os
|
|||||||
import shutil
|
import shutil
|
||||||
|
|
||||||
|
|
||||||
def check_corpora():
|
def task():
|
||||||
|
with scheduler.app.app_context():
|
||||||
|
_handle_corpora()
|
||||||
|
|
||||||
|
def _handle_corpora():
|
||||||
corpora = Corpus.query.all()
|
corpora = Corpus.query.all()
|
||||||
for corpus in [x for x in corpora if x.status == CorpusStatus.SUBMITTED]:
|
for corpus in [x for x in corpora if x.status == CorpusStatus.SUBMITTED]:
|
||||||
_create_build_corpus_service(corpus)
|
_create_build_corpus_service(corpus)
|
||||||
@ -22,6 +26,7 @@ def check_corpora():
|
|||||||
_create_cqpserver_container(corpus)
|
_create_cqpserver_container(corpus)
|
||||||
for corpus in [x for x in corpora if x.status == CorpusStatus.CANCELING_ANALYSIS_SESSION]:
|
for corpus in [x for x in corpora if x.status == CorpusStatus.CANCELING_ANALYSIS_SESSION]:
|
||||||
_remove_cqpserver_container(corpus)
|
_remove_cqpserver_container(corpus)
|
||||||
|
db.session.commit()
|
||||||
|
|
||||||
def _create_build_corpus_service(corpus):
|
def _create_build_corpus_service(corpus):
|
||||||
''' # Docker service settings # '''
|
''' # Docker service settings # '''
|
@ -1,4 +1,4 @@
|
|||||||
from app import db, docker_client, hashids
|
from app import db, docker_client, hashids, scheduler
|
||||||
from app.models import (
|
from app.models import (
|
||||||
Job,
|
Job,
|
||||||
JobResult,
|
JobResult,
|
||||||
@ -15,7 +15,11 @@ import os
|
|||||||
import shutil
|
import shutil
|
||||||
|
|
||||||
|
|
||||||
def check_jobs():
|
def task():
|
||||||
|
with scheduler.app.app_context():
|
||||||
|
_handle_jobs()
|
||||||
|
|
||||||
|
def _handle_jobs():
|
||||||
jobs = Job.query.all()
|
jobs = Job.query.all()
|
||||||
for job in [x for x in jobs if x.status == JobStatus.SUBMITTED]:
|
for job in [x for x in jobs if x.status == JobStatus.SUBMITTED]:
|
||||||
_create_job_service(job)
|
_create_job_service(job)
|
||||||
@ -23,6 +27,7 @@ def check_jobs():
|
|||||||
_checkout_job_service(job)
|
_checkout_job_service(job)
|
||||||
for job in [x for x in jobs if x.status == JobStatus.CANCELING]:
|
for job in [x for x in jobs if x.status == JobStatus.CANCELING]:
|
||||||
_remove_job_service(job)
|
_remove_job_service(job)
|
||||||
|
db.session.commit()
|
||||||
|
|
||||||
def _create_job_service(job):
|
def _create_job_service(job):
|
||||||
''' # Docker service settings # '''
|
''' # Docker service settings # '''
|
17
config.py
17
config.py
@ -15,7 +15,7 @@ class Config:
|
|||||||
''' Configuration class for the Flask application. '''
|
''' Configuration class for the Flask application. '''
|
||||||
|
|
||||||
# region APIFairy
|
# region APIFairy
|
||||||
APIFAIRY_TITLE = 'nopaque'
|
APIFAIRY_TITLE = 'nopaque API'
|
||||||
APIFAIRY_VERSION = '0.0.1'
|
APIFAIRY_VERSION = '0.0.1'
|
||||||
APIFAIRY_APISPEC_PATH = '/api/apispec.json'
|
APIFAIRY_APISPEC_PATH = '/api/apispec.json'
|
||||||
APIFAIRY_UI = 'swagger_ui'
|
APIFAIRY_UI = 'swagger_ui'
|
||||||
@ -60,10 +60,7 @@ class Config:
|
|||||||
|
|
||||||
|
|
||||||
# region Flask-SQLAlchemy
|
# region Flask-SQLAlchemy
|
||||||
SQLALCHEMY_DATABASE_URI = os.environ.get(
|
SQLALCHEMY_DATABASE_URI = os.environ.get('SQLALCHEMY_DATABASE_URI', f'sqlite:///{BASE_DIR}/data.sqlite')
|
||||||
'SQLALCHEMY_DATABASE_URI',
|
|
||||||
f'sqlite:///{BASE_DIR}/data.sqlite'
|
|
||||||
)
|
|
||||||
SQLALCHEMY_RECORD_QUERIES = True
|
SQLALCHEMY_RECORD_QUERIES = True
|
||||||
SQLALCHEMY_TRACK_MODIFICATIONS = False
|
SQLALCHEMY_TRACK_MODIFICATIONS = False
|
||||||
# endregion Flask-SQLAlchemy
|
# endregion Flask-SQLAlchemy
|
||||||
@ -83,14 +80,8 @@ class Config:
|
|||||||
NOPAQUE_DOCKER_REGISTRY_USERNAME = os.environ.get('NOPAQUE_DOCKER_REGISTRY_USERNAME')
|
NOPAQUE_DOCKER_REGISTRY_USERNAME = os.environ.get('NOPAQUE_DOCKER_REGISTRY_USERNAME')
|
||||||
NOPAQUE_DOCKER_REGISTRY_PASSWORD = os.environ.get('NOPAQUE_DOCKER_REGISTRY_PASSWORD')
|
NOPAQUE_DOCKER_REGISTRY_PASSWORD = os.environ.get('NOPAQUE_DOCKER_REGISTRY_PASSWORD')
|
||||||
|
|
||||||
NOPAQUE_LOG_DATE_FORMAT = os.environ.get(
|
NOPAQUE_LOG_DATE_FORMAT = os.environ.get('NOPAQUE_LOG_DATE_FORMAT', '%Y-%m-%d %H:%M:%S')
|
||||||
'NOPAQUE_LOG_DATE_FORMAT',
|
NOPAQUE_LOG_FORMAT = os.environ.get('NOPAQUE_LOG_FORMAT','[%(asctime)s] %(levelname)s: %(message)s')
|
||||||
'%Y-%m-%d %H:%M:%S'
|
|
||||||
)
|
|
||||||
NOPAQUE_LOG_FORMAT = os.environ.get(
|
|
||||||
'NOPAQUE_LOG_DATE_FORMAT',
|
|
||||||
'[%(asctime)s] %(levelname)s in %(pathname)s (function: %(funcName)s, line: %(lineno)d): %(message)s'
|
|
||||||
)
|
|
||||||
NOPAQUE_LOG_LEVEL = os.environ.get('NOPAQUE_LOG_LEVEL', 'WARNING')
|
NOPAQUE_LOG_LEVEL = os.environ.get('NOPAQUE_LOG_LEVEL', 'WARNING')
|
||||||
|
|
||||||
NOPAQUE_PROXY_FIX_ENABLED = os.environ.get('NOPAQUE_PROXY_FIX_ENABLED', 'false').lower() == 'true'
|
NOPAQUE_PROXY_FIX_ENABLED = os.environ.get('NOPAQUE_PROXY_FIX_ENABLED', 'false').lower() == 'true'
|
||||||
|
@ -7,6 +7,10 @@
|
|||||||
# Flask #
|
# Flask #
|
||||||
# https://flask.palletsprojects.com/en/1.1.x/config/ #
|
# https://flask.palletsprojects.com/en/1.1.x/config/ #
|
||||||
##############################################################################
|
##############################################################################
|
||||||
|
# CHOOSE ONE: False, True
|
||||||
|
# DEFAULT: False
|
||||||
|
# FLASK_DEBUG=
|
||||||
|
|
||||||
# CHOOSE ONE: http, https
|
# CHOOSE ONE: http, https
|
||||||
# DEFAULT: http
|
# DEFAULT: http
|
||||||
# PREFERRED_URL_SCHEME=
|
# PREFERRED_URL_SCHEME=
|
||||||
@ -138,7 +142,7 @@ NOPAQUE_DOCKER_REGISTRY_PASSWORD=
|
|||||||
# DEFAULT: [%(asctime)s] %(levelname)s in %(pathname)s (function: %(funcName)s, line: %(lineno)d): %(message)s
|
# DEFAULT: [%(asctime)s] %(levelname)s in %(pathname)s (function: %(funcName)s, line: %(lineno)d): %(message)s
|
||||||
# NOPAQUE_LOG_FORMAT=
|
# NOPAQUE_LOG_FORMAT=
|
||||||
|
|
||||||
# DEFAULT: DEBUG if FLASK_DEBUG == True else WARNING
|
# DEFAULT: WARNING
|
||||||
# CHOOSE ONE: CRITICAL, ERROR, WARNING, INFO, DEBUG
|
# CHOOSE ONE: CRITICAL, ERROR, WARNING, INFO, DEBUG
|
||||||
# NOPAQUE_LOG_LEVEL=
|
# NOPAQUE_LOG_LEVEL=
|
||||||
|
|
||||||
|
Loading…
x
Reference in New Issue
Block a user