2016-08-14 10:35:14 +00:00
|
|
|
import os
|
2017-01-07 10:59:43 +00:00
|
|
|
import time
|
2016-08-14 10:35:14 +00:00
|
|
|
import logging
|
2017-01-07 10:59:43 +00:00
|
|
|
import threading
|
2017-02-04 00:08:12 +00:00
|
|
|
from typing import Callable, Any, Type
|
|
|
|
|
2016-08-14 10:35:14 +00:00
|
|
|
import coloredlogs
|
2017-02-04 00:08:12 +00:00
|
|
|
import sqlalchemy as sa
|
2016-08-14 18:30:48 +00:00
|
|
|
import sqlalchemy.orm.exc
|
2017-02-02 18:45:18 +00:00
|
|
|
from szurubooru import config, db, errors, rest
|
2019-09-21 18:22:07 +00:00
|
|
|
from szurubooru.func import posts, file_uploads, image_hash
|
2016-08-14 10:35:14 +00:00
|
|
|
# pylint: disable=unused-import
|
|
|
|
from szurubooru import api, middleware
|
|
|
|
|
2016-08-14 12:22:53 +00:00
|
|
|
|
2017-02-04 00:08:12 +00:00
|
|
|
def _map_error(
|
|
|
|
ex: Exception,
|
|
|
|
target_class: Type[rest.errors.BaseHttpError],
|
|
|
|
title: str) -> rest.errors.BaseHttpError:
|
2016-09-10 08:14:40 +00:00
|
|
|
return target_class(
|
2016-09-10 08:37:28 +00:00
|
|
|
name=type(ex).__name__,
|
2016-09-10 08:14:40 +00:00
|
|
|
title=title,
|
|
|
|
description=str(ex),
|
|
|
|
extra_fields=getattr(ex, 'extra_fields', {}))
|
|
|
|
|
|
|
|
|
2017-02-04 00:08:12 +00:00
|
|
|
def _on_auth_error(ex: Exception) -> None:
|
2016-09-10 08:14:40 +00:00
|
|
|
raise _map_error(ex, rest.errors.HttpForbidden, 'Authentication error')
|
2016-08-14 10:35:14 +00:00
|
|
|
|
2016-08-14 12:22:53 +00:00
|
|
|
|
2017-02-04 00:08:12 +00:00
|
|
|
def _on_validation_error(ex: Exception) -> None:
|
2016-09-10 08:14:40 +00:00
|
|
|
raise _map_error(ex, rest.errors.HttpBadRequest, 'Validation error')
|
2016-08-14 10:35:14 +00:00
|
|
|
|
2016-08-14 12:22:53 +00:00
|
|
|
|
2017-02-04 00:08:12 +00:00
|
|
|
def _on_search_error(ex: Exception) -> None:
|
2016-09-10 08:14:40 +00:00
|
|
|
raise _map_error(ex, rest.errors.HttpBadRequest, 'Search error')
|
2016-08-14 10:35:14 +00:00
|
|
|
|
2016-08-14 12:22:53 +00:00
|
|
|
|
2017-02-04 00:08:12 +00:00
|
|
|
def _on_integrity_error(ex: Exception) -> None:
|
2016-09-10 08:14:40 +00:00
|
|
|
raise _map_error(ex, rest.errors.HttpConflict, 'Integrity violation')
|
2016-08-14 10:35:14 +00:00
|
|
|
|
2016-08-14 12:22:53 +00:00
|
|
|
|
2017-02-04 00:08:12 +00:00
|
|
|
def _on_not_found_error(ex: Exception) -> None:
|
2016-09-10 08:14:40 +00:00
|
|
|
raise _map_error(ex, rest.errors.HttpNotFound, 'Not found')
|
2016-08-14 10:35:14 +00:00
|
|
|
|
2016-08-14 12:22:53 +00:00
|
|
|
|
2017-02-04 00:08:12 +00:00
|
|
|
def _on_processing_error(ex: Exception) -> None:
|
2016-09-10 08:14:40 +00:00
|
|
|
raise _map_error(ex, rest.errors.HttpBadRequest, 'Processing error')
|
2016-08-14 10:35:14 +00:00
|
|
|
|
2016-08-14 12:22:53 +00:00
|
|
|
|
2017-02-04 00:08:12 +00:00
|
|
|
def _on_third_party_error(ex: Exception) -> None:
|
2017-02-02 17:21:21 +00:00
|
|
|
raise _map_error(
|
|
|
|
ex,
|
|
|
|
rest.errors.HttpInternalServerError,
|
|
|
|
'Server configuration error')
|
|
|
|
|
|
|
|
|
2017-02-04 00:08:12 +00:00
|
|
|
def _on_stale_data_error(_ex: Exception) -> None:
|
2016-08-14 18:30:48 +00:00
|
|
|
raise rest.errors.HttpConflict(
|
2016-09-10 08:37:28 +00:00
|
|
|
name='IntegrityError',
|
|
|
|
title='Integrity violation',
|
|
|
|
description=(
|
|
|
|
'Someone else modified this in the meantime. '
|
|
|
|
'Please try again.'))
|
2016-08-14 18:30:48 +00:00
|
|
|
|
|
|
|
|
2017-02-04 00:08:12 +00:00
|
|
|
def validate_config() -> None:
|
2016-08-14 10:35:14 +00:00
|
|
|
'''
|
|
|
|
Check whether config doesn't contain errors that might prove
|
|
|
|
lethal at runtime.
|
|
|
|
'''
|
|
|
|
from szurubooru.func.auth import RANK_MAP
|
|
|
|
for privilege, rank in config.config['privileges'].items():
|
|
|
|
if rank not in RANK_MAP.values():
|
|
|
|
raise errors.ConfigError(
|
|
|
|
'Rank %r for privilege %r is missing' % (rank, privilege))
|
|
|
|
if config.config['default_rank'] not in RANK_MAP.values():
|
|
|
|
raise errors.ConfigError(
|
|
|
|
'Default rank %r is not on the list of known ranks' % (
|
|
|
|
config.config['default_rank']))
|
|
|
|
|
2018-07-05 23:25:08 +00:00
|
|
|
for key in ['data_url', 'data_dir']:
|
2016-08-14 10:35:14 +00:00
|
|
|
if not config.config[key]:
|
|
|
|
raise errors.ConfigError(
|
|
|
|
'Service is not configured: %r is missing' % key)
|
|
|
|
|
|
|
|
if not os.path.isabs(config.config['data_dir']):
|
|
|
|
raise errors.ConfigError(
|
|
|
|
'data_dir must be an absolute path')
|
|
|
|
|
2016-08-15 17:55:03 +00:00
|
|
|
if not config.config['database']:
|
|
|
|
raise errors.ConfigError('Database is not configured')
|
2016-08-14 10:35:14 +00:00
|
|
|
|
2019-07-22 23:51:22 +00:00
|
|
|
if config.config['smtp']['host']:
|
|
|
|
if not config.config['smtp']['port']:
|
|
|
|
raise errors.ConfigError(
|
|
|
|
'SMTP host is set but port is not set')
|
|
|
|
if not config.config['smtp']['user']:
|
|
|
|
raise errors.ConfigError(
|
|
|
|
'SMTP host is set but username is not set')
|
|
|
|
if not config.config['smtp']['pass']:
|
|
|
|
raise errors.ConfigError(
|
|
|
|
'SMTP host is set but password is not set')
|
|
|
|
if not config.config['smtp']['from']:
|
|
|
|
raise errors.ConfigError(
|
|
|
|
'From address must be set to use mail-based password reset')
|
|
|
|
|
2016-08-14 12:22:53 +00:00
|
|
|
|
2017-02-04 00:08:12 +00:00
|
|
|
def purge_old_uploads() -> None:
|
2017-01-07 10:59:43 +00:00
|
|
|
while True:
|
|
|
|
try:
|
|
|
|
file_uploads.purge_old_uploads()
|
|
|
|
except Exception as ex:
|
|
|
|
logging.exception(ex)
|
|
|
|
time.sleep(60 * 5)
|
|
|
|
|
|
|
|
|
2017-02-04 00:08:12 +00:00
|
|
|
def create_app() -> Callable[[Any, Any], Any]:
|
2016-08-14 10:35:14 +00:00
|
|
|
''' Create a WSGI compatible App object. '''
|
|
|
|
validate_config()
|
|
|
|
coloredlogs.install(fmt='[%(asctime)-15s] %(name)s %(message)s')
|
2017-02-02 18:38:53 +00:00
|
|
|
logging.getLogger('elasticsearch').disabled = True
|
2016-08-14 10:35:14 +00:00
|
|
|
if config.config['debug']:
|
|
|
|
logging.getLogger('szurubooru').setLevel(logging.INFO)
|
|
|
|
if config.config['show_sql']:
|
|
|
|
logging.getLogger('sqlalchemy.engine').setLevel(logging.INFO)
|
|
|
|
|
2017-01-07 10:59:43 +00:00
|
|
|
purge_thread = threading.Thread(target=purge_old_uploads)
|
|
|
|
purge_thread.daemon = True
|
|
|
|
purge_thread.start()
|
2017-02-02 18:39:30 +00:00
|
|
|
|
|
|
|
try:
|
2019-09-21 18:22:07 +00:00
|
|
|
image_hash.get_session().cluster.health(
|
|
|
|
wait_for_status='yellow', request_timeout=30)
|
2017-02-02 18:39:30 +00:00
|
|
|
posts.populate_reverse_search()
|
2017-02-02 18:45:18 +00:00
|
|
|
db.session.commit()
|
2017-02-02 18:39:30 +00:00
|
|
|
except errors.ThirdPartyError:
|
|
|
|
pass
|
2016-11-27 17:42:14 +00:00
|
|
|
|
2016-08-14 10:35:14 +00:00
|
|
|
rest.errors.handle(errors.AuthError, _on_auth_error)
|
|
|
|
rest.errors.handle(errors.ValidationError, _on_validation_error)
|
|
|
|
rest.errors.handle(errors.SearchError, _on_search_error)
|
|
|
|
rest.errors.handle(errors.IntegrityError, _on_integrity_error)
|
|
|
|
rest.errors.handle(errors.NotFoundError, _on_not_found_error)
|
|
|
|
rest.errors.handle(errors.ProcessingError, _on_processing_error)
|
2017-02-02 17:21:21 +00:00
|
|
|
rest.errors.handle(errors.ThirdPartyError, _on_third_party_error)
|
2017-02-04 00:08:12 +00:00
|
|
|
rest.errors.handle(sa.orm.exc.StaleDataError, _on_stale_data_error)
|
2016-08-14 10:35:14 +00:00
|
|
|
|
|
|
|
return rest.application
|
2017-01-20 22:51:51 +00:00
|
|
|
|
|
|
|
|
2017-02-03 20:42:15 +00:00
|
|
|
app = create_app() # pylint: disable=invalid-name
|