mirror of
https://github.com/encounter/decomp.me.git
synced 2026-03-30 11:06:27 -07:00
4ae89872e2
* Organize tests, update settings, backend deps * Remove project import code from backend * Remove some frontent project code * more progress, project cleanup * more preset work * . * upd8z * Admins can delete scratches & preset migration * bye, source presets * fixes & changes * Preset in admin panel & rename migration * mypy * Fix perset viewset & tests * Change compilers hash files * test annotations so Windows doesn't run them * Mark nits * Preset auto ID field * scratch as reference (1/?) * backend fixes to preset (2/3) * Use preset ID instead of name in frontend * update frogger preset * I broke one of the ten commandments of react * finishing touches (3/3 lol) * Update serializers.py
96 lines
2.7 KiB
Python
96 lines
2.7 KiB
Python
from sqlite3 import IntegrityError
|
|
from subprocess import CalledProcessError
|
|
from typing import Any, ClassVar, Optional
|
|
|
|
from rest_framework.response import Response
|
|
from rest_framework.status import HTTP_400_BAD_REQUEST, HTTP_500_INTERNAL_SERVER_ERROR
|
|
|
|
from rest_framework.views import exception_handler
|
|
|
|
|
|
def custom_exception_handler(exc: Exception, context: Any) -> Optional[Response]:
|
|
# Call REST framework's default exception handler first,
|
|
# to get the standard error response.
|
|
response = exception_handler(exc, context)
|
|
|
|
if isinstance(exc, SubprocessError):
|
|
response = Response(
|
|
data={
|
|
"code": exc.SUBPROCESS_NAME,
|
|
"detail": exc.msg,
|
|
},
|
|
status=HTTP_400_BAD_REQUEST,
|
|
)
|
|
elif isinstance(exc, AssertionError) or isinstance(exc, IntegrityError):
|
|
response = Response(
|
|
data={
|
|
"detail": str(exc),
|
|
},
|
|
status=HTTP_500_INTERNAL_SERVER_ERROR,
|
|
)
|
|
|
|
if response is not None and isinstance(response.data, dict):
|
|
response.data["kind"] = exc.__class__.__name__
|
|
|
|
return response
|
|
|
|
|
|
class SubprocessError(Exception):
|
|
SUBPROCESS_NAME: ClassVar[str] = "Subprocess"
|
|
msg: str
|
|
stdout: str
|
|
stderr: str
|
|
|
|
def __init__(self, message: str):
|
|
self.msg = f"{self.SUBPROCESS_NAME} error: {message}"
|
|
|
|
super().__init__(self.msg)
|
|
self.stdout = ""
|
|
self.stderr = ""
|
|
|
|
@staticmethod
|
|
def from_process_error(ex: CalledProcessError) -> "SubprocessError":
|
|
error = SubprocessError(f"{ex.cmd[0]} returned {ex.returncode}")
|
|
error.stdout = ex.stdout
|
|
error.stderr = ex.stderr
|
|
error.msg = ex.stdout
|
|
return error
|
|
|
|
|
|
class DiffError(SubprocessError):
|
|
SUBPROCESS_NAME: ClassVar[str] = "Diff"
|
|
|
|
|
|
class ObjdumpError(SubprocessError):
|
|
SUBPROCESS_NAME: ClassVar[str] = "objdump"
|
|
|
|
|
|
class NmError(SubprocessError):
|
|
SUBPROCESS_NAME: ClassVar[str] = "nm"
|
|
|
|
|
|
class CompilationError(SubprocessError):
|
|
SUBPROCESS_NAME: ClassVar[str] = "Compiler"
|
|
|
|
|
|
class SandboxError(SubprocessError):
|
|
SUBPROCESS_NAME: ClassVar[str] = "Sandbox"
|
|
|
|
|
|
class AssemblyError(SubprocessError):
|
|
SUBPROCESS_NAME: ClassVar[str] = "Compiler"
|
|
|
|
@staticmethod
|
|
def from_process_error(ex: CalledProcessError) -> "SubprocessError":
|
|
error = super(AssemblyError, AssemblyError).from_process_error(ex)
|
|
|
|
error_lines = []
|
|
for line in ex.stdout.splitlines():
|
|
if "asm.s:" in line:
|
|
error_lines.append(line[line.find("asm.s:") + len("asm.s:") :].strip())
|
|
else:
|
|
error_lines.append(line)
|
|
error.msg = "\n".join(error_lines)
|
|
|
|
return error
|