73f328cb65
PR-Hermes-Docsrv-Search-1 closure 측정 (synthesis_ms=30~48s / ev_ms=15005 / query_analyze 45s) 으로 15s LLM_TIMEOUT 빈발 timeout 확인. Mac mini 26B 동시 호출 (gate Semaphore 1 직렬화 후에도 evidence + synthesis + classifier + query_analyzer + verifier 가 sequential 누적) 시 각 호출 30s 까지 필요. 5곳 변경: - synthesis_service.LLM_TIMEOUT_MS 15000 → 30000 - evidence_service.LLM_TIMEOUT_MS 15000 → 30000 - verifier_service.LLM_TIMEOUT_MS 3000 → 10000 - query_analyzer.LLM_TIMEOUT_MS 15000 → 30000 - search.py:522 classifier wait_for 15.0 → 30.0 (classifier_service align) - search.py:641 verifier wait_for 4.0 → 10.0 (verifier_service align) classifier (이전 PR 에서 30s 로 align 완료) 와 동일 정책 — outer wait_for 가 inner LLM_TIMEOUT_MS 를 override 하지 않도록 align. ask 응답 latency 상한 ↑ 의도된 trade-off — 안정성 (refusal_gate conservative_refuse 회피 + grounding/verifier 정상 동작) 우선. 영향: PR-1 fixture 회귀 0 예상 (이전 timeout 이 새 한도 안). B-1 Throughput-1 (priority queue / 모델 분리) 별 PR 진입 시 latency 본격 단축 검토.
195 lines
6.9 KiB
Python
195 lines
6.9 KiB
Python
"""Exaone semantic verifier (Phase 3.5b).
|
|
|
|
답변-근거 간 의미적 모순(contradiction) 감지. rule-based grounding_check 가 못 잡는
|
|
미묘한 모순 포착. classifier 와 동일 패턴: circuit breaker + timeout + fail open.
|
|
|
|
## Severity 3단계
|
|
- strong: direct_negation (완전 모순) → re-gate 교차 자격
|
|
- medium: numeric_conflict, intent_core_mismatch → confidence 하향 (누적 시 강제 low)
|
|
- weak: nuance, unsupported_claim → 로깅 + mild confidence 하향
|
|
|
|
## 핵심 원칙
|
|
- **Verifier strong 단독 refuse 금지** — grounding strong 과 교차해야 refuse
|
|
- **Timeout 3s** — 느리면 없는 게 낫다 (fail open)
|
|
- MLX gate 미사용 (PR #20 이후 Mac mini 26B endpoint — concurrent 안전성 별 검토)
|
|
"""
|
|
|
|
from __future__ import annotations
|
|
|
|
import asyncio
|
|
import os
|
|
import time
|
|
from dataclasses import dataclass, field
|
|
from typing import TYPE_CHECKING, Literal
|
|
|
|
from ai.client import AIClient, _load_prompt, parse_json_response
|
|
from core.config import settings
|
|
from core.utils import setup_logger
|
|
|
|
if TYPE_CHECKING:
|
|
from .evidence_service import EvidenceItem
|
|
|
|
logger = setup_logger("verifier")
|
|
|
|
LLM_TIMEOUT_MS = 10000 # 2026-05-17 B-3: 3s 시 동시 부하 시 verifier 빈발 skip → grounding 약화. Mac mini 26B 가 verifier-style 짧은 LLM call 도 concurrent 호출 시 3s 초과 빈번 — 10s 로 raise
|
|
CIRCUIT_THRESHOLD = 5
|
|
CIRCUIT_RECOVERY_SEC = 60
|
|
|
|
_failure_count = 0
|
|
_circuit_open_until: float | None = None
|
|
|
|
# Phase 3.5 B2: numeric_conflict severity promote 실험.
|
|
# import time 평가 — env 변경 후 process restart 필수 (docker compose restart fastapi).
|
|
# default=0 (off). production 적용은 B3 FP 검증 통과 후만.
|
|
_NUMERIC_PROMOTE = os.getenv("VERIFIER_NUMERIC_PROMOTE", "0") == "1"
|
|
|
|
# severity 매핑 (프롬프트 "critical"/"minor" → 코드 strong/medium/weak)
|
|
# Tier 4 (B2): _NUMERIC_PROMOTE=1 일 때 numeric_conflict critical → strong 으로 격상.
|
|
# minor 는 medium 유지 (FP 위험 분리).
|
|
_SEVERITY_MAP: dict[str, dict[str, Literal["strong", "medium", "weak"]]] = {
|
|
"direct_negation": {"critical": "strong", "minor": "strong"},
|
|
"numeric_conflict": (
|
|
{"critical": "strong", "minor": "medium"} if _NUMERIC_PROMOTE
|
|
else {"critical": "medium", "minor": "medium"}
|
|
),
|
|
"intent_core_mismatch": {"critical": "medium", "minor": "medium"},
|
|
"nuance": {"critical": "weak", "minor": "weak"},
|
|
"unsupported_claim": {"critical": "weak", "minor": "weak"},
|
|
}
|
|
|
|
|
|
@dataclass(slots=True)
|
|
class Contradiction:
|
|
"""개별 모순 발견."""
|
|
type: str # direct_negation / numeric_conflict / intent_core_mismatch / nuance / unsupported_claim
|
|
severity: Literal["strong", "medium", "weak"]
|
|
claim: str
|
|
evidence_ref: str
|
|
explanation: str
|
|
|
|
|
|
@dataclass(slots=True)
|
|
class VerifierResult:
|
|
status: Literal["ok", "timeout", "error", "circuit_open", "skipped"]
|
|
contradictions: list[Contradiction]
|
|
elapsed_ms: float
|
|
|
|
|
|
try:
|
|
VERIFIER_PROMPT = _load_prompt("verifier.txt")
|
|
except FileNotFoundError:
|
|
VERIFIER_PROMPT = ""
|
|
logger.warning("verifier.txt not found — verifier will always skip")
|
|
|
|
|
|
def _build_input(
|
|
answer: str,
|
|
evidence: list[EvidenceItem],
|
|
) -> str:
|
|
"""답변 + evidence spans → 프롬프트."""
|
|
spans = "\n\n".join(
|
|
f"[{e.n}] {(e.title or '').strip()}\n{e.span_text}"
|
|
for e in evidence
|
|
)
|
|
return (
|
|
VERIFIER_PROMPT
|
|
.replace("{answer}", answer)
|
|
.replace("{numbered_evidence}", spans)
|
|
)
|
|
|
|
|
|
def _map_severity(ctype: str, raw_severity: str) -> Literal["strong", "medium", "weak"]:
|
|
"""type + raw severity → 코드 severity 3단계."""
|
|
type_map = _SEVERITY_MAP.get(ctype, {"critical": "weak", "minor": "weak"})
|
|
return type_map.get(raw_severity, "weak")
|
|
|
|
|
|
async def verify(
|
|
query: str,
|
|
answer: str,
|
|
evidence: list[EvidenceItem],
|
|
) -> VerifierResult:
|
|
"""답변-근거 semantic 검증. Parallel with grounding_check.
|
|
|
|
Returns:
|
|
VerifierResult. status "ok" 이 아니면 contradictions 빈 리스트 (fail open).
|
|
"""
|
|
global _failure_count, _circuit_open_until
|
|
t_start = time.perf_counter()
|
|
|
|
if _circuit_open_until and time.time() < _circuit_open_until:
|
|
return VerifierResult("circuit_open", [], 0.0)
|
|
|
|
if not VERIFIER_PROMPT:
|
|
return VerifierResult("skipped", [], 0.0)
|
|
|
|
if not hasattr(settings.ai, "verifier") or settings.ai.verifier is None:
|
|
return VerifierResult("skipped", [], 0.0)
|
|
|
|
if not answer or not evidence:
|
|
return VerifierResult("skipped", [], 0.0)
|
|
|
|
prompt = _build_input(answer, evidence)
|
|
client = AIClient()
|
|
try:
|
|
async with asyncio.timeout(LLM_TIMEOUT_MS / 1000):
|
|
raw = await client._request(settings.ai.verifier, prompt)
|
|
_failure_count = 0
|
|
except asyncio.TimeoutError:
|
|
_failure_count += 1
|
|
if _failure_count >= CIRCUIT_THRESHOLD:
|
|
_circuit_open_until = time.time() + CIRCUIT_RECOVERY_SEC
|
|
logger.error(f"verifier circuit OPEN for {CIRCUIT_RECOVERY_SEC}s")
|
|
logger.warning("verifier timeout")
|
|
return VerifierResult(
|
|
"timeout", [],
|
|
(time.perf_counter() - t_start) * 1000,
|
|
)
|
|
except Exception as e:
|
|
_failure_count += 1
|
|
if _failure_count >= CIRCUIT_THRESHOLD:
|
|
_circuit_open_until = time.time() + CIRCUIT_RECOVERY_SEC
|
|
logger.error(f"verifier circuit OPEN for {CIRCUIT_RECOVERY_SEC}s")
|
|
logger.warning(f"verifier error: {e}")
|
|
return VerifierResult(
|
|
"error", [],
|
|
(time.perf_counter() - t_start) * 1000,
|
|
)
|
|
finally:
|
|
await client.close()
|
|
|
|
elapsed_ms = (time.perf_counter() - t_start) * 1000
|
|
parsed = parse_json_response(raw)
|
|
if not isinstance(parsed, dict):
|
|
logger.warning("verifier parse failed raw=%r", (raw or "")[:200])
|
|
return VerifierResult("error", [], elapsed_ms)
|
|
|
|
# contradiction 파싱
|
|
raw_items = parsed.get("contradictions") or []
|
|
if not isinstance(raw_items, list):
|
|
raw_items = []
|
|
|
|
results: list[Contradiction] = []
|
|
for item in raw_items[:5]:
|
|
if not isinstance(item, dict):
|
|
continue
|
|
ctype = item.get("type", "")
|
|
if ctype not in _SEVERITY_MAP:
|
|
ctype = "unsupported_claim"
|
|
raw_sev = item.get("severity", "minor")
|
|
severity = _map_severity(ctype, raw_sev)
|
|
claim = str(item.get("claim", ""))[:50]
|
|
ev_ref = str(item.get("evidence_ref", ""))[:50]
|
|
explanation = str(item.get("explanation", ""))[:30]
|
|
results.append(Contradiction(ctype, severity, claim, ev_ref, explanation))
|
|
|
|
logger.info(
|
|
"verifier ok query=%r contradictions=%d strong=%d medium=%d elapsed_ms=%.0f",
|
|
query[:60],
|
|
len(results),
|
|
sum(1 for c in results if c.severity == "strong"),
|
|
sum(1 for c in results if c.severity == "medium"),
|
|
elapsed_ms,
|
|
)
|
|
return VerifierResult("ok", results, elapsed_ms)
|