Files
romm/backend/endpoints/states.py
Georges-Antoine Assi 1840390c8a finish mypy fixes
2024-05-21 21:28:17 -04:00

152 lines
5.4 KiB
Python

from decorators.auth import protected_route
from endpoints.responses import MessageResponse
from endpoints.responses.assets import StateSchema, UploadedStatesResponse
from fastapi import APIRouter, HTTPException, Request, UploadFile, status
from handler.database import db_rom_handler, db_screenshot_handler, db_state_handler
from handler.filesystem import fs_asset_handler
from handler.scan_handler import scan_state
from logger.logger import log
router = APIRouter()
@protected_route(router.post, "/states", ["assets.write"])
def add_states(
request: Request,
rom_id: int,
states: list[UploadFile] | None = None,
emulator: str | None = None,
) -> UploadedStatesResponse:
rom = db_rom_handler.get_roms(rom_id)
current_user = request.user
log.info(f"Uploading states to {rom.name}")
if states is None:
log.error("No states were uploaded")
raise HTTPException(
status_code=status.HTTP_500_INTERNAL_SERVER_ERROR,
detail="No states were uploaded",
)
states_path = fs_asset_handler.build_states_file_path(
user=request.user, platform_fs_slug=rom.platform.fs_slug, emulator=emulator
)
for state in states:
fs_asset_handler.write_file(file=state, path=states_path)
# Scan or update state
scanned_state = scan_state(
file_name=state.filename,
user=request.user,
platform_fs_slug=rom.platform.fs_slug,
emulator=emulator,
)
db_state = db_state_handler.get_state_by_filename(
rom_id=rom.id, user_id=current_user.id, file_name=state.filename
)
if db_state:
db_state_handler.update_state(
db_state.id, {"file_size_bytes": scanned_state.file_size_bytes}
)
continue
scanned_state.rom_id = rom.id
scanned_state.user_id = current_user.id
scanned_state.emulator = emulator
db_state_handler.add_state(scanned_state)
rom = db_rom_handler.get_roms(rom_id)
return {
"uploaded": len(states),
"states": [s for s in rom.states if s.user_id == current_user.id],
}
# @protected_route(router.get, "/states", ["assets.read"])
# def get_states(request: Request) -> MessageResponse:
# pass
# @protected_route(router.get, "/states/{id}", ["assets.read"])
# def get_state(request: Request, id: int) -> MessageResponse:
# pass
@protected_route(router.put, "/states/{id}", ["assets.write"])
async def update_state(request: Request, id: int) -> StateSchema:
data = await request.form()
db_state = db_state_handler.get_state(id)
if not db_state:
error = f"Save with ID {id} not found"
log.error(error)
raise HTTPException(status_code=status.HTTP_404_NOT_FOUND, detail=error)
if db_state.user_id != request.user.id:
error = "You are not authorized to update this save state"
log.error(error)
raise HTTPException(status_code=status.HTTP_403_FORBIDDEN, detail=error)
if "file" in data:
file: UploadFile = data["file"]
fs_asset_handler.write_file(file=file, path=db_state.file_path)
db_state_handler.update_state(db_state.id, {"file_size_bytes": file.size})
db_state = db_state_handler.get_state(id)
return db_state
@protected_route(router.post, "/states/delete", ["assets.write"])
async def delete_states(request: Request) -> MessageResponse:
data: dict = await request.json()
state_ids: list = data["states"]
delete_from_fs: bool = data["delete_from_fs"]
if not state_ids:
error = "No states were provided"
log.error(error)
raise HTTPException(status_code=status.HTTP_400_BAD_REQUEST, detail=error)
for state_id in state_ids:
state = db_state_handler.get_state(state_id)
if not state:
error = f"Save with ID {state_id} not found"
log.error(error)
raise HTTPException(status_code=status.HTTP_404_NOT_FOUND, detail=error)
if state.user_id != request.user.id:
error = "You are not authorized to delete this save state"
log.error(error)
raise HTTPException(status_code=status.HTTP_403_FORBIDDEN, detail=error)
db_state_handler.delete_state(state_id)
if delete_from_fs:
log.info(f"Deleting {state.file_name} from filesystem")
try:
fs_asset_handler.remove_file(
file_name=state.file_name, file_path=state.file_path
)
except FileNotFoundError as exc:
error = f"Save file {state.file_name} not found for platform {state.rom.platform_slug}"
log.error(error)
raise HTTPException(
status_code=status.HTTP_404_NOT_FOUND, detail=error
) from exc
if state.screenshot:
db_screenshot_handler.delete_screenshot(state.screenshot.id)
if delete_from_fs:
try:
fs_asset_handler.remove_file(
file_name=state.screenshot.file_name,
file_path=state.screenshot.file_path,
)
except FileNotFoundError:
error = f"Screenshot file {state.screenshot.file_name} not found for state {state.file_name}"
log.error(error)
return {"msg": f"Successfully deleted {len(state_ids)} states"}