2024-05-04 14:55:05 +02:00
|
|
|
from flask import abort, request
|
2019-07-09 15:41:16 +02:00
|
|
|
from flask_login import current_user
|
2020-03-27 12:06:11 +01:00
|
|
|
from functools import wraps
|
2023-03-06 15:02:46 +01:00
|
|
|
from typing import List, Union
|
|
|
|
from werkzeug.exceptions import NotAcceptable
|
2022-09-02 13:02:04 +02:00
|
|
|
from app.models import Permission
|
2021-12-03 14:07:03 +01:00
|
|
|
|
|
|
|
|
|
|
|
def permission_required(permission):
|
|
|
|
def decorator(f):
|
|
|
|
@wraps(f)
|
|
|
|
def decorated_function(*args, **kwargs):
|
|
|
|
if not current_user.can(permission):
|
|
|
|
abort(403)
|
|
|
|
return f(*args, **kwargs)
|
|
|
|
return decorated_function
|
|
|
|
return decorator
|
2019-07-09 15:41:16 +02:00
|
|
|
|
|
|
|
|
2020-03-26 16:14:09 +01:00
|
|
|
def admin_required(f):
|
2021-12-03 14:07:03 +01:00
|
|
|
return permission_required(Permission.ADMINISTRATE)(f)
|
|
|
|
|
|
|
|
|
2024-05-04 14:55:05 +02:00
|
|
|
def socketio_login_required(f):
|
2020-03-26 16:14:09 +01:00
|
|
|
@wraps(f)
|
2024-05-04 14:55:05 +02:00
|
|
|
def wrapper(*args, **kwargs):
|
|
|
|
if current_user.is_authenticated:
|
|
|
|
return f(*args, **kwargs)
|
|
|
|
return {'code': 401, 'body': 'Unauthorized'}
|
|
|
|
return wrapper
|
|
|
|
|
|
|
|
|
|
|
|
def socketio_permission_required(permission):
|
|
|
|
def decorator(f):
|
|
|
|
@wraps(f)
|
|
|
|
def wrapper(*args, **kwargs):
|
|
|
|
if not current_user.can(permission):
|
|
|
|
return {'code': 403, 'body': 'Forbidden'}
|
|
|
|
return f(*args, **kwargs)
|
|
|
|
return wrapper
|
|
|
|
return decorator
|
|
|
|
|
|
|
|
|
|
|
|
def socketio_admin_required(f):
|
|
|
|
return socketio_permission_required(Permission.ADMINISTRATE)(f)
|
2023-03-06 15:02:46 +01:00
|
|
|
|
|
|
|
|
|
|
|
def content_negotiation(
|
2023-03-07 16:32:15 +01:00
|
|
|
produces: Union[str, List[str], None] = None,
|
|
|
|
consumes: Union[str, List[str], None] = None
|
2023-03-06 15:02:46 +01:00
|
|
|
):
|
|
|
|
def decorator(f):
|
|
|
|
@wraps(f)
|
|
|
|
def decorated_function(*args, **kwargs):
|
|
|
|
provided = request.mimetype
|
2023-03-07 16:32:15 +01:00
|
|
|
if consumes is None:
|
|
|
|
consumeables = None
|
|
|
|
elif isinstance(consumes, str):
|
2023-03-06 15:02:46 +01:00
|
|
|
consumeables = {consumes}
|
2023-03-07 16:32:15 +01:00
|
|
|
elif isinstance(consumes, list) and all(isinstance(x, str) for x in consumes):
|
2023-03-06 15:02:46 +01:00
|
|
|
consumeables = {*consumes}
|
2023-03-07 16:32:15 +01:00
|
|
|
else:
|
|
|
|
raise TypeError()
|
2023-03-06 15:02:46 +01:00
|
|
|
accepted = {*request.accept_mimetypes.values()}
|
2023-03-07 16:32:15 +01:00
|
|
|
if produces is None:
|
|
|
|
produceables = None
|
|
|
|
elif isinstance(produces, str):
|
2023-03-06 15:02:46 +01:00
|
|
|
produceables = {produces}
|
2023-03-07 16:32:15 +01:00
|
|
|
elif isinstance(produces, list) and all(isinstance(x, str) for x in produces):
|
2023-03-06 15:02:46 +01:00
|
|
|
produceables = {*produces}
|
2023-03-07 16:32:15 +01:00
|
|
|
else:
|
|
|
|
raise TypeError()
|
|
|
|
if produceables is not None and len(produceables & accepted) == 0:
|
2023-03-06 15:02:46 +01:00
|
|
|
raise NotAcceptable()
|
2023-03-07 16:32:15 +01:00
|
|
|
if consumeables is not None and provided not in consumeables:
|
2023-03-06 15:02:46 +01:00
|
|
|
raise NotAcceptable()
|
|
|
|
return f(*args, **kwargs)
|
|
|
|
return decorated_function
|
|
|
|
return decorator
|