mirror of
https://github.com/TronoSfera/Law.git
synced 2026-05-18 18:13:46 +03:00
187 lines
7.8 KiB
Python
187 lines
7.8 KiB
Python
import os
|
|
import unittest
|
|
|
|
from fastapi.testclient import TestClient
|
|
|
|
os.environ.setdefault("DATABASE_URL", "sqlite+pysqlite:///:memory:")
|
|
os.environ.setdefault("REDIS_URL", "redis://localhost:6379/0")
|
|
os.environ.setdefault("S3_ENDPOINT", "http://localhost:9000")
|
|
os.environ.setdefault("S3_ACCESS_KEY", "test")
|
|
os.environ.setdefault("S3_SECRET_KEY", "test")
|
|
os.environ.setdefault("S3_BUCKET", "test")
|
|
|
|
from app.main import app
|
|
from app.core.http_hardening import _performance_label, _response_security_headers
|
|
from starlette.requests import Request
|
|
|
|
|
|
class HttpHardeningTests(unittest.TestCase):
|
|
def setUp(self):
|
|
self.client = TestClient(app)
|
|
|
|
def tearDown(self):
|
|
self.client.close()
|
|
|
|
def test_health_has_security_headers_and_request_id(self):
|
|
response = self.client.get("/health")
|
|
self.assertEqual(response.status_code, 200)
|
|
|
|
self.assertEqual(response.headers.get("x-content-type-options"), "nosniff")
|
|
self.assertEqual(response.headers.get("x-frame-options"), "DENY")
|
|
self.assertEqual(response.headers.get("referrer-policy"), "no-referrer")
|
|
self.assertEqual(response.headers.get("x-permitted-cross-domain-policies"), "none")
|
|
self.assertEqual(response.headers.get("cross-origin-opener-policy"), "same-origin")
|
|
|
|
request_id = response.headers.get("x-request-id")
|
|
self.assertIsNotNone(request_id)
|
|
self.assertRegex(str(request_id), r"^[A-Za-z0-9._-]{1,128}$")
|
|
|
|
def test_valid_request_id_is_preserved(self):
|
|
external_request_id = "release-check-2026_02_23"
|
|
response = self.client.get("/health", headers={"X-Request-ID": external_request_id})
|
|
self.assertEqual(response.status_code, 200)
|
|
self.assertEqual(response.headers.get("x-request-id"), external_request_id)
|
|
|
|
def test_invalid_request_id_is_replaced(self):
|
|
bad_request_id = "bad id with spaces"
|
|
response = self.client.get("/health", headers={"X-Request-ID": bad_request_id})
|
|
self.assertEqual(response.status_code, 200)
|
|
|
|
response_request_id = response.headers.get("x-request-id")
|
|
self.assertIsNotNone(response_request_id)
|
|
self.assertNotEqual(response_request_id, bad_request_id)
|
|
self.assertRegex(str(response_request_id), r"^[A-Za-z0-9._-]{1,128}$")
|
|
|
|
def test_error_response_keeps_security_headers_and_request_id(self):
|
|
# No public cookie => 401 from dependency, middleware headers must still be present.
|
|
response = self.client.get("/api/public/requests/TRK-UNKNOWN")
|
|
self.assertEqual(response.status_code, 401)
|
|
self.assertEqual(response.headers.get("x-content-type-options"), "nosniff")
|
|
self.assertEqual(response.headers.get("x-frame-options"), "DENY")
|
|
self.assertTrue(bool(response.headers.get("x-request-id")))
|
|
|
|
def test_file_preview_paths_allow_same_origin_framing_only(self):
|
|
scope = {
|
|
"type": "http",
|
|
"http_version": "1.1",
|
|
"method": "GET",
|
|
"scheme": "http",
|
|
"path": "/api/public/uploads/object/123",
|
|
"raw_path": b"/api/public/uploads/object/123",
|
|
"query_string": b"",
|
|
"headers": [],
|
|
"client": ("127.0.0.1", 12345),
|
|
"server": ("testserver", 80),
|
|
}
|
|
headers = _response_security_headers(Request(scope))
|
|
self.assertEqual(headers.get("X-Frame-Options"), "SAMEORIGIN")
|
|
self.assertIn("frame-ancestors 'self'", str(headers.get("Content-Security-Policy")))
|
|
|
|
def test_target_perf_endpoint_has_observability_headers(self):
|
|
response = self.client.get("/api/admin/requests/kanban")
|
|
self.assertEqual(response.status_code, 401)
|
|
self.assertEqual(response.headers.get("x-perf-label"), "admin_kanban")
|
|
self.assertTrue(bool(response.headers.get("x-perf-duration-ms")))
|
|
self.assertIn('desc="admin_kanban"', str(response.headers.get("server-timing")))
|
|
|
|
def test_non_target_endpoint_has_no_perf_headers(self):
|
|
response = self.client.get("/health")
|
|
self.assertEqual(response.status_code, 200)
|
|
self.assertIsNone(response.headers.get("x-perf-label"))
|
|
self.assertIsNone(response.headers.get("x-perf-duration-ms"))
|
|
self.assertIsNone(response.headers.get("server-timing"))
|
|
|
|
def test_performance_label_maps_client_workspace_endpoints(self):
|
|
scope = {
|
|
"type": "http",
|
|
"http_version": "1.1",
|
|
"method": "GET",
|
|
"scheme": "http",
|
|
"path": "/api/public/requests/TRK-1/status-route",
|
|
"raw_path": b"/api/public/requests/TRK-1/status-route",
|
|
"query_string": b"",
|
|
"headers": [],
|
|
"client": ("127.0.0.1", 12345),
|
|
"server": ("testserver", 80),
|
|
}
|
|
self.assertEqual(_performance_label(Request(scope)), "public_request_status_route")
|
|
|
|
def test_performance_label_maps_admin_chat_messages_window(self):
|
|
scope = {
|
|
"type": "http",
|
|
"http_version": "1.1",
|
|
"method": "GET",
|
|
"scheme": "http",
|
|
"path": "/api/admin/chat/requests/123/messages-window",
|
|
"raw_path": b"/api/admin/chat/requests/123/messages-window",
|
|
"query_string": b"",
|
|
"headers": [],
|
|
"client": ("127.0.0.1", 12345),
|
|
"server": ("testserver", 80),
|
|
}
|
|
self.assertEqual(_performance_label(Request(scope)), "admin_chat_messages_window")
|
|
|
|
def test_performance_label_maps_public_chat_messages_window(self):
|
|
scope = {
|
|
"type": "http",
|
|
"http_version": "1.1",
|
|
"method": "GET",
|
|
"scheme": "http",
|
|
"path": "/api/public/chat/requests/TRK-1/messages-window",
|
|
"raw_path": b"/api/public/chat/requests/TRK-1/messages-window",
|
|
"query_string": b"",
|
|
"headers": [],
|
|
"client": ("127.0.0.1", 12345),
|
|
"server": ("testserver", 80),
|
|
}
|
|
self.assertEqual(_performance_label(Request(scope)), "public_chat_messages_window")
|
|
|
|
def test_performance_label_maps_admin_metrics_overview(self):
|
|
scope = {
|
|
"type": "http",
|
|
"http_version": "1.1",
|
|
"method": "GET",
|
|
"scheme": "http",
|
|
"path": "/api/admin/metrics/overview",
|
|
"raw_path": b"/api/admin/metrics/overview",
|
|
"query_string": b"",
|
|
"headers": [],
|
|
"client": ("127.0.0.1", 12345),
|
|
"server": ("testserver", 80),
|
|
}
|
|
self.assertEqual(_performance_label(Request(scope)), "admin_metrics_overview")
|
|
|
|
def test_performance_label_maps_admin_metrics_overview_sla(self):
|
|
scope = {
|
|
"type": "http",
|
|
"http_version": "1.1",
|
|
"method": "GET",
|
|
"scheme": "http",
|
|
"path": "/api/admin/metrics/overview-sla",
|
|
"raw_path": b"/api/admin/metrics/overview-sla",
|
|
"query_string": b"",
|
|
"headers": [],
|
|
"client": ("127.0.0.1", 12345),
|
|
"server": ("testserver", 80),
|
|
}
|
|
self.assertEqual(_performance_label(Request(scope)), "admin_metrics_overview_sla")
|
|
|
|
def test_non_file_paths_keep_deny_framing(self):
|
|
scope = {
|
|
"type": "http",
|
|
"http_version": "1.1",
|
|
"method": "GET",
|
|
"scheme": "http",
|
|
"path": "/api/public/requests/TRK-1",
|
|
"raw_path": b"/api/public/requests/TRK-1",
|
|
"query_string": b"",
|
|
"headers": [],
|
|
"client": ("127.0.0.1", 12345),
|
|
"server": ("testserver", 80),
|
|
}
|
|
headers = _response_security_headers(Request(scope))
|
|
self.assertEqual(headers.get("X-Frame-Options"), "DENY")
|
|
csp = str(headers.get("Content-Security-Policy"))
|
|
self.assertIn("frame-ancestors 'none'", csp)
|
|
self.assertIn("script-src 'self'", csp)
|
|
self.assertIn("connect-src 'self'", csp)
|