Merge remote-tracking branch 'gitea/main' into codex/security-supply-chain-contracts-20260512

This commit is contained in:
Your Name
2026-05-17 22:27:00 +08:00
8 changed files with 163 additions and 15 deletions

View File

@@ -97,7 +97,7 @@ class DecideApprovalResponse(BaseModel):
response_model=ListRunsResponse,
summary="列出 Runs",
description=(
"返回 awooop_run_state 記錄,支援 project_id / state / remediation_status filter 與分頁。\n\n"
"返回 awooop_run_state 記錄,支援 project_id / state / remediation_status / incident_id filter 與分頁。\n\n"
"- 按 created_at DESC 排序\n"
"- 注意:此路徑為 /runs/list 以避免與 runs.py 的 /runs/{run_id} 衝突"
),
@@ -109,6 +109,7 @@ async def list_runs(
None,
description="AI 補救證據狀態 filterno_evidence/read_only_dry_run/write_observed/blocked/observed",
),
incident_id: str | None = Query(None, description="關聯 Incident ID filter可選"),
page: int = Query(1, ge=1, description="頁碼,從 1 開始"),
per_page: int = Query(_DEFAULT_PER_PAGE, ge=1, le=_MAX_PER_PAGE, description="每頁筆數"),
) -> dict[str, Any]:
@@ -116,6 +117,7 @@ async def list_runs(
project_id=project_id,
state=state,
remediation_status=remediation_status,
incident_id=incident_id,
page=page,
per_page=per_page,
)

View File

@@ -141,11 +141,13 @@ async def list_runs(
project_id: str | None,
state: str | None,
remediation_status: str | None,
incident_id: str | None,
page: int,
per_page: int,
) -> dict[str, Any]:
"""列出 runs支援 project_id、state、remediation_status filter 與分頁。"""
"""列出 runs支援 project_id、state、remediation_status、incident_id filter 與分頁。"""
_validate_remediation_status_filter(remediation_status)
_validate_incident_id_filter(incident_id)
async with get_db_context("awoooi") as db:
stmt = select(AwoooPRunState).order_by(AwoooPRunState.created_at.desc())
@@ -155,7 +157,7 @@ async def list_runs(
stmt = stmt.where(AwoooPRunState.state == state)
offset = (page - 1) * per_page
if remediation_status:
if remediation_status or incident_id:
result = await db.execute(stmt)
candidate_rows = list(result.scalars().all())
context_limit = _list_filter_context_limit(len(candidate_rows))
@@ -176,6 +178,10 @@ async def list_runs(
remediation_summaries.get(row.run_id),
remediation_status,
)
and _remediation_summary_matches_incident_id(
remediation_summaries.get(row.run_id),
incident_id,
)
]
total = len(filtered_rows)
rows = filtered_rows[offset : offset + per_page]
@@ -521,14 +527,36 @@ def _validate_remediation_status_filter(value: str | None) -> None:
)
def _validate_incident_id_filter(value: str | None) -> None:
if value is None:
return
if not _INCIDENT_ID_RE.fullmatch(value):
raise HTTPException(
status_code=status.HTTP_422_UNPROCESSABLE_ENTITY,
detail="incident_id 格式錯誤,必須是 INC-YYYYMMDD-XXXX",
)
def _remediation_summary_matches_status(
summary: dict[str, Any] | None,
remediation_status: str,
remediation_status: str | None,
) -> bool:
if remediation_status is None:
return True
status_value = str((summary or {}).get("status") or "no_evidence")
return status_value == remediation_status
def _remediation_summary_matches_incident_id(
summary: dict[str, Any] | None,
incident_id: str | None,
) -> bool:
if incident_id is None:
return True
incident_ids = (summary or {}).get("incident_ids")
return isinstance(incident_ids, list) and incident_id in incident_ids
async def _build_run_remediation_summaries(
*,
runs: list[AwoooPRunState],

View File

@@ -30,6 +30,7 @@ import os
import re
from dataclasses import dataclass
from datetime import UTC, datetime
from urllib.parse import quote
from uuid import NAMESPACE_URL, UUID, uuid5
import httpx
@@ -71,6 +72,7 @@ _TELEGRAM_BOT_URL_RE = re.compile(r"(api\.telegram\.org/bot)[^/\s]+")
_INCIDENT_ID_RE = re.compile(r"\bINC-\d{8}-[A-Z0-9]{4,}\b")
_CODE_REF_RE = re.compile(r"<code>([0-9a-f]{7,12})</code>", re.IGNORECASE)
_TELEGRAM_HTML_CHUNK_LIMIT = 3600
_AWOOOP_WEB_BASE_URL = "https://awoooi.wooo.work"
def _top_gateway_bucket(
@@ -196,6 +198,23 @@ def _format_remediation_history_lines(history: dict[str, object] | None) -> list
]
def _awooop_runs_url_for_incident(incident_id: str) -> str:
safe_incident_id = quote(str(incident_id or ""), safe="")
return (
f"{_AWOOOP_WEB_BASE_URL}/zh-TW/awooop/runs"
f"?project_id=awoooi&incident_id={safe_incident_id}"
)
def _awooop_runs_button_row(incident_id: str) -> list[dict[str, str]]:
if not incident_id:
return []
return [{
"text": "🧭 AwoooP",
"url": _awooop_runs_url_for_incident(incident_id),
}]
def _latest_remediation_history_item(history: dict[str, object] | None) -> dict[str, object]:
if not history:
return {}
@@ -2187,6 +2206,10 @@ class TelegramGateway:
tuning_nonce = self._security.generate_callback_nonce(approval_id, "tune")
buttons.append([{"text": "⚡ 執行自動調優", "callback_data": tuning_nonce}])
awooop_row = _awooop_runs_button_row(incident_id)
if awooop_row:
buttons.append(awooop_row)
return {"inline_keyboard": buttons}
# ── YAML Fallback 路徑(原有邏輯,不改動任何行為)────────────────────
@@ -2232,6 +2255,9 @@ class TelegramGateway:
{"text": "📋 詳情", "callback_data": f"detail:{incident_id}"},
{"text": "🔕 忽略", "callback_data": silence_nonce},
])
awooop_row = _awooop_runs_button_row(incident_id)
if awooop_row:
rows.append(awooop_row)
buttons = rows
else:
# 舊版通用鍵(向下相容)
@@ -2249,6 +2275,9 @@ class TelegramGateway:
{"text": "🔄 重診", "callback_data": f"reanalyze:{incident_id}"},
{"text": "📊 歷史", "callback_data": f"history:{incident_id}"},
])
awooop_row = _awooop_runs_button_row(incident_id)
if awooop_row:
buttons.append(awooop_row)
logger.info(
"telegram_keyboard_built",
@@ -2837,12 +2866,14 @@ class TelegramGateway:
# read-only 查類按鈕2-part info 格式handler 已在 handle_callback 實作)
# detail/history 均在 INFO_ACTIONS 白名單,無 nonce 無副作用
keyboard = {
"inline_keyboard": [[
{"text": "📋 詳情", "callback_data": f"detail:{incident_id}"},
{"text": "📊 歷史", "callback_data": f"history:{incident_id}"},
]]
}
inline_keyboard = [[
{"text": "📋 詳情", "callback_data": f"detail:{incident_id}"},
{"text": "📊 歷史", "callback_data": f"history:{incident_id}"},
]]
awooop_row = _awooop_runs_button_row(incident_id)
if awooop_row:
inline_keyboard.append(awooop_row)
keyboard = {"inline_keyboard": inline_keyboard}
return await self._send_request(
"sendMessage",
{
@@ -5189,13 +5220,17 @@ class TelegramGateway:
# Step 1: 換掉按鈕 (移除 Row 1 批准/拒絕/靜默,保留 Row 2 資訊按鈕)
if keep_info_buttons:
new_keyboard = {"inline_keyboard": [
inline_keyboard = [
[
{"text": "📋 詳情", "callback_data": f"detail:{incident_id}"},
{"text": "🔄 重診", "callback_data": f"reanalyze:{incident_id}"},
{"text": "📊 歷史", "callback_data": f"history:{incident_id}"},
],
]}
]
awooop_row = _awooop_runs_button_row(incident_id)
if awooop_row:
inline_keyboard.append(awooop_row)
new_keyboard = {"inline_keyboard": inline_keyboard}
else:
new_keyboard = {"inline_keyboard": []}
@@ -6942,6 +6977,9 @@ class TelegramGateway:
{"text": "📋 詳情", "callback_data": f"detail:{incident_id}"},
{"text": "📊 歷史", "callback_data": f"history:{incident_id}"},
]]
awooop_row = _awooop_runs_button_row(incident_id)
if awooop_row:
info_buttons.append(awooop_row)
await self._send_request(
"editMessageReplyMarkup",
{

View File

@@ -6,6 +6,7 @@ from src.services.platform_operator_service import (
_list_filter_context_limit,
_outbound_timeline_title,
_run_remediation_list_summary,
_remediation_summary_matches_incident_id,
_remediation_summary_matches_status,
_remediation_timeline_summary,
_timeline_sort_key,
@@ -172,6 +173,18 @@ def test_remediation_summary_matches_status_filter() -> None:
assert _remediation_summary_matches_status(None, "no_evidence")
def test_remediation_summary_matches_incident_id_filter() -> None:
assert _remediation_summary_matches_incident_id(
{"incident_ids": ["INC-20260514-F85F21"]},
"INC-20260514-F85F21",
)
assert not _remediation_summary_matches_incident_id(
{"incident_ids": ["INC-20260514-F85F21"]},
"INC-20260513-79ED5E",
)
assert _remediation_summary_matches_incident_id(None, None)
def test_list_filter_context_limit_scales_with_candidate_rows() -> None:
assert _list_filter_context_limit(2) == 500
assert _list_filter_context_limit(4176) == 16704

View File

@@ -71,6 +71,42 @@ def test_telegram_html_chunks_preserve_complete_lines() -> None:
assert all(chunk.count("<b>") == chunk.count("</b>") for chunk in chunks)
def test_awooop_runs_url_for_incident_uses_public_incident_filter() -> None:
"""Telegram URL button 必須導到公開 AwoooP Run list並帶 incident filter。"""
url = telegram_gateway_module._awooop_runs_url_for_incident("INC-20260514-F85F21")
assert url == (
"https://awoooi.wooo.work/zh-TW/awooop/runs"
"?project_id=awoooi&incident_id=INC-20260514-F85F21"
)
@pytest.mark.asyncio
async def test_build_inline_keyboard_includes_awooop_deep_link() -> None:
"""主告警卡的 read-only 按鈕列要能回到 AwoooP evidence view。"""
gateway = TelegramGateway()
keyboard = await gateway._build_inline_keyboard(
approval_id="INC-20260514-F85F21",
include_auto_tuning=False,
incident_id="INC-20260514-F85F21",
)
buttons = [
button
for row in keyboard["inline_keyboard"]
for button in row
]
awooop_buttons = [button for button in buttons if button["text"] == "🧭 AwoooP"]
assert awooop_buttons == [{
"text": "🧭 AwoooP",
"url": (
"https://awoooi.wooo.work/zh-TW/awooop/runs"
"?project_id=awoooi&incident_id=INC-20260514-F85F21"
),
}]
@pytest.mark.asyncio
async def test_send_html_line_message_falls_back_to_plain_text_on_parse_error(monkeypatch):
"""Telegram HTML parse 400 時要送純文字 fallback不可回報成歷史查詢失敗。"""

View File

@@ -1804,7 +1804,9 @@
"manualGate": "Next: human approval",
"filters": {
"label": "AI evidence filter",
"all": "All AI evidence"
"all": "All AI evidence",
"incidentLabel": "Incident ID filter",
"incidentPlaceholder": "Enter Incident ID"
},
"statuses": {
"noEvidence": "No dry-run yet",

View File

@@ -1805,7 +1805,9 @@
"manualGate": "下一步:人工審批",
"filters": {
"label": "AI 證據篩選",
"all": "所有 AI 證據"
"all": "所有 AI 證據",
"incidentLabel": "Incident ID 篩選",
"incidentPlaceholder": "輸入 Incident ID"
},
"statuses": {
"noEvidence": "尚無試跑",

View File

@@ -116,6 +116,7 @@ interface RecentEventsResponse {
const API_BASE = process.env.NEXT_PUBLIC_API_URL ?? "";
const PER_PAGE = 50;
const AUTO_REFRESH_INTERVAL = 30_000; // 30 秒
const INCIDENT_ID_FILTER_RE = /^INC-\d{8}-[A-Z0-9]{4,}$/;
const STATE_CONFIG: Record<
RunState,
@@ -518,10 +519,20 @@ export default function RunsPage() {
const [projectFilter, setProjectFilter] = useState<string>("");
const [statusFilter, setStatusFilter] = useState<string>("");
const [evidenceFilter, setEvidenceFilter] = useState<"" | RemediationStatus>("");
const [incidentFilter, setIncidentFilter] = useState<string>("");
const [page, setPage] = useState(1);
const [lastRefresh, setLastRefresh] = useState<Date | null>(null);
const intervalRef = useRef<ReturnType<typeof setInterval> | null>(null);
useEffect(() => {
const params = new URLSearchParams(window.location.search);
const linkedProject = params.get("project_id") || "";
const linkedIncident = params.get("incident_id") || "";
if (linkedProject) setProjectFilter(linkedProject);
if (linkedIncident) setIncidentFilter(linkedIncident);
if (linkedProject || linkedIncident) setPage(1);
}, []);
// 取得租戶清單
useEffect(() => {
fetch(`${API_BASE}/api/v1/platform/tenants`)
@@ -540,6 +551,10 @@ export default function RunsPage() {
if (projectFilter) params.set("project_id", projectFilter);
if (statusFilter) params.set("state", statusFilter);
if (evidenceFilter) params.set("remediation_status", evidenceFilter);
const normalizedIncidentFilter = incidentFilter.trim().toUpperCase();
if (INCIDENT_ID_FILTER_RE.test(normalizedIncidentFilter)) {
params.set("incident_id", normalizedIncidentFilter);
}
params.set("page", String(page));
params.set("per_page", String(PER_PAGE));
@@ -572,7 +587,7 @@ export default function RunsPage() {
} finally {
setLoading(false);
}
}, [projectFilter, statusFilter, evidenceFilter, page]);
}, [projectFilter, statusFilter, evidenceFilter, incidentFilter, page]);
// 初次載入
useEffect(() => {
@@ -791,6 +806,18 @@ export default function RunsPage() {
</select>
<ChevronDown className="absolute right-2 top-1/2 -translate-y-1/2 w-3.5 h-3.5 text-muted-foreground pointer-events-none" aria-hidden="true" />
</div>
{/* Incident Filter */}
<input
value={incidentFilter}
onChange={(e) => {
setIncidentFilter(e.target.value.trim().toUpperCase());
setPage(1);
}}
className="min-w-[260px] rounded-lg border border-border bg-background px-3 py-1.5 font-mono text-sm text-foreground focus:outline-none focus:ring-2 focus:ring-brand-accent/50"
aria-label={tEvidence("filters.incidentLabel")}
placeholder={tEvidence("filters.incidentPlaceholder")}
/>
</div>
{/* Error State */}