"""B15 wiring — AI-Act Rechtsgrundlage-Check für LLM-Vendors. Hängt sich an `state["extra_findings"]` an und rendert einen V2-Block (`ai_legal_basis_html`). """ from __future__ import annotations import html import logging from compliance.services.ai_legal_basis_check import check_ai_legal_basis logger = logging.getLogger(__name__) def run_b15(state: dict) -> None: new = check_ai_legal_basis(state) if not new: return extras = state.get("extra_findings") or [] extras.extend(new) state["extra_findings"] = extras state["ai_legal_basis_html"] = _render(new) logger.info("B15 ai-legal-basis: %d finding(s)", len(new)) def _render(findings: list[dict]) -> str: cards = [] for f in findings: sev = (f.get("severity") or "").upper() color = "#dc2626" if sev == "HIGH" else "#f59e0b" meta = ( "
" f"Provider: {html.escape(f.get('provider') or '?')} · " f"Doc: {html.escape(f.get('doc_type') or '?')}
" ) cards.append( f"
" f"
" f"{sev} · {html.escape(f.get('check_id') or '')}
" f"
" f"{html.escape(f.get('title') or '')}
" f"
" f"{html.escape(f.get('norm') or '')}
" f"{meta}" f"
" f"{html.escape(f.get('evidence') or '')}
" f"
" f"→ Empfehlung: " f"{html.escape(f.get('action') or '')}
" "
" ) return ( "
" "

" "🤖 AI-Act Rechtsgrundlage (LLM-Vendor auf berechtigtem Interesse)" "

" + "".join(cards) + "
" )