mirror of
https://github.com/ruvnet/RuView
synced 2026-06-17 11:33:19 +00:00
81cc241b9e
The Rust port at v2/ has been the primary codebase since the rename in #427. The Python implementation at v1/ is no longer the active target; the only load-bearing path is the deterministic proof bundle at v1/data/proof/ (per ADR-011 / ADR-028 witness verification). Move the whole Python tree into archive/v1/ and document the policy in archive/README.md: no new features, bug fixes only when they affect a still-load-bearing path (currently just the proof), CI continues to verify the proof on every push and PR. Path references updated in 26 files via path-pattern sed (only matches v1/<known-child> patterns, never bare v1 or API URLs like /api/v1/). Two double-prefix typos (archive/archive/v1/) caught and hand-fixed in verify-pipeline.yml and ADR-011. Validated: - Python proof verify.py imports cleanly at archive/v1/data/proof/ (numpy/scipy still required; CI installs requirements-lock.txt from archive/v1/ now) - cargo test --workspace --no-default-features → 1,539 passed, 0 failed, 8 ignored (unaffected by Python tree relocation) - ESP32-S3 on COM7 untouched (no firmware paths changed) After-merge: contributors should re-run any local `python v1/...` commands as `python archive/v1/...` (CLAUDE.md and CHANGELOG already updated).
79 lines
2.9 KiB
Python
79 lines
2.9 KiB
Python
"""Tests for error handling in the API layer."""
|
|
|
|
import pytest
|
|
from unittest.mock import MagicMock, patch
|
|
from fastapi.testclient import TestClient
|
|
|
|
|
|
class TestExceptionHandlers:
|
|
"""Test the exception handlers registered on the FastAPI app."""
|
|
|
|
def _get_app(self):
|
|
"""Import app lazily to avoid side effects."""
|
|
with patch("src.api.main.get_settings") as mock_gs, \
|
|
patch("src.api.main.get_domain_config") as mock_gdc, \
|
|
patch("src.api.main.get_pose_service") as mock_ps, \
|
|
patch("src.api.main.get_stream_service") as mock_ss, \
|
|
patch("src.api.main.get_hardware_service") as mock_hs, \
|
|
patch("src.api.main.connection_manager") as mock_cm, \
|
|
patch("src.api.main.PoseStreamHandler") as mock_psh:
|
|
mock_gs.return_value = MagicMock(
|
|
app_name="test", version="0.1", environment="test",
|
|
is_production=False, enable_rate_limiting=False,
|
|
enable_authentication=False, docs_url="/docs",
|
|
redoc_url="/redoc", openapi_url="/openapi.json",
|
|
api_prefix="/api/v1",
|
|
)
|
|
mock_gs.return_value.get_logging_config.return_value = {
|
|
"version": 1, "disable_existing_loggers": False,
|
|
"handlers": {}, "loggers": {},
|
|
}
|
|
mock_gs.return_value.get_cors_config.return_value = {
|
|
"allow_origins": ["*"], "allow_methods": ["*"],
|
|
"allow_headers": ["*"],
|
|
}
|
|
# Re-import to pick up patches
|
|
import importlib
|
|
import src.api.main as m
|
|
importlib.reload(m)
|
|
return m.app
|
|
|
|
|
|
class TestErrorResponseModel:
|
|
def test_error_json_structure(self):
|
|
"""Verify error JSON has code, message, type fields."""
|
|
error = {
|
|
"error": {
|
|
"code": 404,
|
|
"message": "Not found",
|
|
"type": "http_error"
|
|
}
|
|
}
|
|
assert error["error"]["code"] == 404
|
|
assert "message" in error["error"]
|
|
assert "type" in error["error"]
|
|
|
|
def test_validation_error_structure(self):
|
|
error = {
|
|
"error": {
|
|
"code": 422,
|
|
"message": "Validation error",
|
|
"type": "validation_error",
|
|
"details": []
|
|
}
|
|
}
|
|
assert error["error"]["type"] == "validation_error"
|
|
assert isinstance(error["error"]["details"], list)
|
|
|
|
def test_internal_error_masks_details(self):
|
|
"""In production, internal errors should not leak stack traces."""
|
|
error = {
|
|
"error": {
|
|
"code": 500,
|
|
"message": "Internal server error",
|
|
"type": "internal_error"
|
|
}
|
|
}
|
|
assert "traceback" not in str(error)
|
|
assert error["error"]["message"] == "Internal server error"
|