191 lines
5.8 KiB
Python
191 lines
5.8 KiB
Python
from json import dumps
|
|
from typing import Dict, List, Optional, Union
|
|
|
|
from flask import Request, Response, current_app
|
|
from flask_sqlalchemy import SQLAlchemy
|
|
|
|
from fittrackee import appLog
|
|
from fittrackee.files import display_readable_file_size
|
|
|
|
|
|
def get_empty_data_for_datatype(data_type: str) -> Union[str, List]:
|
|
return '' if data_type in ['gpx', 'chart_data'] else []
|
|
|
|
|
|
class HttpResponse(Response):
|
|
def __init__(
|
|
self,
|
|
response: Optional[Union[str, Dict]] = None,
|
|
status_code: Optional[int] = None,
|
|
content_type: Optional[str] = None,
|
|
) -> None:
|
|
if isinstance(response, dict):
|
|
response = dumps(response)
|
|
content_type = (
|
|
'application/json' if content_type is None else content_type
|
|
)
|
|
super().__init__(
|
|
response=response,
|
|
status=status_code,
|
|
content_type=content_type,
|
|
)
|
|
|
|
|
|
class GenericErrorResponse(HttpResponse):
|
|
def __init__(
|
|
self,
|
|
status_code: int,
|
|
message: Union[str, List],
|
|
status: Optional[str] = None,
|
|
) -> None:
|
|
response = {
|
|
'status': 'error' if status is None else status,
|
|
'message': message,
|
|
}
|
|
super().__init__(
|
|
response=response,
|
|
status_code=status_code,
|
|
)
|
|
|
|
|
|
class InvalidPayloadErrorResponse(GenericErrorResponse):
|
|
def __init__(
|
|
self,
|
|
message: Optional[Union[str, List]] = None,
|
|
status: Optional[str] = None,
|
|
) -> None:
|
|
message = 'invalid payload' if message is None else message
|
|
super().__init__(status_code=400, message=message, status=status)
|
|
|
|
|
|
class DataInvalidPayloadErrorResponse(HttpResponse):
|
|
def __init__(self, data_type: str, status: Optional[str] = None) -> None:
|
|
response = {
|
|
'status': 'error' if status is None else status,
|
|
'data': {data_type: get_empty_data_for_datatype(data_type)},
|
|
}
|
|
super().__init__(response=response, status_code=400)
|
|
|
|
|
|
class UnauthorizedErrorResponse(GenericErrorResponse):
|
|
def __init__(self, message: Optional[str] = None) -> None:
|
|
message = (
|
|
'invalid token, please request a new token'
|
|
if message is None
|
|
else message
|
|
)
|
|
super().__init__(status_code=401, message=message)
|
|
|
|
|
|
class ForbiddenErrorResponse(GenericErrorResponse):
|
|
def __init__(self, message: Optional[str] = None) -> None:
|
|
message = 'you do not have permissions' if message is None else message
|
|
super().__init__(status_code=403, message=message)
|
|
|
|
|
|
class NotFoundErrorResponse(GenericErrorResponse):
|
|
def __init__(self, message: str) -> None:
|
|
super().__init__(status_code=404, message=message, status='not found')
|
|
|
|
|
|
class UserNotFoundErrorResponse(NotFoundErrorResponse):
|
|
def __init__(self) -> None:
|
|
super().__init__(message='user does not exist')
|
|
|
|
|
|
class DataNotFoundErrorResponse(HttpResponse):
|
|
def __init__(self, data_type: str, message: Optional[str] = None) -> None:
|
|
response = {
|
|
'status': 'not found',
|
|
'data': {data_type: get_empty_data_for_datatype(data_type)},
|
|
}
|
|
if message:
|
|
response['message'] = message
|
|
super().__init__(response=response, status_code=404)
|
|
|
|
|
|
class PayloadTooLargeErrorResponse(GenericErrorResponse):
|
|
def __init__(
|
|
self, file_type: str, file_size: Optional[int], max_size: Optional[int]
|
|
) -> None:
|
|
readable_file_size = (
|
|
f'({display_readable_file_size(file_size)}) ' if file_size else ''
|
|
)
|
|
readable_max_size = (
|
|
display_readable_file_size(max_size) if max_size else 'limit'
|
|
)
|
|
message = (
|
|
f'Error during {file_type} upload, file size {readable_file_size}'
|
|
f'exceeds {readable_max_size}.'
|
|
)
|
|
super().__init__(status_code=413, message=message, status='fail')
|
|
|
|
|
|
class InternalServerErrorResponse(GenericErrorResponse):
|
|
def __init__(
|
|
self, message: Optional[str] = None, status: Optional[str] = None
|
|
):
|
|
message = (
|
|
'error, please try again or contact the administrator'
|
|
if message is None
|
|
else message
|
|
)
|
|
super().__init__(status_code=500, message=message, status=status)
|
|
|
|
|
|
def handle_error_and_return_response(
|
|
error: Exception,
|
|
message: Optional[str] = None,
|
|
status: Optional[str] = None,
|
|
db: Optional[SQLAlchemy] = None,
|
|
) -> HttpResponse:
|
|
if db is not None:
|
|
db.session.rollback()
|
|
appLog.error(error)
|
|
return InternalServerErrorResponse(message=message, status=status)
|
|
|
|
|
|
def get_error_response_if_file_is_invalid(
|
|
file_type: str, req: Request
|
|
) -> Optional[HttpResponse]:
|
|
if 'file' not in req.files:
|
|
return InvalidPayloadErrorResponse('no file part', 'fail')
|
|
|
|
file = req.files['file']
|
|
if not file.filename or file.filename == '':
|
|
return InvalidPayloadErrorResponse('no selected file', 'fail')
|
|
|
|
allowed_extensions = (
|
|
'WORKOUT_ALLOWED_EXTENSIONS'
|
|
if file_type == 'workout'
|
|
else 'PICTURE_ALLOWED_EXTENSIONS'
|
|
)
|
|
|
|
file_extension = (
|
|
file.filename.rsplit('.', 1)[1].lower()
|
|
if '.' in file.filename
|
|
else None
|
|
)
|
|
max_file_size = current_app.config['max_single_file_size']
|
|
|
|
if not (
|
|
file_extension
|
|
and file_extension in current_app.config[allowed_extensions]
|
|
):
|
|
return InvalidPayloadErrorResponse(
|
|
'file extension not allowed', 'fail'
|
|
)
|
|
|
|
if (
|
|
file_extension != 'zip'
|
|
and req.content_length is not None
|
|
and req.content_length > max_file_size
|
|
):
|
|
return PayloadTooLargeErrorResponse(
|
|
file_type=file_type,
|
|
file_size=req.content_length,
|
|
max_size=max_file_size,
|
|
)
|
|
|
|
return None
|