101 lines
3.5 KiB
Python
101 lines
3.5 KiB
Python
"""全量穿透扫描编排(P1.5)。
|
|
|
|
把场景检测器的结果转化为线索,记录扫描覆盖范围(证明全量性)与数据版本(可追溯)。
|
|
当前为同步执行;后续可包装为 Celery 异步任务(接口保持不变)。
|
|
"""
|
|
|
|
from __future__ import annotations
|
|
|
|
import uuid
|
|
from dataclasses import dataclass
|
|
|
|
from sqlalchemy.orm import Session
|
|
|
|
from app.clues import service as clue_svc
|
|
from app.clues.models import Clue
|
|
from app.scenarios import churn_fraud as cf
|
|
from app.scenarios import split_contract as sc
|
|
|
|
MODEL_VERSION = "mock-llm@0.1"
|
|
|
|
|
|
@dataclass
|
|
class ScanResult:
|
|
scenario_code: str
|
|
scanned_count: int
|
|
clue: Clue | None
|
|
|
|
|
|
def run_split_contract_scan(
|
|
session: Session,
|
|
contracts: list[sc.ContractRecord],
|
|
approval_threshold: float,
|
|
shared_controller: bool = False,
|
|
data_version_id: uuid.UUID | None = None,
|
|
) -> ScanResult:
|
|
"""场景一拆单扫描:检测→评分→(命中则)生成线索。"""
|
|
finding = sc.detect_threshold_edge(contracts, approval_threshold)
|
|
score = sc.split_risk_score(finding, shared_controller)
|
|
clue = None
|
|
if score > 0:
|
|
rationale = sc.build_rationale(finding, approval_threshold, shared_controller)
|
|
clue = clue_svc.create_clue(
|
|
session,
|
|
title="疑似政企拆单规避审批",
|
|
risk_domain="收入",
|
|
scenario_code="R8",
|
|
score=score,
|
|
rationale=rationale,
|
|
evidence={
|
|
"near_threshold_contracts": [c.contract_id for c in finding.near_threshold],
|
|
"edge_ratio": finding.ratio,
|
|
"near_threshold_amount": finding.total_amount,
|
|
"approval_threshold": approval_threshold,
|
|
"shared_controller": shared_controller,
|
|
},
|
|
subjects={"customers": sorted({c.customer_key for c in finding.near_threshold})},
|
|
amount_involved=finding.total_amount,
|
|
model_version=MODEL_VERSION,
|
|
data_version_id=data_version_id,
|
|
)
|
|
return ScanResult("R8", len(contracts), clue)
|
|
|
|
|
|
def run_churn_scan(
|
|
session: Session,
|
|
retention_curve: list[cf.CohortPoint],
|
|
commission_paid: float,
|
|
active_ratio: float,
|
|
zero_usage_ratio: float,
|
|
channel_key: str,
|
|
data_version_id: uuid.UUID | None = None,
|
|
) -> ScanResult:
|
|
"""场景二养卡骗补扫描:时序断崖 + 佣金质量不匹配→线索。"""
|
|
finding = cf.detect_pulse_decay(retention_curve)
|
|
mismatch = cf.commission_quality_mismatch(commission_paid, active_ratio, zero_usage_ratio)
|
|
score = cf.churn_risk_score(finding, mismatch)
|
|
clue = None
|
|
if score >= 0.5:
|
|
rationale = cf.build_rationale(finding, mismatch)
|
|
clue = clue_svc.create_clue(
|
|
session,
|
|
title="疑似养卡骗补(脉冲增长+规律退订)",
|
|
risk_domain="成本",
|
|
scenario_code="R9",
|
|
score=score,
|
|
rationale=rationale,
|
|
evidence={
|
|
"cliff_month": finding.cliff_month,
|
|
"max_drop": finding.max_drop,
|
|
"commission_paid": commission_paid,
|
|
"active_ratio": active_ratio,
|
|
"zero_usage_ratio": zero_usage_ratio,
|
|
"mismatch": mismatch,
|
|
},
|
|
subjects={"channel": channel_key},
|
|
amount_involved=commission_paid,
|
|
model_version=MODEL_VERSION,
|
|
data_version_id=data_version_id,
|
|
)
|
|
return ScanResult("R9", len(retention_curve), clue)
|