feat(cra): standalone POST /api/v1/cra/assess endpoint

Live HTTP entry for the deterministic CRA assessment — repo-scanner findings
in, CRA Annex I mapping + risk + curated measures + NIST/OWASP golden-set
crosswalk out. Project-less (works for any customer, no CE-RA/FMEA required);
reuses the tested mapper, same logic the MCP server exposes. Additive endpoint
(no contract baseline change); no DB.

Co-Authored-By: Claude Opus 4.7 <noreply@anthropic.com>
This commit is contained in:
Benjamin Admin
2026-06-14 07:19:01 +02:00
parent 7f03ffadcc
commit 34a678caef
3 changed files with 82 additions and 0 deletions
@@ -0,0 +1,40 @@
"""Contract test for the standalone CRA assess endpoint.
Mounts only the cra_assess router on a minimal app — no full app/DB startup.
"""
from fastapi import FastAPI
from fastapi.testclient import TestClient
from compliance.api.cra_assess_routes import router
app = FastAPI()
app.include_router(router, prefix="/api")
client = TestClient(app)
def test_assess_maps_findings_with_crosswalk():
r = client.post("/api/v1/cra/assess", json={"findings": [
{"id": "x", "title": "default password", "cwe": "CWE-259", "severity": "critical"},
{"id": "y", "category": "dependency", "title": "outdated lib", "severity": "high"},
]})
assert r.status_code == 200
d = r.json()
assert d["findings_total"] == 2
by_id = {m["finding_id"]: m for m in d["mapped"]}
assert by_id["x"]["primary_requirement"] == "CRA-AI-8"
assert by_id["x"]["risk_level"] == "CRITICAL"
assert "IA-5" in by_id["x"]["nist_refs"]
assert by_id["y"]["primary_requirement"] == "CRA-AI-22"
assert any(o["code"] == "A06:2021" for o in by_id["y"]["owasp_refs"])
def test_assess_empty_is_ok():
r = client.post("/api/v1/cra/assess", json={"findings": []})
assert r.status_code == 200
assert r.json()["findings_total"] == 0
def test_assess_requires_finding_id():
# id is required by the schema -> 422
r = client.post("/api/v1/cra/assess", json={"findings": [{"title": "no id"}]})
assert r.status_code == 422