feat(agents): Sprint 1.12 Phase 2 — Cookie-Policy v3 + ImpressumAgent v3 finetune
CI / detect-changes (push) Successful in 8s
CI / branch-name (push) Has been skipped
CI / guardrail-integrity (push) Has been skipped
CI / secret-scan (push) Has been skipped
CI / dep-audit (push) Has been skipped
CI / build-sha-integrity (push) Failing after 4s
CI / validate-canonical-controls (push) Successful in 11s
CI / loc-budget (push) Successful in 14s
CI / go-lint (push) Has been skipped
CI / python-lint (push) Has been skipped
CI / nodejs-lint (push) Has been skipped
CI / nodejs-build (push) Has been skipped
CI / test-go (push) Has been skipped
CI / iace-gt-coverage (push) Has been skipped
CI / test-python-backend (push) Successful in 30s
CI / test-python-document-crawler (push) Has been skipped
CI / test-python-dsms-gateway (push) Has been skipped
CI / sbom-scan (push) Has been skipped
CI / detect-changes (push) Successful in 8s
CI / branch-name (push) Has been skipped
CI / guardrail-integrity (push) Has been skipped
CI / secret-scan (push) Has been skipped
CI / dep-audit (push) Has been skipped
CI / build-sha-integrity (push) Failing after 4s
CI / validate-canonical-controls (push) Successful in 11s
CI / loc-budget (push) Successful in 14s
CI / go-lint (push) Has been skipped
CI / python-lint (push) Has been skipped
CI / nodejs-lint (push) Has been skipped
CI / nodejs-build (push) Has been skipped
CI / test-go (push) Has been skipped
CI / iace-gt-coverage (push) Has been skipped
CI / test-python-backend (push) Successful in 30s
CI / test-python-document-crawler (push) Has been skipped
CI / test-python-dsms-gateway (push) Has been skipped
CI / sbom-scan (push) Has been skipped
ImpressumAgent v3 (Refactor):
- v3_engine: laedt direkt alle 75 doc_check_controls['impressum'] ohne
Sidecar-Filter (Sidecar war zu streng, lieferte nur 3 von 75 MCs).
- Layer 0 Boost prueft pass+fail_criteria gegen meine 12 Patterns mit
erweiterten Initial-Seeds (User-Vorgabe 2026-06-09:
manuelle Initial-Seeds OK, Auto-Learning erweitert zur Laufzeit).
- ETO-Smoke: 75 DB-MCs · 7 Pattern-Boosts · 24 Boost-Overrides
(versus 3 DB-MCs vorher).
CookiePolicyAgent v3 (Refactor):
- cookie_policy/v3_engine.py + cookie_policy/regex_boost.py
- Laedt direkt alle 381 Cookie-MCs aus doc_check_controls
- Layer 0 mit 12 eigenen Patterns als Initial-Seed
- KB-Layer (CMP-Vendor-Cross-Check) bleibt erhalten
- agent_version='3.0'
Tests: 27/27 gruen (12 v3-impressum, 6 cookie-policy, 9 cross-placement).
Alte v2-cookie-tests umgeschrieben auf v3-Pipeline-Mock.
Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
This commit is contained in:
@@ -1,14 +1,12 @@
|
|||||||
"""Cookie-Policy-Agent v2 — BaseSpecialistAgent.
|
"""Cookie-Policy-Agent v3 — baut auf doc_check_controls (381 DB-MCs).
|
||||||
|
|
||||||
Prüft den Cookie-Policy-DOKUMENT-Text (NICHT das Banner — das macht
|
Sprint 1.12 Phase 2 — analog zu impressum/agent.py:
|
||||||
der Cookie-Banner-Themen-Agent). Konsumiert optional context.cmp_vendors
|
Layer 0 — Regex-Boost (meine 12 Patterns aus mcs.py)
|
||||||
für Konsistenz-Checks gegen die tatsächlich beobachtete Cookie-Liste.
|
Layer 1 — Keyword-Match aus pass_criteria der 381 Cookie-MCs
|
||||||
|
Layer 2 — BGE-M3 Embedding-Match
|
||||||
|
Layer 3 — Semantic-Validator (LLM) + Auto-Learning-Library
|
||||||
|
|
||||||
Eskalations-Stufen:
|
Output-Layer (Linter / Rollup / Methodik-UI) bleibt 1:1.
|
||||||
1. MC (regex) — schnell, deterministisch
|
|
||||||
2. cookie_library_lookup gegen state.context.cmp_vendors (wenn vorhanden)
|
|
||||||
3. LLM (qwen2.5:7b) für strukturelle/semantische Lücken
|
|
||||||
4. OVH 120b als Fallback
|
|
||||||
"""
|
"""
|
||||||
|
|
||||||
from __future__ import annotations
|
from __future__ import annotations
|
||||||
@@ -28,132 +26,128 @@ from .._base import (
|
|||||||
SourceType,
|
SourceType,
|
||||||
lint_output,
|
lint_output,
|
||||||
)
|
)
|
||||||
from .._escalation import cascade
|
from .._pattern_library import record as record_pattern
|
||||||
from .._rollup import rollup
|
from .._rollup import rollup
|
||||||
|
from .._semantic_validator import build_rename_action, validate_present
|
||||||
from .mcs import MC_IDS, MCS
|
from .mcs import MC_IDS, MCS
|
||||||
|
from .v3_engine import run_v3_pipeline
|
||||||
|
|
||||||
logger = logging.getLogger(__name__)
|
logger = logging.getLogger(__name__)
|
||||||
|
|
||||||
|
|
||||||
_SYSTEM_PROMPT = """Du bist ein deutscher Datenschutz-Anwalt mit Fokus
|
_SEV_TO_ENUM = {
|
||||||
TDDDG § 25 + DSGVO Art. 13 + EuGH Planet49 + BGH Cookie-II. Aufgabe:
|
"CRITICAL": Severity.HIGH,
|
||||||
eine Cookie-Richtlinie auf strukturelle und inhaltliche LÜCKEN prüfen,
|
"HIGH": Severity.HIGH,
|
||||||
die einer regex-basierten Vorprüfung entgangen sind.
|
"MEDIUM": Severity.MEDIUM,
|
||||||
|
"LOW": Severity.LOW,
|
||||||
WICHTIG:
|
"INFO": Severity.INFO,
|
||||||
- KEINE Bewertung "rechtssicher" / "garantiert" / "konform".
|
}
|
||||||
- Wenn unsicher: leeres Array zurückgeben statt zu halluzinieren.
|
|
||||||
- Wörtliches Zitat als evidence bei jeder Lücke.
|
|
||||||
|
|
||||||
Antworte NUR mit JSON, Schema:
|
|
||||||
{"findings": [
|
|
||||||
{"field_id": "...", "severity": "HIGH|MEDIUM|LOW",
|
|
||||||
"title": "...", "evidence": "wörtliches Zitat",
|
|
||||||
"action": "konkrete Empfehlung"}
|
|
||||||
]}
|
|
||||||
|
|
||||||
Typische Lücken-Kategorien:
|
|
||||||
- pseudo_purpose: "Siehe dazugehörige Datenverarbeitung" ohne konkrete Aussage
|
|
||||||
- duration_floskel: "solange erforderlich" ohne Zeitangabe
|
|
||||||
- vendor_unklar: "möglicherweise Drittanbieter" ohne Liste
|
|
||||||
- retention_inkonsistent: Tabelle nennt Tage, Fließtext nennt "session"
|
|
||||||
- drittland_fehlend: US-Vendor genannt (Google, Meta) aber Schrems-II
|
|
||||||
nicht thematisiert
|
|
||||||
- banner_reopen_fehlt: "Cookie-Einstellungen ändern" Link fehlt
|
|
||||||
"""
|
|
||||||
|
|
||||||
|
|
||||||
class CookiePolicyAgent(BaseSpecialistAgent):
|
class CookiePolicyAgent(BaseSpecialistAgent):
|
||||||
agent_id = "cookie_policy"
|
agent_id = "cookie_policy"
|
||||||
agent_version = "1.0"
|
agent_version = "3.0"
|
||||||
doc_type = "cookie"
|
doc_type = "cookie"
|
||||||
owned_mc_ids = MC_IDS
|
owned_mc_ids = MC_IDS
|
||||||
|
|
||||||
async def evaluate(self, agent_input: AgentInput) -> AgentOutput:
|
async def evaluate(self, agent_input: AgentInput) -> AgentOutput:
|
||||||
start = datetime.now(timezone.utc)
|
start = datetime.now(timezone.utc)
|
||||||
text = (agent_input.text or "").strip()
|
text = (agent_input.text or "").strip()
|
||||||
|
scope = set(agent_input.business_scope or [])
|
||||||
coverage: list[McCoverage] = []
|
coverage: list[McCoverage] = []
|
||||||
findings: list[Finding] = []
|
findings: list[Finding] = []
|
||||||
esc_logs: list[EscalationLog] = []
|
esc_logs: list[EscalationLog] = []
|
||||||
|
notes_parts: list[str] = []
|
||||||
|
|
||||||
if len(text) < 100:
|
if len(text) < 100:
|
||||||
for mc in MCS:
|
for mc in MCS:
|
||||||
coverage.append(McCoverage(
|
coverage.append(McCoverage(
|
||||||
mc_id=mc.mc_id, status="skipped",
|
mc_id=mc.mc_id, status="skipped",
|
||||||
reason="cookie policy text too short or empty",
|
reason="text too short",
|
||||||
))
|
))
|
||||||
return self._finalize(
|
return self._finalize(
|
||||||
start, findings, esc_logs, coverage, confidence=0.0,
|
start, findings, esc_logs, coverage,
|
||||||
|
confidence=0.0,
|
||||||
notes="Cookie-Policy-Text zu kurz oder leer.",
|
notes="Cookie-Policy-Text zu kurz oder leer.",
|
||||||
)
|
)
|
||||||
|
|
||||||
for mc in MCS:
|
results, telemetry = await run_v3_pipeline(text, scope)
|
||||||
matched = [p for p in mc.patterns if p.search(text)]
|
notes_parts.append(
|
||||||
if mc.require_all:
|
f"v3-pipeline: {telemetry.get('total_mcs', 0)} DB-MCs · "
|
||||||
ok = len(matched) == len(mc.patterns)
|
f"{telemetry.get('layer_0_field_hits', 0)} Pattern-Boosts · "
|
||||||
else:
|
f"{telemetry.get('layer_0_boost_overrides', 0)} Boost-Overrides"
|
||||||
ok = bool(matched)
|
)
|
||||||
if ok:
|
|
||||||
coverage.append(McCoverage(
|
seen: set[str] = set()
|
||||||
mc_id=mc.mc_id, status="ok",
|
for r in results:
|
||||||
reason=f"{len(matched)}/{len(mc.patterns)} patterns hit",
|
mc_id = r.get("control_id") or ""
|
||||||
))
|
if not mc_id or mc_id in seen:
|
||||||
continue
|
continue
|
||||||
sev = self._sev(mc.severity_if_missing)
|
seen.add(mc_id)
|
||||||
action = self._build_action(mc)
|
passed = bool(r.get("passed"))
|
||||||
|
sev = _SEV_TO_ENUM.get(
|
||||||
|
(r.get("severity") or "MEDIUM").upper(), Severity.MEDIUM,
|
||||||
|
)
|
||||||
|
coverage.append(McCoverage(
|
||||||
|
mc_id=mc_id,
|
||||||
|
status="ok" if passed else sev.value.lower(),
|
||||||
|
reason=str(r.get("matched_text") or r.get("hint") or "")[:120],
|
||||||
|
))
|
||||||
|
if passed:
|
||||||
|
continue
|
||||||
|
label = r.get("label") or r.get("hint") or ""
|
||||||
findings.append(Finding(
|
findings.append(Finding(
|
||||||
check_id=f"COOKIE-POLICY-AGENT-{mc.field_id.upper()}",
|
check_id=f"DBMC-{mc_id}",
|
||||||
agent=self.agent_id,
|
agent=self.agent_id,
|
||||||
agent_version=self.agent_version,
|
agent_version=self.agent_version,
|
||||||
field_id=mc.field_id,
|
field_id=mc_id,
|
||||||
severity=sev,
|
severity=sev,
|
||||||
severity_reason="missing",
|
severity_reason="db_mc_failed",
|
||||||
title=f"Cookie-Policy-Lücke: '{mc.label}'",
|
title=str(label)[:200] or f"DB-MC {mc_id} nicht erfüllt",
|
||||||
norm=mc.norm,
|
norm=str(r.get("regulation") or "") +
|
||||||
action=action,
|
(f" Art. {r.get('article')}"
|
||||||
confidence=0.92,
|
if r.get("article") else ""),
|
||||||
|
evidence="",
|
||||||
|
action=str(r.get("hint") or "")[:400]
|
||||||
|
or "Bitte gegen die Cookie-Pflichten prüfen.",
|
||||||
|
confidence=0.9,
|
||||||
sources=[EvidenceSource(
|
sources=[EvidenceSource(
|
||||||
source_type=SourceType.MC,
|
source_type=SourceType.MC,
|
||||||
source_id=mc.mc_id,
|
source_id=mc_id,
|
||||||
detail=f"0/{len(mc.patterns)} pattern hit",
|
detail=str(r.get("source") or "keyword_match")[:120],
|
||||||
|
confidence=0.9,
|
||||||
)],
|
)],
|
||||||
))
|
))
|
||||||
|
|
||||||
|
boost_ids = set(telemetry.get("layer_0_field_ids") or [])
|
||||||
|
for mc in MCS:
|
||||||
coverage.append(McCoverage(
|
coverage.append(McCoverage(
|
||||||
mc_id=mc.mc_id,
|
mc_id=mc.mc_id,
|
||||||
status=sev.value.lower(),
|
status="ok" if mc.field_id in boost_ids else "na",
|
||||||
reason="missing",
|
reason=("regex-boost hit"
|
||||||
|
if mc.field_id in boost_ids
|
||||||
|
else "kein Pattern-Treffer (kein Veto)"),
|
||||||
))
|
))
|
||||||
|
|
||||||
# KB-Layer: wenn cmp_vendors im Kontext, checke ob die Policy
|
await self._semantic_demote(text, findings, coverage)
|
||||||
# alle beobachteten Vendoren erwähnt
|
|
||||||
kb_findings = self._kb_layer(text, agent_input.context or {})
|
kb_findings = self._kb_layer(text, agent_input.context or {})
|
||||||
findings.extend(kb_findings)
|
findings.extend(kb_findings)
|
||||||
|
|
||||||
# LLM-Eskalation für subtile Lücken (Pseudo-Zwecke, Floskeln)
|
|
||||||
llm_findings, llm_logs = await self._maybe_escalate(text)
|
|
||||||
esc_logs.extend(llm_logs)
|
|
||||||
seen = {f.field_id for f in findings if f.field_id}
|
|
||||||
for f in llm_findings:
|
|
||||||
if f.field_id and f.field_id in seen:
|
|
||||||
continue
|
|
||||||
findings.append(f)
|
|
||||||
|
|
||||||
confs = [f.confidence for f in findings if f.confidence] or [0.95]
|
confs = [f.confidence for f in findings if f.confidence] or [0.95]
|
||||||
overall = sum(confs) / len(confs)
|
overall = sum(confs) / len(confs)
|
||||||
|
|
||||||
return self._finalize(start, findings, esc_logs, coverage,
|
return self._finalize(
|
||||||
confidence=overall)
|
start, findings, esc_logs, coverage,
|
||||||
|
confidence=overall, notes=" · ".join(notes_parts),
|
||||||
|
)
|
||||||
|
|
||||||
def _kb_layer(
|
def _kb_layer(self, text: str, context: dict) -> list[Finding]:
|
||||||
self, text: str, context: dict,
|
"""Wenn cmp_vendors im Kontext: prüfe ob alle in Policy genannt."""
|
||||||
) -> list[Finding]:
|
|
||||||
"""Wenn cmp_vendors gegeben: prüfe ob alle Vendoren in der Policy
|
|
||||||
erwähnt werden. Sonst Skip (keine Cross-Check ohne Datenbasis)."""
|
|
||||||
cmp_vendors = context.get("cmp_vendors") or []
|
cmp_vendors = context.get("cmp_vendors") or []
|
||||||
if not cmp_vendors:
|
if not cmp_vendors:
|
||||||
return []
|
return []
|
||||||
text_lc = text.lower()
|
text_lc = text.lower()
|
||||||
# Extrahiere Top-Vendor-Namen aus dem CMP
|
|
||||||
seen_names: set[str] = set()
|
seen_names: set[str] = set()
|
||||||
for v in cmp_vendors:
|
for v in cmp_vendors:
|
||||||
if not isinstance(v, dict):
|
if not isinstance(v, dict):
|
||||||
@@ -161,13 +155,10 @@ class CookiePolicyAgent(BaseSpecialistAgent):
|
|||||||
name = (v.get("name") or v.get("vendor") or "").strip()
|
name = (v.get("name") or v.get("vendor") or "").strip()
|
||||||
if name and len(name) > 2:
|
if name and len(name) > 2:
|
||||||
seen_names.add(name)
|
seen_names.add(name)
|
||||||
missing: list[str] = []
|
missing = [n for n in sorted(seen_names)
|
||||||
for n in sorted(seen_names):
|
if n.lower() not in text_lc]
|
||||||
if n.lower() not in text_lc:
|
|
||||||
missing.append(n)
|
|
||||||
if not missing:
|
if not missing:
|
||||||
return []
|
return []
|
||||||
# Ein Sammel-Finding pro Lücke
|
|
||||||
sample = missing[:8]
|
sample = missing[:8]
|
||||||
return [Finding(
|
return [Finding(
|
||||||
check_id="COOKIE-POLICY-AGENT-CMP-VS-POLICY",
|
check_id="COOKIE-POLICY-AGENT-CMP-VS-POLICY",
|
||||||
@@ -177,76 +168,82 @@ class CookiePolicyAgent(BaseSpecialistAgent):
|
|||||||
severity=Severity.MEDIUM,
|
severity=Severity.MEDIUM,
|
||||||
severity_reason="cmp_observed_vendors_not_in_policy",
|
severity_reason="cmp_observed_vendors_not_in_policy",
|
||||||
title=(
|
title=(
|
||||||
f"{len(missing)} im CMP beobachtete Vendor(en) "
|
f"{len(missing)} im CMP beobachtete Vendoren "
|
||||||
"fehlen in der Cookie-Policy"
|
"fehlen in der Cookie-Policy"
|
||||||
),
|
),
|
||||||
norm="DSGVO Art. 13 Abs. 1 lit. e (Empfänger vollständig nennen)",
|
norm="DSGVO Art. 13 Abs. 1 lit. e (Empfänger vollständig)",
|
||||||
evidence=f"Fehlend: {', '.join(sample)}"
|
evidence=f"Fehlend: {', '.join(sample)}"
|
||||||
+ (" …" if len(missing) > 8 else ""),
|
+ (" …" if len(missing) > 8 else ""),
|
||||||
action=(
|
action=(
|
||||||
"Die im Cookie-Consent-Banner beobachteten Vendoren "
|
"Die im Cookie-Consent-Banner beobachteten Vendoren "
|
||||||
"(Tracker/Werbenetzwerke) müssen vollständig in der "
|
"müssen vollständig in der Cookie-Richtlinie genannt sein."
|
||||||
"Cookie-Richtlinie aufgelistet sein."
|
|
||||||
),
|
),
|
||||||
confidence=0.88,
|
confidence=0.88,
|
||||||
sources=[EvidenceSource(
|
sources=[EvidenceSource(
|
||||||
source_type=SourceType.MC,
|
source_type=SourceType.CROSS,
|
||||||
source_id="CMP-CROSS-CHECK",
|
source_id="CMP-CROSS-CHECK",
|
||||||
detail=f"{len(missing)} missing of {len(seen_names)}",
|
detail=f"{len(missing)} missing of {len(seen_names)}",
|
||||||
)],
|
)],
|
||||||
)]
|
)]
|
||||||
|
|
||||||
async def _maybe_escalate(
|
async def _semantic_demote(
|
||||||
self, text: str,
|
self, text: str, findings: list[Finding],
|
||||||
) -> tuple[list[Finding], list[EscalationLog]]:
|
coverage: list[McCoverage],
|
||||||
user_prompt = (
|
) -> None:
|
||||||
f"COOKIE-POLICY-TEXT:\n{text[:4500]}\n\n"
|
candidates = [
|
||||||
"Liste subtile Lücken nach TDDDG § 25 + DSGVO Art. 13. "
|
f for f in findings
|
||||||
"Nur JSON."
|
if f.severity in (Severity.HIGH.value, Severity.MEDIUM.value)
|
||||||
|
and f.severity_reason == "db_mc_failed"
|
||||||
|
]
|
||||||
|
if not candidates:
|
||||||
|
return
|
||||||
|
result = await validate_present(
|
||||||
|
text, [(f.field_id, f.title[:80]) for f in candidates],
|
||||||
)
|
)
|
||||||
res, logs = await cascade(_SYSTEM_PROMPT, user_prompt)
|
if not result:
|
||||||
if res is None or not isinstance(res.parsed, (dict, list)):
|
return
|
||||||
return [], logs
|
for finding in candidates:
|
||||||
raw = (res.parsed.get("findings")
|
row = result.get(finding.field_id)
|
||||||
if isinstance(res.parsed, dict) else res.parsed)
|
if not row or not row.get("found"):
|
||||||
if not isinstance(raw, list):
|
|
||||||
return [], logs
|
|
||||||
out: list[Finding] = []
|
|
||||||
for item in raw:
|
|
||||||
if not isinstance(item, dict):
|
|
||||||
continue
|
continue
|
||||||
fid = str(item.get("field_id") or "unknown")[:40]
|
if row.get("confidence", 0) < 0.6:
|
||||||
sev_raw = str(item.get("severity") or "MEDIUM").upper()
|
continue
|
||||||
sev = self._sev(sev_raw)
|
label_used = row.get("label_used") or "abweichendes Label"
|
||||||
out.append(Finding(
|
conf = float(row.get("confidence") or 0.8)
|
||||||
check_id=f"COOKIE-POLICY-AGENT-LLM-{fid.upper()}",
|
finding.severity = Severity.LOW.value
|
||||||
agent=self.agent_id,
|
finding.severity_reason = "label_mismatch"
|
||||||
agent_version=self.agent_version,
|
finding.title = (
|
||||||
field_id=fid,
|
f"Label '{label_used}' weicht von Standard ab"
|
||||||
severity=sev,
|
)
|
||||||
severity_reason="llm_detected",
|
finding.evidence = str(row.get("evidence") or "")[:200]
|
||||||
title=str(item.get("title") or "")[:200],
|
finding.action = build_rename_action(
|
||||||
norm="TDDDG § 25 + DSGVO Art. 13 (LLM-Analyse)",
|
finding.field_id, label_used,
|
||||||
evidence=str(item.get("evidence") or "")[:300],
|
)
|
||||||
action=str(item.get("action") or "")[:400],
|
finding.confidence = conf
|
||||||
confidence=0.7,
|
finding.sources.append(EvidenceSource(
|
||||||
sources=[EvidenceSource(
|
source_type=SourceType.LLM_LOCAL,
|
||||||
source_type=res.stage,
|
source_id="semantic_validator",
|
||||||
source_id=res.model,
|
detail=f"LLM-confirmed: '{label_used}'",
|
||||||
detail=f"prompt_chars={len(user_prompt)}",
|
confidence=conf,
|
||||||
confidence=0.7,
|
|
||||||
)],
|
|
||||||
))
|
))
|
||||||
return out, logs
|
for c in coverage:
|
||||||
|
if c.mc_id == f"DBMC-{finding.field_id}":
|
||||||
|
c.status = "low"
|
||||||
|
c.reason = f"label_mismatch: '{label_used}'"
|
||||||
|
try:
|
||||||
|
record_pattern(
|
||||||
|
field_id=finding.field_id,
|
||||||
|
label_used=label_used,
|
||||||
|
confidence=conf,
|
||||||
|
agent_id=self.agent_id,
|
||||||
|
)
|
||||||
|
except Exception as e:
|
||||||
|
logger.warning("pattern-library record failed: %s", e)
|
||||||
|
|
||||||
def _finalize(
|
def _finalize(
|
||||||
self,
|
self, start: datetime, findings: list[Finding],
|
||||||
start: datetime,
|
esc_logs: list[EscalationLog], coverage: list[McCoverage],
|
||||||
findings: list[Finding],
|
confidence: float, notes: str = "",
|
||||||
esc_logs: list[EscalationLog],
|
|
||||||
coverage: list[McCoverage],
|
|
||||||
confidence: float,
|
|
||||||
notes: str = "",
|
|
||||||
) -> AgentOutput:
|
) -> AgentOutput:
|
||||||
end = datetime.now(timezone.utc)
|
end = datetime.now(timezone.utc)
|
||||||
recs = rollup(findings)
|
recs = rollup(findings)
|
||||||
@@ -270,78 +267,3 @@ class CookiePolicyAgent(BaseSpecialistAgent):
|
|||||||
mc_low=sum(1 for c in coverage if c.status == "low"),
|
mc_low=sum(1 for c in coverage if c.status == "low"),
|
||||||
)
|
)
|
||||||
return lint_output(out)
|
return lint_output(out)
|
||||||
|
|
||||||
@staticmethod
|
|
||||||
def _sev(value: str) -> Severity:
|
|
||||||
v = (value or "").upper()
|
|
||||||
if v == "HIGH":
|
|
||||||
return Severity.HIGH
|
|
||||||
if v == "MEDIUM":
|
|
||||||
return Severity.MEDIUM
|
|
||||||
if v == "LOW":
|
|
||||||
return Severity.LOW
|
|
||||||
return Severity.INFO
|
|
||||||
|
|
||||||
@staticmethod
|
|
||||||
def _build_action(mc) -> str:
|
|
||||||
suggestions = {
|
|
||||||
"categories_named": (
|
|
||||||
"Die Cookie-Richtlinie sollte die Kategorien essentiell, "
|
|
||||||
"funktional, analytics und marketing klar benennen und "
|
|
||||||
"abgrenzen."
|
|
||||||
),
|
|
||||||
"purpose_described": (
|
|
||||||
"Pro Cookie-Kategorie den Verarbeitungszweck konkret "
|
|
||||||
"benennen (keine Pauschal-Formulierungen wie "
|
|
||||||
"'verschiedene Zwecke')."
|
|
||||||
),
|
|
||||||
"retention_duration": (
|
|
||||||
"Speicherdauer pro Cookie konkret angeben "
|
|
||||||
"(z.B. 'Session', '30 Tage', '2 Jahre') statt "
|
|
||||||
"'solange erforderlich'."
|
|
||||||
),
|
|
||||||
"vendor_recipients": (
|
|
||||||
"Alle Empfänger / Drittanbieter namentlich auflisten "
|
|
||||||
"(z.B. Google LLC, Meta Platforms Inc., …) inkl. Sitz."
|
|
||||||
),
|
|
||||||
"opt_out_mechanism": (
|
|
||||||
"Konkreten Opt-Out-Weg beschreiben: Banner-Reopen-Link, "
|
|
||||||
"Browser-Einstellungen, Vendor-spezifische Opt-Out-URLs."
|
|
||||||
),
|
|
||||||
"banner_reopen": (
|
|
||||||
"Sichtbaren Link 'Cookie-Einstellungen ändern' in die "
|
|
||||||
"Policy aufnehmen, der den CMP-Banner wieder öffnet."
|
|
||||||
),
|
|
||||||
"version_date": (
|
|
||||||
"Stand der Cookie-Richtlinie sichtbar angeben "
|
|
||||||
"(z.B. 'Stand: 1. Juni 2026')."
|
|
||||||
),
|
|
||||||
"third_country_transfer": (
|
|
||||||
"Bei Drittland-Transfer (USA u.a.) Hinweis auf "
|
|
||||||
"Schrems-II-Risiko + verwendete Schutzmaßnahmen "
|
|
||||||
"(SCC, DPF) ergänzen."
|
|
||||||
),
|
|
||||||
"legal_basis": (
|
|
||||||
"Rechtsgrundlage pro Kategorie benennen: § 25 Abs. 1 "
|
|
||||||
"TDDDG (Einwilligung) bzw. § 25 Abs. 2 TDDDG "
|
|
||||||
"(unbedingt erforderlich)."
|
|
||||||
),
|
|
||||||
"cookie_table_or_list": (
|
|
||||||
"Detail-Tabelle mit Cookie-Namen, Vendor, Zweck und "
|
|
||||||
"Laufzeit pro Cookie ergänzen (DSK-Best-Practice)."
|
|
||||||
),
|
|
||||||
"dpo_contact": (
|
|
||||||
"Kontaktmöglichkeit zum DSB oder Datenschutz-Team "
|
|
||||||
"in der Cookie-Richtlinie nennen (z.B. "
|
|
||||||
"datenschutz@<domain>)."
|
|
||||||
),
|
|
||||||
"browser_settings_hint": (
|
|
||||||
"Hinweis auf Browser-Einstellungen zum Blockieren/"
|
|
||||||
"Löschen von Cookies (Chrome, Firefox, Safari, Edge) "
|
|
||||||
"ergänzen."
|
|
||||||
),
|
|
||||||
}
|
|
||||||
return suggestions.get(mc.field_id, (
|
|
||||||
f"{mc.label} in der Cookie-Richtlinie ergänzen "
|
|
||||||
f"({mc.norm})."
|
|
||||||
))
|
|
||||||
|
|||||||
+115
@@ -0,0 +1,115 @@
|
|||||||
|
"""Layer-0 Regex-Boost für Cookie-Policy-Agent v3.
|
||||||
|
|
||||||
|
Analog zu impressum/regex_boost.py: meine 12 Cookie-Policy-Patterns
|
||||||
|
(aus mcs.py) werden als Vor-Stufe vor dem Keyword-Match aus
|
||||||
|
doc_check_controls (381 Cookie-MCs) genutzt. Wenn Pattern hits, kann
|
||||||
|
das thematisch passende DB-MC zu PASS überschrieben werden.
|
||||||
|
|
||||||
|
User-Vorgabe 2026-06-09: manuelle Initial-Seeds sind erlaubt, das
|
||||||
|
Auto-Learning ergänzt zur Laufzeit.
|
||||||
|
"""
|
||||||
|
|
||||||
|
from __future__ import annotations
|
||||||
|
|
||||||
|
import logging
|
||||||
|
|
||||||
|
from .mcs import MCS
|
||||||
|
|
||||||
|
logger = logging.getLogger(__name__)
|
||||||
|
|
||||||
|
|
||||||
|
# Initial-Seed pro field_id — auf Cookie-Policy-Pflichten abgestimmt.
|
||||||
|
BOOST_KEYWORDS: dict[str, tuple[str, ...]] = {
|
||||||
|
"categories_named": (
|
||||||
|
"kategorie", "essentiell", "funktional", "analytics",
|
||||||
|
"marketing", "notwendig", "tracking",
|
||||||
|
),
|
||||||
|
"purpose_described": (
|
||||||
|
"zweck", "zwecke", "verarbeitungszweck", "verwendungszweck",
|
||||||
|
"dient zu", "dient zur",
|
||||||
|
),
|
||||||
|
"retention_duration": (
|
||||||
|
"speicherdauer", "laufzeit", "dauer", "gültigkeitsdauer",
|
||||||
|
"session", "persistent", "tag", "monat", "jahr",
|
||||||
|
),
|
||||||
|
"vendor_recipients": (
|
||||||
|
"empfänger", "vendor", "drittanbieter", "third-party",
|
||||||
|
"drittland", "anbieter", "verantwortlicher",
|
||||||
|
),
|
||||||
|
"opt_out_mechanism": (
|
||||||
|
"opt-out", "widerruf", "widerrufen", "deaktivieren",
|
||||||
|
"abwählen", "einstellungen ändern",
|
||||||
|
),
|
||||||
|
"banner_reopen": (
|
||||||
|
"cookie-einstellungen", "banner", "präferenzen",
|
||||||
|
"einwilligung verwalten", "consent",
|
||||||
|
),
|
||||||
|
"version_date": (
|
||||||
|
"stand", "aktualisierung", "version", "letzte änderung",
|
||||||
|
"gültig ab",
|
||||||
|
),
|
||||||
|
"third_country_transfer": (
|
||||||
|
"drittland", "drittstaat", "usa", "scc",
|
||||||
|
"standardvertragsklauseln", "angemessenheitsbeschluss",
|
||||||
|
"data privacy framework", "dpf",
|
||||||
|
),
|
||||||
|
"legal_basis": (
|
||||||
|
"rechtsgrundlage", "einwilligung", "berechtigtes interesse",
|
||||||
|
"art. 6", "§ 25 tdddg", "tdddg",
|
||||||
|
),
|
||||||
|
"cookie_table_or_list": (
|
||||||
|
"tabelle", "liste", "cookie-name", "_ga", "_fbp",
|
||||||
|
"optanonconsent",
|
||||||
|
),
|
||||||
|
"dpo_contact": (
|
||||||
|
"datenschutzbeauftragter", "datenschutz-team", "dsb",
|
||||||
|
"datenschutz@",
|
||||||
|
),
|
||||||
|
"browser_settings_hint": (
|
||||||
|
"browser-einstellungen", "chrome", "firefox", "safari",
|
||||||
|
"edge", "cookies löschen", "cookies blockieren",
|
||||||
|
),
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
def compute_regex_boosts(text: str) -> set[str]:
|
||||||
|
"""Welche field_ids wurden im Cookie-Policy-Text durch Patterns
|
||||||
|
erkannt?"""
|
||||||
|
if not text or len(text) < 50:
|
||||||
|
return set()
|
||||||
|
hits: set[str] = set()
|
||||||
|
for mc in MCS:
|
||||||
|
# require_all / any-Logik aus mcs.py respektieren
|
||||||
|
if mc.require_all:
|
||||||
|
ok = all(p.search(text) for p in mc.patterns)
|
||||||
|
else:
|
||||||
|
ok = any(p.search(text) for p in mc.patterns)
|
||||||
|
if ok:
|
||||||
|
hits.add(mc.field_id)
|
||||||
|
return hits
|
||||||
|
|
||||||
|
|
||||||
|
def boost_matches_db_mc(
|
||||||
|
boosts: set[str],
|
||||||
|
pass_criteria: list,
|
||||||
|
fail_criteria: list | None = None,
|
||||||
|
) -> str | None:
|
||||||
|
"""≥2 Boost-Keywords im kombinierten pass+fail-Text → match."""
|
||||||
|
if not boosts:
|
||||||
|
return None
|
||||||
|
parts: list[str] = []
|
||||||
|
for c in (pass_criteria or []):
|
||||||
|
if c: parts.append(str(c).lower())
|
||||||
|
for c in (fail_criteria or []):
|
||||||
|
if c: parts.append(str(c).lower())
|
||||||
|
if not parts:
|
||||||
|
return None
|
||||||
|
crit_text = " ".join(parts)
|
||||||
|
best: tuple[int, str] | None = None
|
||||||
|
for field_id in boosts:
|
||||||
|
kws = BOOST_KEYWORDS.get(field_id) or ()
|
||||||
|
match_count = sum(1 for kw in kws if kw in crit_text)
|
||||||
|
if match_count >= 2:
|
||||||
|
if best is None or match_count > best[0]:
|
||||||
|
best = (match_count, field_id)
|
||||||
|
return best[1] if best else None
|
||||||
@@ -0,0 +1,141 @@
|
|||||||
|
"""Cookie-Policy v3-Pipeline — analog zu impressum/v3_engine.py.
|
||||||
|
|
||||||
|
Lädt 381 Cookie-MCs aus compliance.doc_check_controls (doc_type='cookie'),
|
||||||
|
ruft den deterministischen Keyword-Check + Embedding-Match + Boost-Override.
|
||||||
|
"""
|
||||||
|
|
||||||
|
from __future__ import annotations
|
||||||
|
|
||||||
|
import logging
|
||||||
|
from typing import Any
|
||||||
|
|
||||||
|
from .regex_boost import boost_matches_db_mc, compute_regex_boosts
|
||||||
|
|
||||||
|
logger = logging.getLogger(__name__)
|
||||||
|
|
||||||
|
|
||||||
|
async def run_v3_pipeline(
|
||||||
|
text: str, business_scope: set[str],
|
||||||
|
) -> tuple[list[dict[str, Any]], dict[str, Any]]:
|
||||||
|
if not text or len(text) < 100:
|
||||||
|
return [], {"reason": "text too short"}
|
||||||
|
|
||||||
|
# Layer 0: meine Pattern-Boosts
|
||||||
|
boosts = compute_regex_boosts(text)
|
||||||
|
boost_field_ids = sorted(boosts)
|
||||||
|
|
||||||
|
# Layer 1: alle 381 Cookie-MCs aus DB laden
|
||||||
|
controls = await _load_cookie_mcs()
|
||||||
|
results: list[dict[str, Any]] = []
|
||||||
|
if controls:
|
||||||
|
try:
|
||||||
|
from compliance.services.rag_document_checker import (
|
||||||
|
_check_mc_deterministic,
|
||||||
|
)
|
||||||
|
text_lower = text.lower().replace("\xad", "")
|
||||||
|
for mc in controls:
|
||||||
|
r = _check_mc_deterministic(text_lower, mc)
|
||||||
|
if r:
|
||||||
|
r["_pass_criteria"] = mc.get("pass_criteria")
|
||||||
|
r["_fail_criteria"] = mc.get("fail_criteria")
|
||||||
|
results.append(r)
|
||||||
|
except Exception as e:
|
||||||
|
logger.warning("layer-1 keyword check failed: %s", e)
|
||||||
|
|
||||||
|
# Layer 2: Embedding-Match für failed MCs
|
||||||
|
failed_for_embed = [
|
||||||
|
c for c, r in zip(controls, results)
|
||||||
|
if r and not r.get("passed")
|
||||||
|
]
|
||||||
|
if failed_for_embed:
|
||||||
|
try:
|
||||||
|
from compliance.services.mc_embedding_matcher import (
|
||||||
|
ensure_mc_embeddings, embedding_match,
|
||||||
|
)
|
||||||
|
await ensure_mc_embeddings()
|
||||||
|
semantic_passes = await embedding_match(
|
||||||
|
text, failed_for_embed, doc_type="cookie",
|
||||||
|
)
|
||||||
|
if semantic_passes:
|
||||||
|
for r in results:
|
||||||
|
cid = r.get("control_id")
|
||||||
|
if cid in semantic_passes and not r.get("passed"):
|
||||||
|
r["passed"] = True
|
||||||
|
r["matched_text"] = "[layer-2 embedding match]"
|
||||||
|
r["source"] = (r.get("source") or "") + "+embedding"
|
||||||
|
except Exception as e:
|
||||||
|
logger.warning("layer-2 embedding skipped: %s", e)
|
||||||
|
|
||||||
|
# Layer 0 Boost-Override
|
||||||
|
boost_overrides = 0
|
||||||
|
for r in results:
|
||||||
|
if r.get("passed"):
|
||||||
|
continue
|
||||||
|
pass_crit = r.get("_pass_criteria") or []
|
||||||
|
fail_crit = r.get("_fail_criteria") or []
|
||||||
|
if not pass_crit and not fail_crit:
|
||||||
|
pass_crit = [r.get("hint") or r.get("label") or ""]
|
||||||
|
matched_field = boost_matches_db_mc(boosts, pass_crit, fail_crit)
|
||||||
|
if matched_field:
|
||||||
|
r["passed"] = True
|
||||||
|
r["matched_text"] = f"[regex-boost layer 0 — {matched_field}]"
|
||||||
|
r["source"] = (r.get("source") or "") + "+regex_boost"
|
||||||
|
boost_overrides += 1
|
||||||
|
|
||||||
|
layer_1_pass = sum(1 for r in results if r.get("passed")
|
||||||
|
and "+regex_boost" not in (r.get("source") or "")
|
||||||
|
and "+embedding" not in (r.get("source") or ""))
|
||||||
|
telemetry = {
|
||||||
|
"layer_0_field_hits": len(boost_field_ids),
|
||||||
|
"layer_0_field_ids": boost_field_ids,
|
||||||
|
"layer_1_pass": layer_1_pass,
|
||||||
|
"layer_0_boost_overrides": boost_overrides,
|
||||||
|
"total_mcs": len(results),
|
||||||
|
}
|
||||||
|
return results, telemetry
|
||||||
|
|
||||||
|
|
||||||
|
async def _load_cookie_mcs() -> list[dict]:
|
||||||
|
"""Lädt alle 381 Cookie-MCs aus compliance.doc_check_controls."""
|
||||||
|
try:
|
||||||
|
import json
|
||||||
|
from classroom_engine.database import SessionLocal
|
||||||
|
from sqlalchemy import text as _sa_text
|
||||||
|
db = SessionLocal()
|
||||||
|
try:
|
||||||
|
rows = db.execute(_sa_text(
|
||||||
|
"SELECT id, control_id, control_uuid, title, regulation, "
|
||||||
|
" article, check_question, pass_criteria, "
|
||||||
|
" fail_criteria, severity "
|
||||||
|
"FROM compliance.doc_check_controls "
|
||||||
|
"WHERE doc_type='cookie' "
|
||||||
|
"ORDER BY severity DESC, title"
|
||||||
|
)).fetchall()
|
||||||
|
finally:
|
||||||
|
db.close()
|
||||||
|
out = []
|
||||||
|
for r in rows:
|
||||||
|
def _parse(v):
|
||||||
|
if isinstance(v, list): return v
|
||||||
|
if isinstance(v, str):
|
||||||
|
try:
|
||||||
|
j = json.loads(v)
|
||||||
|
return j if isinstance(j, list) else [v]
|
||||||
|
except Exception: return [v]
|
||||||
|
return []
|
||||||
|
out.append({
|
||||||
|
"id": str(r[0]),
|
||||||
|
"control_id": r[1],
|
||||||
|
"control_uuid": str(r[2]) if r[2] else "",
|
||||||
|
"title": r[3] or "",
|
||||||
|
"regulation": r[4] or "",
|
||||||
|
"article": r[5] or "",
|
||||||
|
"check_question": r[6] or "",
|
||||||
|
"pass_criteria": _parse(r[7]),
|
||||||
|
"fail_criteria": _parse(r[8]),
|
||||||
|
"severity": r[9] or "MEDIUM",
|
||||||
|
})
|
||||||
|
return out
|
||||||
|
except Exception as e:
|
||||||
|
logger.warning("_load_cookie_mcs failed: %s", e)
|
||||||
|
return []
|
||||||
@@ -29,49 +29,70 @@ logger = logging.getLogger(__name__)
|
|||||||
# Für jedes meiner field_id: welche Wörter erscheinen typisch in
|
# Für jedes meiner field_id: welche Wörter erscheinen typisch in
|
||||||
# der pass_criteria der zugehörigen DB-MCs? Wenn diese Wörter im
|
# der pass_criteria der zugehörigen DB-MCs? Wenn diese Wörter im
|
||||||
# pass_criteria gefunden werden, ist es vermutlich derselbe MC.
|
# pass_criteria gefunden werden, ist es vermutlich derselbe MC.
|
||||||
|
# Initial-Seed der Standard-Synonyme pro field_id. User-Vorgabe
|
||||||
|
# 2026-06-09: manuelle Erweiterung als Initial-Seed ist OK; das
|
||||||
|
# LLM-basierte Auto-Learning (Sprint 1.10/1.11) ergänzt zur Laufzeit
|
||||||
|
# weitere Tail-Schreibweisen, sodass über die Zeit asymptotisch
|
||||||
|
# weniger LLM-Calls nötig sind.
|
||||||
BOOST_KEYWORDS: dict[str, tuple[str, ...]] = {
|
BOOST_KEYWORDS: dict[str, tuple[str, ...]] = {
|
||||||
"name_anbieter": (
|
"name_anbieter": (
|
||||||
"rechtsform", "anschrift", "anbieter", "firmensitz", "firmenname",
|
# Adresse / Anschrift
|
||||||
"diensteanbieter", "verantwortlich",
|
"anschrift", "adresse", "postadresse", "postalisch",
|
||||||
|
"geschäftsadresse", "geschäftssitz", "firmensitz",
|
||||||
|
"niederlassung", "niederlassungsort", "sitz", "ort",
|
||||||
|
"straße", "hausnummer", "plz",
|
||||||
|
# Firmenname / Rechtsform
|
||||||
|
"firma", "firmenname", "rechtsform", "kaufmann",
|
||||||
|
"anbieter", "diensteanbieter", "verantwortlich",
|
||||||
|
"anbieterkennzeichnung", "unternehmen",
|
||||||
),
|
),
|
||||||
"kontakt_email": (
|
"kontakt_email": (
|
||||||
"e-mail", "email", "elektronische", "kontaktmöglichkeit",
|
"e-mail", "email", "elektronische", "kontaktmöglichkeit",
|
||||||
"mailadresse",
|
"kontaktdaten", "mailadresse", "e-mail-adresse",
|
||||||
),
|
),
|
||||||
"kontakt_telefon": (
|
"kontakt_telefon": (
|
||||||
"telefon", "rufnummer", "telefonnummer", "phone", "kontaktdaten",
|
"telefon", "rufnummer", "telefonnummer", "phone",
|
||||||
"telekommunikation",
|
"kontaktdaten", "telekommunikation", "fax",
|
||||||
),
|
),
|
||||||
"handelsregister": (
|
"handelsregister": (
|
||||||
"handelsregister", "registergericht", "hrb", "registernummer",
|
"handelsregister", "registergericht", "hrb", "hra",
|
||||||
|
"registernummer", "registereintrag",
|
||||||
|
"handelsregisternummer", "handelsregisterauszug",
|
||||||
),
|
),
|
||||||
"ust_id": (
|
"ust_id": (
|
||||||
"umsatzsteuer", "ust-id", "umsatzsteueridentifikation", "ust-idnr",
|
"umsatzsteuer", "ust-id", "ust-idnr",
|
||||||
|
"umsatzsteueridentifikation",
|
||||||
|
"umsatzsteueridentifikationsnummer", "vat",
|
||||||
),
|
),
|
||||||
"vertretungsberechtigte": (
|
"vertretungsberechtigte": (
|
||||||
"geschäftsführer", "vorstand", "vertretungsberechtigt",
|
"geschäftsführer", "geschäftsführung", "vorstand",
|
||||||
"vertretung", "gesellschafter",
|
"vorsitzender", "vorstandsvorsitzender",
|
||||||
|
"vertretungsberechtigt", "vertretung", "vertreten",
|
||||||
|
"gesellschafter", "kaufmann", "inhaber",
|
||||||
),
|
),
|
||||||
"vertretungsberechtigte_label_korrekt": (
|
"vertretungsberechtigte_label_korrekt": (
|
||||||
"deutsche", "bezeichnung", "rechtsform",
|
"geschäftsführer", "vorstand", "deutsche", "bezeichnung",
|
||||||
|
"rechtsform",
|
||||||
),
|
),
|
||||||
"aufsichtsbehoerde": (
|
"aufsichtsbehoerde": (
|
||||||
"aufsichtsbehörde", "aufsicht", "behörde", "regulierungsbehörde",
|
"aufsichtsbehörde", "aufsicht", "behörde",
|
||||||
|
"regulierungsbehörde", "ihk", "bafin", "bnetza", "kba",
|
||||||
),
|
),
|
||||||
"verantwortlicher_redaktion": (
|
"verantwortlicher_redaktion": (
|
||||||
"redaktion", "verantwortlich", "rstv", "mstv",
|
"redaktion", "verantwortlich", "rstv", "mstv",
|
||||||
"journalistisch", "publizistisch",
|
"journalistisch", "publizistisch", "v.i.s.d.p",
|
||||||
),
|
),
|
||||||
"verbraucher_streitbeilegung": (
|
"verbraucher_streitbeilegung": (
|
||||||
"streitbeilegung", "vsbg", "verbraucherschlichtung",
|
"streitbeilegung", "vsbg", "verbraucherschlichtung",
|
||||||
"schlichtungsstelle",
|
"schlichtungsstelle", "verbraucherschlichtungsstelle",
|
||||||
),
|
),
|
||||||
"berufsangaben": (
|
"berufsangaben": (
|
||||||
"berufsbezeichnung", "berufsordnung", "kammer", "berufsrecht",
|
"berufsbezeichnung", "berufsordnung", "kammer",
|
||||||
|
"berufsrecht", "berufsverband",
|
||||||
),
|
),
|
||||||
"odr_link": (
|
"odr_link": (
|
||||||
"online-streitbeilegung", "os-plattform", "odr",
|
"online-streitbeilegung", "os-plattform", "odr",
|
||||||
"europäische kommission",
|
"europäische kommission", "ec.europa.eu",
|
||||||
),
|
),
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -94,22 +115,36 @@ def compute_regex_boosts(text: str, business_scope: set[str]) -> set[str]:
|
|||||||
return hits
|
return hits
|
||||||
|
|
||||||
|
|
||||||
def boost_matches_db_mc(boosts: set[str], pass_criteria: list) -> str | None:
|
def boost_matches_db_mc(
|
||||||
|
boosts: set[str],
|
||||||
|
pass_criteria: list,
|
||||||
|
fail_criteria: list | None = None,
|
||||||
|
) -> str | None:
|
||||||
"""Hat ein gebooster field_id genug Keyword-Überlapp mit den
|
"""Hat ein gebooster field_id genug Keyword-Überlapp mit den
|
||||||
pass_criteria einer DB-MC, um den MC zu boost'en?
|
pass_criteria + fail_criteria einer DB-MC, um den MC zu boost'en?
|
||||||
|
|
||||||
Returns: field_id (matched), oder None.
|
Returns: field_id (matched, mit höchstem Keyword-Match-Count), oder None.
|
||||||
Vorsichtig: ≥2 Boost-Keywords müssen im pass_criteria-Text auftauchen,
|
|
||||||
sonst zu permissiv.
|
Schwelle: ≥2 unique Boost-Keywords im kombinierten Text.
|
||||||
|
Beide criteria-Listen werden berücksichtigt — fail_criteria-Wörter
|
||||||
|
wie 'Keine Adresse angegeben' helfen das MC eindeutig zuzuordnen.
|
||||||
"""
|
"""
|
||||||
if not boosts or not pass_criteria:
|
if not boosts:
|
||||||
return None
|
return None
|
||||||
crit_text = " ".join(
|
crit_parts: list[str] = []
|
||||||
str(c) for c in pass_criteria if c
|
for c in (pass_criteria or []):
|
||||||
).lower()
|
if c:
|
||||||
|
crit_parts.append(str(c).lower())
|
||||||
|
for c in (fail_criteria or []):
|
||||||
|
if c:
|
||||||
|
crit_parts.append(str(c).lower())
|
||||||
|
if not crit_parts:
|
||||||
|
return None
|
||||||
|
crit_text = " ".join(crit_parts)
|
||||||
best: tuple[int, str] | None = None
|
best: tuple[int, str] | None = None
|
||||||
for field_id in boosts:
|
for field_id in boosts:
|
||||||
kws = BOOST_KEYWORDS.get(field_id) or ()
|
kws = BOOST_KEYWORDS.get(field_id) or ()
|
||||||
|
# zähle UNIQUE hits — gleiches keyword im selben Text zählt einmal
|
||||||
match_count = sum(1 for kw in kws if kw in crit_text)
|
match_count = sum(1 for kw in kws if kw in crit_text)
|
||||||
if match_count >= 2:
|
if match_count >= 2:
|
||||||
if best is None or match_count > best[0]:
|
if best is None or match_count > best[0]:
|
||||||
|
|||||||
@@ -43,44 +43,67 @@ async def run_v3_pipeline(
|
|||||||
logger.info("v3 Layer-0 boosts: %d hits — %s",
|
logger.info("v3 Layer-0 boosts: %d hits — %s",
|
||||||
len(boost_field_ids), boost_field_ids)
|
len(boost_field_ids), boost_field_ids)
|
||||||
|
|
||||||
# Layer 1+2: bestehender rag_document_checker (Keyword + Embedding)
|
# Layer 1: lade ALLE 75 doc_check_controls für 'impressum' direkt
|
||||||
try:
|
# aus DB. Sidecar-Klassifizierung wird bewusst übersprungen — der
|
||||||
from compliance.services.rag_document_checker import (
|
# Agent soll auf der vollen MC-Liste arbeiten (Layer 3 LLM-Validator
|
||||||
check_document_with_controls,
|
# demoted Pattern-Misses zu LOW, sodass Breitenwirkung kein Risiko ist).
|
||||||
)
|
controls = await _load_impressum_mcs()
|
||||||
results = await check_document_with_controls(
|
results: list[dict[str, Any]] = []
|
||||||
text=text,
|
if controls:
|
||||||
doc_type="impressum",
|
try:
|
||||||
doc_title="Impressum (Agent-Test)",
|
from compliance.services.rag_document_checker import (
|
||||||
db_url=db_url,
|
_check_mc_deterministic,
|
||||||
max_controls=0,
|
)
|
||||||
use_agent=False,
|
text_lower = text.lower().replace("\xad", "")
|
||||||
business_scope=business_scope,
|
for mc in controls:
|
||||||
)
|
r = _check_mc_deterministic(text_lower, mc)
|
||||||
except Exception as e:
|
if r:
|
||||||
logger.warning("rag_document_checker failed: %s — using boosts only",
|
# pass_criteria im Result behalten für Boost-Layer
|
||||||
e)
|
r["_pass_criteria"] = mc.get("pass_criteria")
|
||||||
results = []
|
r["_fail_criteria"] = mc.get("fail_criteria")
|
||||||
|
results.append(r)
|
||||||
|
except Exception as e:
|
||||||
|
logger.warning("layer-1 keyword check failed: %s", e)
|
||||||
|
results = []
|
||||||
|
|
||||||
|
# Layer 2: Embedding-Match für die failed MCs
|
||||||
|
failed_for_embed = [c for c, r in zip(controls, results)
|
||||||
|
if r and not r.get("passed")]
|
||||||
|
if failed_for_embed:
|
||||||
|
try:
|
||||||
|
from compliance.services.mc_embedding_matcher import (
|
||||||
|
ensure_mc_embeddings, embedding_match,
|
||||||
|
)
|
||||||
|
await ensure_mc_embeddings()
|
||||||
|
semantic_passes = await embedding_match(
|
||||||
|
text, failed_for_embed, doc_type="impressum",
|
||||||
|
)
|
||||||
|
if semantic_passes:
|
||||||
|
for r in results:
|
||||||
|
cid = r.get("control_id")
|
||||||
|
if cid in semantic_passes and not r.get("passed"):
|
||||||
|
r["passed"] = True
|
||||||
|
r["matched_text"] = "[layer-2 embedding match]"
|
||||||
|
r["source"] = (r.get("source") or "") + "+embedding"
|
||||||
|
except Exception as e:
|
||||||
|
logger.warning("layer-2 embedding skipped: %s", e)
|
||||||
|
|
||||||
layer_1_pass = sum(1 for r in results if r.get("passed"))
|
layer_1_pass = sum(1 for r in results if r.get("passed"))
|
||||||
layer_1_fail = sum(1 for r in results
|
layer_1_fail = sum(1 for r in results
|
||||||
if r.get("passed") is False)
|
if r.get("passed") is False)
|
||||||
|
|
||||||
# Layer 0 Override: failed MCs deren pass_criteria zu einem meiner
|
# Layer 0 Override: failed MCs deren pass/fail_criteria zu einem meiner
|
||||||
# gebooster field_ids passt → überschreiben zu PASS
|
# gebooster field_ids passen → überschreiben zu PASS. Wir haben
|
||||||
|
# pass_criteria + fail_criteria in r drin (Layer-1 hat sie behalten).
|
||||||
boost_overrides = 0
|
boost_overrides = 0
|
||||||
for r in results:
|
for r in results:
|
||||||
if r.get("passed"):
|
if r.get("passed"):
|
||||||
continue
|
continue
|
||||||
# rag_document_checker nimmt pass_criteria intern weg vor
|
pass_crit = r.get("_pass_criteria") or []
|
||||||
# dem Return; wir laden sie nochmal (oder bekommen sie via
|
fail_crit = r.get("_fail_criteria") or []
|
||||||
# 'hint'). Hier rufen wir das per Helper.
|
if not pass_crit and not fail_crit:
|
||||||
crit = r.get("_pass_criteria") or []
|
pass_crit = [r.get("hint") or r.get("label") or ""]
|
||||||
if not crit:
|
matched_field = boost_matches_db_mc(boosts, pass_crit, fail_crit)
|
||||||
# Fallback: aus dem Hint (= check_question) Boost-Match
|
|
||||||
# versuchen.
|
|
||||||
crit = [r.get("hint") or ""]
|
|
||||||
matched_field = boost_matches_db_mc(boosts, crit)
|
|
||||||
if matched_field:
|
if matched_field:
|
||||||
r["passed"] = True
|
r["passed"] = True
|
||||||
r["matched_text"] = (
|
r["matched_text"] = (
|
||||||
@@ -102,3 +125,52 @@ async def run_v3_pipeline(
|
|||||||
}
|
}
|
||||||
logger.info("v3 telemetry: %s", telemetry)
|
logger.info("v3 telemetry: %s", telemetry)
|
||||||
return results, telemetry
|
return results, telemetry
|
||||||
|
|
||||||
|
|
||||||
|
async def _load_impressum_mcs() -> list[dict]:
|
||||||
|
"""Lädt alle Impressum-MCs aus compliance.doc_check_controls — ohne
|
||||||
|
Sidecar-Filter. v3_engine nimmt die volle Breite."""
|
||||||
|
try:
|
||||||
|
import json
|
||||||
|
from classroom_engine.database import SessionLocal
|
||||||
|
from sqlalchemy import text as _sa_text
|
||||||
|
db = SessionLocal()
|
||||||
|
try:
|
||||||
|
rows = db.execute(_sa_text(
|
||||||
|
"SELECT id, control_id, control_uuid, title, regulation, "
|
||||||
|
" article, check_question, pass_criteria, "
|
||||||
|
" fail_criteria, severity "
|
||||||
|
"FROM compliance.doc_check_controls "
|
||||||
|
"WHERE doc_type='impressum' "
|
||||||
|
"ORDER BY severity DESC, title"
|
||||||
|
)).fetchall()
|
||||||
|
finally:
|
||||||
|
db.close()
|
||||||
|
out: list[dict] = []
|
||||||
|
for r in rows:
|
||||||
|
def _parse(v):
|
||||||
|
if isinstance(v, list):
|
||||||
|
return v
|
||||||
|
if isinstance(v, str):
|
||||||
|
try:
|
||||||
|
j = json.loads(v)
|
||||||
|
return j if isinstance(j, list) else [v]
|
||||||
|
except Exception:
|
||||||
|
return [v]
|
||||||
|
return []
|
||||||
|
out.append({
|
||||||
|
"id": str(r[0]),
|
||||||
|
"control_id": r[1],
|
||||||
|
"control_uuid": str(r[2]) if r[2] else "",
|
||||||
|
"title": r[3] or "",
|
||||||
|
"regulation": r[4] or "",
|
||||||
|
"article": r[5] or "",
|
||||||
|
"check_question": r[6] or "",
|
||||||
|
"pass_criteria": _parse(r[7]),
|
||||||
|
"fail_criteria": _parse(r[8]),
|
||||||
|
"severity": r[9] or "MEDIUM",
|
||||||
|
})
|
||||||
|
return out
|
||||||
|
except Exception as e:
|
||||||
|
logger.warning("_load_impressum_mcs failed: %s", e)
|
||||||
|
return []
|
||||||
|
|||||||
@@ -70,6 +70,27 @@ def test_boost_matches_db_mc_returns_none_when_unrelated():
|
|||||||
assert boost_matches_db_mc(boosts, pass_crit) is None
|
assert boost_matches_db_mc(boosts, pass_crit) is None
|
||||||
|
|
||||||
|
|
||||||
|
def test_boost_matches_db_mc_uses_fail_criteria():
|
||||||
|
"""Wörter aus fail_criteria sollen die Zuordnung mit unterstützen."""
|
||||||
|
boosts = {"name_anbieter"}
|
||||||
|
pass_crit = ["Sichtbar"]
|
||||||
|
fail_crit = ["Keine Postadresse angegeben", "Adresse fehlt"]
|
||||||
|
matched = boost_matches_db_mc(boosts, pass_crit, fail_crit)
|
||||||
|
assert matched == "name_anbieter"
|
||||||
|
|
||||||
|
|
||||||
|
def test_boost_matches_db_mc_eto_address_case():
|
||||||
|
"""Konkreter ETO-Fall: AUTH-1954-A07 'Postadresse + Geschäftssitz'."""
|
||||||
|
boosts = {"name_anbieter"}
|
||||||
|
pass_crit = [
|
||||||
|
"Vollständige Postadresse (Straße, Hausnummer, PLZ, Ort, Land)",
|
||||||
|
"Oder: Eindeutige Angabe des Geschäftssitzes",
|
||||||
|
"Adresse ist aktuell und korrekt",
|
||||||
|
]
|
||||||
|
matched = boost_matches_db_mc(boosts, pass_crit)
|
||||||
|
assert matched == "name_anbieter"
|
||||||
|
|
||||||
|
|
||||||
def test_boost_keywords_cover_all_field_ids():
|
def test_boost_keywords_cover_all_field_ids():
|
||||||
"""Jedes mcs.py field_id muss in BOOST_KEYWORDS ein Eintrag haben."""
|
"""Jedes mcs.py field_id muss in BOOST_KEYWORDS ein Eintrag haben."""
|
||||||
from compliance.services.specialist_agents.impressum.mcs import MCS
|
from compliance.services.specialist_agents.impressum.mcs import MCS
|
||||||
|
|||||||
@@ -1,4 +1,4 @@
|
|||||||
"""Tests für Cookie-Policy-Agent."""
|
"""Tests für Cookie-Policy-Agent v3 (Sprint 1.12 Phase 2)."""
|
||||||
|
|
||||||
from __future__ import annotations
|
from __future__ import annotations
|
||||||
|
|
||||||
@@ -22,122 +22,123 @@ Wir verwenden auf unserer Website verschiedene Cookies. Diese werden
|
|||||||
in folgende Kategorien eingeteilt:
|
in folgende Kategorien eingeteilt:
|
||||||
|
|
||||||
1. Essentielle Cookies (unbedingt erforderlich)
|
1. Essentielle Cookies (unbedingt erforderlich)
|
||||||
Zweck: Diese Cookies dienen der grundlegenden Funktion der Website.
|
Zweck: grundlegende Funktion der Website.
|
||||||
Rechtsgrundlage: § 25 Abs. 2 TDDDG
|
Rechtsgrundlage: § 25 Abs. 2 TDDDG
|
||||||
Laufzeit: Session
|
Laufzeit: Session
|
||||||
|
|
||||||
2. Funktionale Cookies
|
2. Funktionale Cookies — Zweck: Präferenzen speichern. Laufzeit: 30 Tage
|
||||||
Zweck: Speichern Ihre Präferenzen wie Sprache und Region.
|
|
||||||
Rechtsgrundlage: Art. 6 Abs. 1 lit. a DSGVO
|
|
||||||
Laufzeit: 30 Tage
|
|
||||||
|
|
||||||
3. Analytics-Cookies (Performance)
|
3. Analytics-Cookies — Drittanbieter: Google LLC, USA
|
||||||
Drittanbieter: Google LLC, USA
|
Cookies: _ga, _gid · Laufzeit: 24 Monate
|
||||||
Zweck: Nutzungsstatistiken erheben.
|
Drittland: USA — Standardvertragsklauseln + DPF
|
||||||
Laufzeit: 24 Monate
|
|
||||||
Cookies: _ga, _gid
|
|
||||||
Drittland: USA — Standardvertragsklauseln + Data Privacy Framework
|
|
||||||
|
|
||||||
4. Marketing-Cookies (Tracking)
|
4. Marketing — Drittanbieter: Meta Platforms Inc.
|
||||||
Drittanbieter: Meta Platforms Inc., USA
|
Cookies: _fbp, _fbc · Laufzeit: 90 Tage
|
||||||
Cookies: _fbp, _fbc
|
|
||||||
Laufzeit: 90 Tage
|
|
||||||
|
|
||||||
Sie können Ihre Cookie-Einstellungen jederzeit ändern über den Link
|
|
||||||
unten oder das Banner erneut öffnen.
|
|
||||||
|
|
||||||
Browser-Einstellungen: Auch in Chrome, Firefox, Safari und Edge
|
|
||||||
können Sie Cookies blockieren oder löschen.
|
|
||||||
|
|
||||||
|
Cookie-Einstellungen jederzeit ändern.
|
||||||
|
Browser-Einstellungen: Chrome, Firefox, Safari, Edge.
|
||||||
Kontakt: datenschutz@example.com
|
Kontakt: datenschutz@example.com
|
||||||
Datenschutzbeauftragter: Max Mustermann
|
Datenschutzbeauftragter: Max Mustermann
|
||||||
"""
|
"""
|
||||||
|
|
||||||
|
|
||||||
GAPPY_POLICY = """Cookies
|
|
||||||
|
|
||||||
Wir verwenden Cookies um die Website zu betreiben.
|
|
||||||
Cookies werden so lange gespeichert wie nötig.
|
|
||||||
"""
|
|
||||||
|
|
||||||
|
|
||||||
def _run(coro):
|
def _run(coro):
|
||||||
return asyncio.get_event_loop().run_until_complete(coro)
|
return asyncio.get_event_loop().run_until_complete(coro)
|
||||||
|
|
||||||
|
|
||||||
def test_agent_is_registered():
|
@pytest.fixture
|
||||||
agent = REGISTRY.get("cookie_policy")
|
def mock_v3_pipeline(monkeypatch):
|
||||||
assert agent is not None
|
"""Mockt run_v3_pipeline für deterministische Tests offline."""
|
||||||
assert agent.doc_type == "cookie"
|
async def _fake(text, scope):
|
||||||
|
results = [
|
||||||
|
{"control_id": "COOKIE-MC-001",
|
||||||
def test_short_text_skipped(monkeypatch):
|
"passed": True, "severity": "MEDIUM",
|
||||||
async def _no_cascade(*a, **kw): return None, []
|
"label": "Cookie-Kategorien benannt",
|
||||||
|
"regulation": "TDDDG", "article": "§ 25",
|
||||||
|
"hint": "", "matched_text": "essentiell", "source": "kw"},
|
||||||
|
{"control_id": "COOKIE-MC-002",
|
||||||
|
"passed": False, "severity": "HIGH",
|
||||||
|
"label": "Versionsdatum / Stand der Policy",
|
||||||
|
"regulation": "DSGVO", "article": "Art. 5",
|
||||||
|
"hint": "Bitte 'Stand: TT.MM.JJJJ' angeben",
|
||||||
|
"matched_text": "", "source": ""},
|
||||||
|
]
|
||||||
|
telemetry = {
|
||||||
|
"layer_0_field_hits": 4,
|
||||||
|
"layer_0_field_ids": ["categories_named", "purpose_described",
|
||||||
|
"retention_duration", "version_date"],
|
||||||
|
"layer_1_pass": 1,
|
||||||
|
"layer_0_boost_overrides": 0,
|
||||||
|
"total_mcs": 2,
|
||||||
|
}
|
||||||
|
return results, telemetry
|
||||||
monkeypatch.setattr(
|
monkeypatch.setattr(
|
||||||
"compliance.services.specialist_agents.cookie_policy.agent.cascade",
|
"compliance.services.specialist_agents.cookie_policy.agent.run_v3_pipeline",
|
||||||
_no_cascade,
|
_fake,
|
||||||
)
|
)
|
||||||
|
async def _no_validator(*a, **kw): return {}
|
||||||
|
monkeypatch.setattr(
|
||||||
|
"compliance.services.specialist_agents.cookie_policy.agent.validate_present",
|
||||||
|
_no_validator,
|
||||||
|
)
|
||||||
|
|
||||||
|
|
||||||
|
def test_agent_is_registered():
|
||||||
|
a = REGISTRY.get("cookie_policy")
|
||||||
|
assert a is not None
|
||||||
|
assert a.doc_type == "cookie"
|
||||||
|
assert a.agent_version == "3.0"
|
||||||
|
|
||||||
|
|
||||||
|
def test_short_text_skipped(mock_v3_pipeline):
|
||||||
agent = CookiePolicyAgent()
|
agent = CookiePolicyAgent()
|
||||||
out = _run(agent.evaluate(AgentInput(doc_type="cookie", text="x")))
|
out = _run(agent.evaluate(AgentInput(doc_type="cookie", text="x")))
|
||||||
assert out.mc_total > 0
|
|
||||||
assert all(c.status == "skipped" for c in out.mc_coverage)
|
assert all(c.status == "skipped" for c in out.mc_coverage)
|
||||||
|
assert not out.findings
|
||||||
|
|
||||||
|
|
||||||
def test_full_policy_has_few_high_findings(monkeypatch):
|
def test_agent_uses_db_mcs(mock_v3_pipeline):
|
||||||
async def _no_cascade(*a, **kw): return None, []
|
|
||||||
monkeypatch.setattr(
|
|
||||||
"compliance.services.specialist_agents.cookie_policy.agent.cascade",
|
|
||||||
_no_cascade,
|
|
||||||
)
|
|
||||||
agent = CookiePolicyAgent()
|
|
||||||
out = _run(agent.evaluate(AgentInput(doc_type="cookie", text=FULL_POLICY)))
|
|
||||||
high = [f for f in out.findings if f.severity == Severity.HIGH.value]
|
|
||||||
assert not high, f"unexpected HIGH findings: {[f.field_id for f in high]}"
|
|
||||||
|
|
||||||
|
|
||||||
def test_gappy_policy_triggers_high(monkeypatch):
|
|
||||||
async def _no_cascade(*a, **kw): return None, []
|
|
||||||
monkeypatch.setattr(
|
|
||||||
"compliance.services.specialist_agents.cookie_policy.agent.cascade",
|
|
||||||
_no_cascade,
|
|
||||||
)
|
|
||||||
agent = CookiePolicyAgent()
|
agent = CookiePolicyAgent()
|
||||||
out = _run(agent.evaluate(AgentInput(doc_type="cookie",
|
out = _run(agent.evaluate(AgentInput(doc_type="cookie",
|
||||||
text=GAPPY_POLICY)))
|
text=FULL_POLICY)))
|
||||||
field_ids = {f.field_id for f in out.findings}
|
db_findings = [f for f in out.findings
|
||||||
# 4 Kategorien fehlen, Vendoren fehlen, Opt-Out fehlt, Tabelle fehlt
|
if f.check_id.startswith("DBMC-")]
|
||||||
assert "categories_named" in field_ids
|
assert len(db_findings) == 1
|
||||||
assert "vendor_recipients" in field_ids
|
assert db_findings[0].check_id == "DBMC-COOKIE-MC-002"
|
||||||
assert "opt_out_mechanism" in field_ids
|
assert db_findings[0].severity == Severity.HIGH.value
|
||||||
|
|
||||||
|
|
||||||
def test_cmp_vendor_cross_check_emits_finding(monkeypatch):
|
def test_agent_emits_boost_coverage(mock_v3_pipeline):
|
||||||
async def _no_cascade(*a, **kw): return None, []
|
agent = CookiePolicyAgent()
|
||||||
monkeypatch.setattr(
|
out = _run(agent.evaluate(AgentInput(doc_type="cookie",
|
||||||
"compliance.services.specialist_agents.cookie_policy.agent.cascade",
|
text=FULL_POLICY)))
|
||||||
_no_cascade,
|
# 2 DB-MCs + 12 Pattern-Boost-Slots = 14 coverage entries minimum
|
||||||
)
|
assert out.mc_total >= 14
|
||||||
|
boost_ok = [c for c in out.mc_coverage
|
||||||
|
if c.mc_id.startswith("CP-MC-") and c.status == "ok"]
|
||||||
|
assert len(boost_ok) == 4
|
||||||
|
|
||||||
|
|
||||||
|
def test_agent_notes_telemetry(mock_v3_pipeline):
|
||||||
|
agent = CookiePolicyAgent()
|
||||||
|
out = _run(agent.evaluate(AgentInput(doc_type="cookie",
|
||||||
|
text=FULL_POLICY)))
|
||||||
|
assert "v3-pipeline" in out.notes
|
||||||
|
assert "Pattern-Boosts" in out.notes
|
||||||
|
|
||||||
|
|
||||||
|
def test_cmp_vendor_cross_check_emits_finding(mock_v3_pipeline):
|
||||||
|
"""KB-Layer: CMP-Vendoren-Cross-Check bleibt erhalten in v3."""
|
||||||
agent = CookiePolicyAgent()
|
agent = CookiePolicyAgent()
|
||||||
out = _run(agent.evaluate(AgentInput(
|
out = _run(agent.evaluate(AgentInput(
|
||||||
doc_type="cookie", text=FULL_POLICY,
|
doc_type="cookie", text=FULL_POLICY,
|
||||||
context={"cmp_vendors": [
|
context={"cmp_vendors": [
|
||||||
{"name": "Hotjar"}, # NICHT in Policy
|
{"name": "Hotjar"}, # nicht in Policy
|
||||||
{"name": "Google LLC"}, # IN Policy
|
{"name": "Google LLC"}, # in Policy
|
||||||
]},
|
]},
|
||||||
)))
|
)))
|
||||||
field_ids = {f.field_id for f in out.findings}
|
field_ids = {f.field_id for f in out.findings}
|
||||||
assert "vendor_consistency" in field_ids
|
assert "vendor_consistency" in field_ids
|
||||||
cmp_f = next(f for f in out.findings
|
f = next(f for f in out.findings
|
||||||
if f.field_id == "vendor_consistency")
|
if f.field_id == "vendor_consistency")
|
||||||
assert "Hotjar" in cmp_f.evidence
|
assert "Hotjar" in f.evidence
|
||||||
assert "Google" not in cmp_f.evidence
|
|
||||||
|
|
||||||
|
|
||||||
def test_recommendations_are_built():
|
|
||||||
agent = CookiePolicyAgent()
|
|
||||||
out = _run(agent.evaluate(AgentInput(doc_type="cookie",
|
|
||||||
text=GAPPY_POLICY)))
|
|
||||||
assert out.recommendations
|
|
||||||
# Jede Recommendation hat mind. ein related_finding
|
|
||||||
for r in out.recommendations:
|
|
||||||
assert r.related_finding_ids
|
|
||||||
|
|||||||
Reference in New Issue
Block a user