feat(awooop): summarize callback capture in runs list
All checks were successful
CD Pipeline / tests (push) Successful in 38s
Code Review / ai-code-review (push) Successful in 12s
CD Pipeline / build-and-deploy (push) Successful in 3m49s
CD Pipeline / post-deploy-checks (push) Successful in 1m43s

This commit is contained in:
Your Name
2026-05-25 11:16:27 +08:00
parent 814a44d539
commit e1e640f5d5
5 changed files with 187 additions and 0 deletions

View File

@@ -1078,6 +1078,24 @@ def _callback_reply_evidence_capture_status(
}
def _callback_reply_capture_status_from_outbound(
row: AwoooPOutboundMessage,
callback_reply: Mapping[str, Any],
) -> dict[str, Any]:
"""Build capture status directly from one outbound source envelope."""
source_envelope = _as_dict(row.source_envelope)
return _callback_reply_evidence_capture_status(
callback_reply=callback_reply,
persisted_awooop_status_chain=(
_as_dict(source_envelope.get("awooop_status_chain")) or None
),
persisted_km_stale_completion_summary=(
_as_dict(source_envelope.get("km_stale_completion_summary")) or None
),
event_at=row.sent_at or row.queued_at,
)
def _callback_reply_event_item(row: Mapping[str, Any]) -> dict[str, Any]:
"""Convert one callback reply outbound row into a read-only evidence item."""
callback_reply = _as_dict(row.get("callback_reply"))
@@ -1481,6 +1499,13 @@ def _run_callback_reply_summary(
"latest_incident_id": None,
"latest_at": None,
"latest_provider_message_id": None,
"capture_status": "no_callback",
"capture_captured": 0,
"capture_partial": 0,
"capture_not_captured": 0,
"latest_capture_status": None,
"latest_capture_missing": [],
"latest_capture_next_action": None,
}
sorted_rows = sorted(
@@ -1496,6 +1521,26 @@ def _run_callback_reply_summary(
failed = statuses.count("callback_reply_failed")
latest_status = str(latest_callback.get("status") or "")
summary_status = _callback_reply_public_status(latest_callback)
capture_rows = [
_callback_reply_capture_status_from_outbound(row, callback)
for row, callback in sorted_rows
]
capture_statuses = [
str(capture.get("status") or "observed")
for capture in capture_rows
]
capture_not_captured = capture_statuses.count("not_captured")
capture_partial = capture_statuses.count("partial")
capture_captured = capture_statuses.count("captured")
latest_capture = capture_rows[0] if capture_rows else {}
if capture_not_captured > 0:
capture_status = "not_captured"
elif capture_partial > 0:
capture_status = "partial"
elif capture_captured > 0 and capture_captured == len(capture_rows):
capture_status = "captured"
else:
capture_status = "observed"
return {
"schema_version": "awooop_run_callback_reply_summary_v1",
@@ -1511,6 +1556,13 @@ def _run_callback_reply_summary(
"latest_incident_id": latest_callback.get("incident_id"),
"latest_at": latest_row.sent_at or latest_row.queued_at,
"latest_provider_message_id": latest_row.provider_message_id,
"capture_status": capture_status,
"capture_captured": capture_captured,
"capture_partial": capture_partial,
"capture_not_captured": capture_not_captured,
"latest_capture_status": latest_capture.get("status"),
"latest_capture_missing": latest_capture.get("missing") or [],
"latest_capture_next_action": latest_capture.get("next_action"),
}

View File

@@ -270,6 +270,12 @@ def test_run_callback_reply_summary_marks_latest_fallback() -> None:
assert summary["latest_incident_id"] == "INC-20260513-79ED5E"
assert summary["latest_provider_message_id"] == "101"
assert summary["needs_human"] is False
assert summary["capture_status"] == "not_captured"
assert summary["capture_not_captured"] == 2
assert summary["latest_capture_missing"] == [
"awooop_status_chain",
"km_stale_completion_summary",
]
def test_run_callback_reply_summary_marks_failed_as_human_attention() -> None:
@@ -291,6 +297,54 @@ def test_run_callback_reply_summary_marks_failed_as_human_attention() -> None:
assert summary["status"] == "failed"
assert summary["failed"] == 1
assert summary["needs_human"] is True
assert summary["capture_status"] == "not_captured"
def test_run_callback_reply_summary_counts_capture_statuses() -> None:
summary = _run_callback_reply_summary([
SimpleNamespace(
source_envelope={
"callback_reply": {
"status": "callback_reply_sent",
"action": "detail",
"incident_id": "INC-20260513-79ED5E",
},
"awooop_status_chain": {
"schema_version": "awooop_status_chain_callback_reply_snapshot_v1",
},
"km_stale_completion_summary": {
"schema_version": (
"km_stale_owner_review_callback_reply_snapshot_v1"
),
},
},
sent_at=datetime(2026, 5, 18, 6, 3, 0),
queued_at=datetime(2026, 5, 18, 6, 3, 0),
provider_message_id="102",
),
SimpleNamespace(
source_envelope={
"callback_reply": {
"status": "callback_reply_sent",
"action": "history",
"incident_id": "INC-20260513-79ED5E",
},
"awooop_status_chain": {
"schema_version": "awooop_status_chain_callback_reply_snapshot_v1",
},
},
sent_at=datetime(2026, 5, 18, 6, 4, 0),
queued_at=datetime(2026, 5, 18, 6, 4, 0),
provider_message_id="103",
),
])
assert summary["capture_status"] == "partial"
assert summary["capture_captured"] == 1
assert summary["capture_partial"] == 1
assert summary["capture_not_captured"] == 0
assert summary["latest_capture_status"] == "partial"
assert summary["latest_capture_missing"] == ["km_stale_completion_summary"]
def test_run_callback_reply_summary_marks_no_callback() -> None:
@@ -305,6 +359,7 @@ def test_run_callback_reply_summary_marks_no_callback() -> None:
assert summary["status"] == "no_callback"
assert summary["total"] == 0
assert summary["capture_status"] == "no_callback"
def test_list_runs_response_preserves_callback_reply_summary() -> None:
@@ -336,6 +391,18 @@ def test_list_runs_response_preserves_callback_reply_summary() -> None:
"latest_incident_id": "INC-20260513-79ED5E",
"latest_at": "2026-05-18T07:31:37",
"latest_provider_message_id": "telegram_callback_reply:failed",
"capture_status": "not_captured",
"capture_captured": 0,
"capture_partial": 0,
"capture_not_captured": 1,
"latest_capture_status": "not_captured",
"latest_capture_missing": [
"awooop_status_chain",
"km_stale_completion_summary",
],
"latest_capture_next_action": (
"press_telegram_detail_or_history_after_rollout"
),
},
}
],
@@ -347,6 +414,9 @@ def test_list_runs_response_preserves_callback_reply_summary() -> None:
dumped = response.model_dump(mode="json")
assert dumped["runs"][0]["callback_reply_summary"]["status"] == "failed"
assert dumped["runs"][0]["callback_reply_summary"]["needs_human"] is True
assert dumped["runs"][0]["callback_reply_summary"]["capture_status"] == (
"not_captured"
)
def test_callback_reply_event_item_surfaces_run_link_and_human_flag() -> None:

View File

@@ -2764,6 +2764,18 @@
"emptyShort": "No detail / history callback yet",
"latest": "{action} · {incidentId}",
"needsHuman": "Callback failure needs human review",
"captureLine": "Snapshot: {status}; captured {captured} / partial {partial} / not captured {notCaptured}",
"captureMissing": "Missing: {items}",
"captureStatuses": {
"captured": "Captured",
"partial": "Partially captured",
"not_captured": "Not captured",
"observed": "Recorded"
},
"captureItems": {
"awooopStatusChain": "AwoooP status chain",
"kmCompletionSummary": "KM owner-review snapshot"
},
"filters": {
"label": "TG Callback filter",
"all": "All TG callbacks"

View File

@@ -2765,6 +2765,18 @@
"emptyShort": "尚無詳情 / 歷史 callback",
"latest": "{action} · {incidentId}",
"needsHuman": "Callback 失敗需人工確認",
"captureLine": "Snapshot{status};已捕捉 {captured} / 部分 {partial} / 未捕捉 {notCaptured}",
"captureMissing": "尚缺:{items}",
"captureStatuses": {
"captured": "已捕捉",
"partial": "部分捕捉",
"not_captured": "未捕捉",
"observed": "已記錄"
},
"captureItems": {
"awooopStatusChain": "AwoooP 狀態鏈",
"kmCompletionSummary": "KM owner-review snapshot"
},
"filters": {
"label": "TG Callback 篩選",
"all": "所有 TG Callback"

View File

@@ -117,6 +117,13 @@ interface CallbackReplySummary {
latest_incident_id?: string | null;
latest_at?: string | null;
latest_provider_message_id?: string | null;
capture_status?: CallbackEvidenceCaptureState | "no_callback" | string | null;
capture_captured?: number;
capture_partial?: number;
capture_not_captured?: number;
latest_capture_status?: CallbackEvidenceCaptureState | string | null;
latest_capture_missing?: string[];
latest_capture_next_action?: string | null;
}
interface CallbackEvidenceCaptureStatus {
@@ -908,8 +915,29 @@ function CallbackReplyCell({ summary }: { summary?: CallbackReplySummary | null
const status = normalizeCallbackReplyStatus(summary);
const config = CALLBACK_REPLY_CONFIG[status];
const total = summary?.total ?? 0;
const captureStatus = total > 0
? normalizeCallbackEvidenceCaptureStatus(summary?.capture_status)
: null;
const captureConfig = captureStatus
? {
captured: "text-[#17602a]",
partial: "text-[#8a5a08]",
not_captured: "text-[#9f2f25]",
observed: "text-[#1f5b9b]",
}[captureStatus]
: null;
const latestAction = summary?.latest_action ? String(summary.latest_action) : null;
const latestIncidentId = summary?.latest_incident_id ? String(summary.latest_incident_id) : null;
const captureItemLabels: Record<string, string> = {
awooop_status_chain: t("captureItems.awooopStatusChain"),
km_stale_completion_summary: t("captureItems.kmCompletionSummary"),
};
const latestCaptureMissing = Array.isArray(summary?.latest_capture_missing)
? summary.latest_capture_missing
: [];
const latestCaptureMissingText = latestCaptureMissing.length > 0
? latestCaptureMissing.map((item) => captureItemLabels[item] ?? item).join(" / ")
: null;
const countText = total > 0
? t("count", {
total,
@@ -944,6 +972,19 @@ function CallbackReplyCell({ summary }: { summary?: CallbackReplySummary | null
{latestText}
</span>
)}
{captureStatus && (
<span
className={cn("text-xs leading-5", captureConfig)}
title={latestCaptureMissingText ? t("captureMissing", { items: latestCaptureMissingText }) : undefined}
>
{t("captureLine", {
status: t(`captureStatuses.${captureStatus}` as never),
captured: summary?.capture_captured ?? 0,
partial: summary?.capture_partial ?? 0,
notCaptured: summary?.capture_not_captured ?? 0,
})}
</span>
)}
{summary?.needs_human && (
<span className="text-xs font-semibold text-[#9f2f25]">
{t("needsHuman")}