V10.494 add market intel candidate queue review gate

This commit is contained in:
OoO
2026-05-31 13:08:53 +08:00
parent c196a0e228
commit cef27c77df
12 changed files with 1227 additions and 167 deletions

View File

@@ -4,6 +4,7 @@
================================================================================
【已完成】
- V10.494 新增市場情報 MCP Fetch Candidate Queue Review 安全預覽 gate只審核 candidate handoff 後的人工 queue review 草案,要求候選 key 對齊、review_state 停在 needs_review、allowed actions 限定人工確認/否決/延後、queue_write_status 維持 not_persistedAPI 不建立 queue、不更新 review_state、不寫 DB、不連外、不掛 scheduler。
- V10.493 新增市場情報 MCP Fetch Candidate Handoff Review 安全預覽 gate只審核 parser review 後的候選交接包,確認 source/candidate key 對齊、queue policy 仍是 manual preview、候選數維持小批次、無 raw/secret/side-effectAPI 不建立 queue、不寫 DB、不讀 artifact、不連外、不掛 scheduler。
- V10.492 收緊 PChome 近門檻自動回刷隊列:`retryable_candidate_revalidation` 不再把 `identity_veto`、`unit_comparable`、`true_low_confidence` 納入每日自動回刷;只處理 `recoverable_low_score` 與 legacy `low_score / refresh_low_score`,並要求無 hard veto、仍在 `exact_identity`、且具備同品線/identity anchor 證據。這讓「可救回」與「正確阻擋」在操作層面真正分流,避免為了壓低 low_score 而重跑不該自動推進的候選。
- V10.491 新增市場情報 MCP Fetch Result Parser Review 安全預覽 gate只審核操作員貼回的 parser 結構化摘要,對齊 receipt source/path、候選必要欄位、公開 URL、小批次上限與 raw HTML/secret/side-effect 風險API 不讀 artifact、不執行 parser CLI、不抓外站、不寫 DB、不掛 scheduler。

View File

@@ -350,7 +350,7 @@ YOUTUBE_API_KEY = os.getenv('YOUTUBE_API_KEY', '')
# ==========================================
# 系統版本與路徑
# ==========================================
SYSTEM_VERSION = "V10.493"
SYSTEM_VERSION = "V10.494"
LOG_FILE_PATH = os.path.join(BASE_DIR, 'logs/system.log')
public_url = PUBLIC_URL # 用於模板顯示

View File

@@ -165,6 +165,7 @@ EwoooC 目前已有 MOMO EDM / 節慶活動資料、`promo_products`、PChome
- 2026-05-29 追加 MCP fetch run receipt gate`services.market_intel.mcp_fetch_run_receipt``/api/market_intel/mcp_fetch_run_receipt` 在操作員 shell 完成 dry-run fetch 後審核 receipt id、artifact path、platform/source/receipt path 對帳、公開 URL、request/error budget、secret 外洩與 API/DB/scheduler 副作用旗標API/UI 不執行 CLI、不發外部 request、不保存 receipt、不開 DB、不寫入、不掛 scheduler只放行到後續 result parser review。
- 2026-05-31 追加 MCP fetch result parser review gate`services.market_intel.mcp_fetch_result_parser_review``/api/market_intel/mcp_fetch_result_parser_review` 在 receipt 通過後審核操作員貼回的 parser 結構化摘要,檢查 receipt source/receipt path 對帳、parser artifact path、活動/商品候選必要欄位、公開 URL、小批次候選上限、raw HTML/secret 外洩與 API/DB/scheduler 副作用旗標API/UI 不讀 artifact、不執行 parser CLI、不發外部 request、不保存 parser result、不寫入、不掛 scheduler只放行到候選 handoff review。
- 2026-05-31 追加 MCP fetch candidate handoff review gate`services.market_intel.mcp_fetch_candidate_handoff_review``/api/market_intel/mcp_fetch_candidate_handoff_review` 在 parser review 通過後審核候選交接包,檢查 source/candidate key 對齊、queue policy 是否仍是 `manual_candidate_review` / `preview_only`、小批次候選上限、操作員無寫入/無連外/無排程確認、raw HTML/secret 外洩與 side-effect flagsAPI/UI 不建立 queue、不讀 artifact、不寫 DB、不掛 scheduler只放行到人工 candidate queue review。
- 2026-05-31 追加 MCP fetch candidate queue review gate`services.market_intel.mcp_fetch_candidate_queue_review``/api/market_intel/mcp_fetch_candidate_queue_review` 在 handoff review 通過後審核人工 queue review 草案,檢查候選 key 對齊、`review_state=needs_review`、allowed actions 人工限定、`queue_write_status=not_persisted`、操作員無寫入/無連外/無排程確認、raw HTML/secret 外洩與 side-effect flagsAPI/UI 不建立 queue、不更新 review_state、不讀 artifact、不寫 DB、不掛 scheduler只放行到 writer preflight。
- 2026-05-18 追加 scheduler attach plan preview`services.market_intel.scheduler_plan``/api/market_intel/scheduler_plan` 描述未來 `campaign_discovery_daily``campaign_product_probe``product_match_review_seed` 三個 job 的 cadence、gate、fallback 與安全邊界。此階段不註冊 scheduler job、不啟動 crawler、不連外、不寫 DB排程掛載必須等 migration、seed、MCP fetch gate、manual sample 與人工批准全過。
- 2026-05-18 追加 match review plan preview`services.market_intel.match_review_plan``/api/market_intel/match_review_plan` 定義商品比對訊號、分數門檻、`needs_review → confirmed/rejected` HITL 流程與安全邊界。此階段不建立 review queue、不自動 confirmed、不寫 `market_product_matches`、不呼叫 MCP價格只能作為輔助訊號不能單獨決定同品比對。
- 2026-05-18 追加 opportunity plan preview`services.market_intel.opportunity_plan``/api/market_intel/opportunity_plan` 定義競品低價威脅、促銷缺口、深折重疊、活動即將結束四類規則與分級策略。此階段不建立 opportunity queue、不派送 Telegram、不產生 AI 摘要、不寫 DB高風險項必須先有 confirmed match 與 DB evidence 才能升級。

View File

@@ -42,6 +42,7 @@
- 2026-05-24 追記:同步市場情報 MCP fetch target review gate 後的 `routes/market_intel_routes.py``services/market_intel/deployment_readiness.py` 行數;本次新增邏輯已放在獨立 `services/market_intel/mcp_fetch_target_review.py`route 僅保留薄 glue後續市場情報 MCP route 應拆出子 Blueprint 或 route registration helper。
- 2026-05-24 追記:同步市場情報 MCP fetch run package gate 後的 `routes/market_intel_routes.py``services/market_intel/deployment_readiness.py` 行數;本次新增 endpoint 已拆到 `routes/market_intel_mcp_run_routes.py`,主 Blueprint 只新增 extension import後續 MCP route 應延續此模式。
- 2026-05-24 追記:同步市場情報 MCP fetch run readiness gate 後的 `services/market_intel/deployment_readiness.py` 行數;本次新增 endpoint 延續 `routes/market_intel_mcp_run_routes.py` route extension新增邏輯放在獨立 `services/market_intel/mcp_fetch_run_readiness.py`
- 2026-05-31 追記:同步市場情報 MCP fetch candidate queue review gate 後的 `services/market_intel/deployment_readiness.py` 行數;本次新增邏輯維持在獨立 `services/market_intel/mcp_fetch_candidate_queue_review.py`route 延續 `routes/market_intel_mcp_run_routes.py` extension。
- 2026-05-24 追記:同步背景 Code Review 111 fallback 保護合併後的 `services/code_review_pipeline_service.py` 行數;此處只更新 inventory不變更 Code Review 行為。
- 2026-05-21 追記:同步 PChome/LUDEYA 商品線名稱漂移比對更新後的 `services/marketplace_product_matcher.py` 行數;此處只更新 inventory不變更模組化決策。
- 2026-05-21 追記:同步 MAC/Yuskin/AHC 名稱漂移與 bundle equivalent matcher 更新後的 `services/marketplace_product_matcher.py` 行數;此處只更新 inventory不變更模組化決策。
@@ -94,7 +95,7 @@
| 805 | `routes/bot_api_routes.py` | P2 Bot API Blueprint | route glue / bot action service |
| 1319 | `routes/market_intel_review_report_routes.py` | P2 market intel review report Blueprint | review report route glue / export payload / phase handoff orchestration |
| 917 | `routes/market_intel_routes.py` | P2 market intel Blueprint | page route / API route glue / MCP gate route registration helper |
| 1040 | `services/market_intel/deployment_readiness.py` | P2 market intel deployment readiness | preflight gates / readiness payload / route contract helpers |
| 1181 | `services/market_intel/deployment_readiness.py` | P2 market intel deployment readiness | preflight gates / readiness payload / route contract helpers |
| 846 | `services/market_intel/candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_receipt.py` | P2 market intel review receipt pipeline | AI summary / persistence / Telegram dispatch / report catalog run receipt orchestration |
## 市場情報開發前置禁區

View File

@@ -89,6 +89,7 @@
- 2026-05-31 起,`V10.491` 新增市場情報 MCP Fetch Result Parser Review gate在 receipt gate 後只審核操作員 shell parser 貼回的結構化摘要,對齊 source/path、公開 URL、候選必要欄位、小批次上限、raw HTML/secret/side-effect 風險;仍不讀 artifact、不執行 CLI、不連外、不寫 DB、不掛 scheduler。
- 2026-05-31 起,`V10.492` 收緊 PChome 近門檻自動回刷:`run_retryable_candidate_revalidation()` 只回刷 `recoverable_low_score` 與 legacy `low_score / refresh_low_score`,且 SQL 端要求 `hard_veto=false``comparison_mode=exact_identity`、diagnostic reasons 命中同品線/identity anchor`identity_veto``unit_comparable``true_low_confidence` 不再進每日自動回刷隊列,需等待新證據或人工處理。
- 2026-05-31 起,`V10.493` 新增市場情報 MCP Fetch Candidate Handoff Review gate在 parser review 通過後只審核候選交接包,要求 source/candidate key 完全對齊、queue policy 維持 manual preview、小批次上限與操作員無寫入/無連外/無排程確認;仍不建立 queue、不寫 DB、不讀 artifact、不連外、不掛 scheduler。
- 2026-05-31 起,`V10.494` 新增市場情報 MCP Fetch Candidate Queue Review gate在 handoff review 通過後只審核人工 queue review 草案,要求候選 key 完全對齊、review_state 只停在 `needs_review`、allowed actions 限人工操作、queue_write_status 維持 `not_persisted`;仍不建立 queue、不更新 review_state、不寫 DB、不連外、不掛 scheduler。
## 3. 12 Agent 決策信封整合

View File

@@ -13,6 +13,7 @@
## 📅 詳細更新日誌 (考古存檔)
### 2026-05-24PChome 近門檻身份回收第二輪
- **V10.494 市場情報 MCP Fetch Candidate Queue Review gate**: 新增 `/api/market_intel/mcp_fetch_candidate_queue_review` 與 UI preview只審核 candidate handoff 後的人工 queue review 草案;要求候選 key 完全對齊、`review_state=needs_review`、allowed actions 限人工確認/否決/延後、`queue_write_status=not_persisted`,且 API 不建立 queue、不更新 review_state、不寫 DB、不連外、不掛 scheduler。
- **V10.493 市場情報 MCP Fetch Candidate Handoff Review gate**: 新增 `/api/market_intel/mcp_fetch_candidate_handoff_review` 與 UI preview只審核 parser review 後的候選交接包;要求 source/candidate key 完全對齊、queue policy 維持 `manual_candidate_review` / `preview_only`、候選數維持小批次,且 API 不建立 queue、不寫 DB、不讀 artifact、不連外、不掛 scheduler。
- **V10.491 市場情報 MCP Fetch Result Parser Review gate**: 新增 `/api/market_intel/mcp_fetch_result_parser_review` 與 UI preview只審核操作員 shell parser 後貼回的結構化摘要API 不讀 artifact、不執行 parser CLI、不抓外站、不寫檔、不開 DB、不掛 scheduler且會阻擋 raw HTML/body/snapshot、secret/token 欄位與 side-effect flags。
- **V10.489 PChome 低分同款人工覆核回收與 gate-pass 風險邊界**: `marketplace_product_matcher` 新增三個窄範圍 focused identityTS6 超美白香氛誘霜 120g/ml、W 修護保養蝸牛特潤修護面膜 6 片、Derma 大地 Eco 植萃護膚油 2 入。這些樣本只升到 `identity_review / manual_review`,不進 `price_alert_exact`;同版補 Clarins 身體油不同線、命名組合品數量反轉、isLeaf 香型數量不一致 hard vetoHOOOME 大理石暖燈單側設計差留人工覆核。

View File

@@ -22,6 +22,9 @@ from services.market_intel.mcp_fetch_result_parser_review import (
from services.market_intel.mcp_fetch_candidate_handoff_review import (
build_mcp_fetch_candidate_handoff_review_preview,
)
from services.market_intel.mcp_fetch_candidate_queue_review import (
build_mcp_fetch_candidate_queue_review_preview,
)
@market_intel_bp.route("/api/market_intel/mcp_fetch_run_package", methods=["GET", "POST"])
@@ -204,3 +207,46 @@ def market_intel_mcp_fetch_candidate_handoff_review():
phase=service.phase,
)
)
@market_intel_bp.route(
"/api/market_intel/mcp_fetch_candidate_queue_review",
methods=["GET", "POST"],
)
@login_required
def market_intel_mcp_fetch_candidate_queue_review():
handoff_review_package = {}
handoff_review_result = None
candidate_queue_review = None
if request.method == "POST":
payload = request.get_json(silent=True) or {}
package = (
payload.get("candidate_queue_review")
or payload.get("queue_review")
or payload
)
handoff_review_package = (
package.get("handoff_review_package")
or package.get("candidate_handoff_review")
or package.get("handoff_review")
or {}
)
handoff_review_result = (
package.get("handoff_review_result")
or package.get("mcp_fetch_candidate_handoff_review")
)
candidate_queue_review = (
package.get("candidate_queue_review")
or package.get("queue_review")
or package.get("review_payload")
)
service = MarketIntelService()
return jsonify(
build_mcp_fetch_candidate_queue_review_preview(
handoff_review_package=handoff_review_package,
handoff_review_result=handoff_review_result,
candidate_queue_review=candidate_queue_review,
phase=service.phase,
)
)

View File

@@ -78,6 +78,9 @@ from services.market_intel.mcp_fetch_result_parser_review import (
from services.market_intel.mcp_fetch_candidate_handoff_review import (
build_mcp_fetch_candidate_handoff_review_preview,
)
from services.market_intel.mcp_fetch_candidate_queue_review import (
build_mcp_fetch_candidate_queue_review_preview,
)
from services.market_intel.mcp_manual_fetch_handoff import (
build_mcp_manual_fetch_handoff_preview,
)
@@ -205,6 +208,11 @@ PRODUCTION_SMOKE_TARGETS = (
+ ("/api/market_intel/mcp_fetch_candidate_handoff_review",)
+ PRODUCTION_SMOKE_TARGETS[-1:]
)
PRODUCTION_SMOKE_TARGETS = (
PRODUCTION_SMOKE_TARGETS[:-1]
+ ("/api/market_intel/mcp_fetch_candidate_queue_review",)
+ PRODUCTION_SMOKE_TARGETS[-1:]
)
def _run_review_preview_safe(payload, mode):
return bool(payload["mode"] == mode and all(not payload.get(key) for key in BLOCKED_RUN_REVIEW_KEYS))
def build_deployment_readiness_preview(*, service, market_intel_tables, schema_smoke_builder):
@@ -249,6 +257,11 @@ def build_deployment_readiness_preview(*, service, market_intel_tables, schema_s
phase=service.phase,
)
)
mcp_fetch_candidate_queue_review = (
build_mcp_fetch_candidate_queue_review_preview(
phase=service.phase,
)
)
scheduler_plan = service.build_scheduler_plan()
manual_sample_plan = service.build_manual_sample_plan()
manual_sample_acceptance = service.build_manual_sample_acceptance()
@@ -627,6 +640,28 @@ def build_deployment_readiness_preview(*, service, market_intel_tables, schema_s
and not mcp_fetch_candidate_handoff_review["file_written"]
and not mcp_fetch_candidate_handoff_review["scheduler_attached"]
),
"mcp_fetch_candidate_queue_review_preview_safe": bool(
mcp_fetch_candidate_queue_review["mode"]
== "mcp_fetch_candidate_queue_review_preview"
and not mcp_fetch_candidate_queue_review["payload_persisted"]
and not mcp_fetch_candidate_queue_review["candidate_queue_created"]
and not mcp_fetch_candidate_queue_review["candidate_queue_persisted"]
and not mcp_fetch_candidate_queue_review[
"candidate_review_state_updated"
]
and not mcp_fetch_candidate_queue_review["network_request_allowed"]
and not mcp_fetch_candidate_queue_review["api_executes_cli"]
and not mcp_fetch_candidate_queue_review[
"api_opens_database_connection"
]
and not mcp_fetch_candidate_queue_review["api_writes_database"]
and not mcp_fetch_candidate_queue_review["api_uses_external_network"]
and not mcp_fetch_candidate_queue_review["database_write_executed"]
and not mcp_fetch_candidate_queue_review["fetch_executed"]
and not mcp_fetch_candidate_queue_review["cli_executed"]
and not mcp_fetch_candidate_queue_review["file_written"]
and not mcp_fetch_candidate_queue_review["scheduler_attached"]
),
"scheduler_plan_preview_safe": bool(
scheduler_plan["mode"] == "scheduler_attach_plan_preview"
and not scheduler_plan["scheduler_registration_executed"]
@@ -1068,6 +1103,7 @@ def build_deployment_readiness_preview(*, service, market_intel_tables, schema_s
"mcp_fetch_run_receipt": mcp_fetch_run_receipt,
"mcp_fetch_result_parser_review": mcp_fetch_result_parser_review,
"mcp_fetch_candidate_handoff_review": mcp_fetch_candidate_handoff_review,
"mcp_fetch_candidate_queue_review": mcp_fetch_candidate_queue_review,
"scheduler_plan": scheduler_plan,
"manual_sample_plan": manual_sample_plan,
"manual_sample_acceptance": manual_sample_acceptance,

View File

@@ -0,0 +1,636 @@
"""市場情報 MCP fetch candidate queue review preview。
本模組只審核 candidate handoff 後的人工 queue review 草案;
不建立 queue、不寫 DB、不讀 artifact、不發 HTTP request、不掛 scheduler。
"""
from services.market_intel.mcp_fetch_candidate_handoff_review import (
build_mcp_fetch_candidate_handoff_review_preview,
)
MAX_QUEUE_REVIEW_ITEMS = 80
SAFE_REVIEW_STATES = {"needs_review"}
SAFE_REVIEW_MODES = {"operator_queue_review_preview"}
SAFE_TARGET_QUEUES = {"manual_candidate_review"}
SAFE_ALLOWED_ACTIONS = {
"confirm_campaign_candidate",
"confirm_product_candidate",
"reject_candidate",
"defer_candidate",
}
FORBIDDEN_SECRET_KEYS = (
"approval_token",
"approval-token",
"api_key",
"authorization",
"bearer",
"client_secret",
"cookie",
"password",
"secret",
"session_cookie",
"token",
)
SAFE_SECRET_METADATA_KEYS = {
"operator_confirmed_no_secret_payload",
}
FORBIDDEN_RAW_PAYLOAD_KEYS = (
"body_html",
"document_html",
"html",
"page_html",
"raw_body",
"raw_html",
"raw_snapshot",
"response_body",
)
_BLOCKED_SIDE_EFFECT_KEYS = (
"allow_api_execution",
"allow_database_write",
"allow_external_network_in_api",
"allow_scheduler_attach",
"api_executed_cli",
"api_executes_cli",
"api_executes_docker",
"api_executes_health_check",
"api_executes_ssh",
"api_opens_database_connection",
"api_uses_external_network",
"api_writes_database",
"attach_scheduler",
"candidate_queue_created",
"candidate_queue_persisted",
"candidate_review_state_updated",
"cli_executed",
"command_executed",
"database_commit_executed",
"database_session_created",
"database_write_executed",
"external_network_executed",
"fetch_executed",
"file_written",
"network_request_allowed",
"payload_persisted",
"scheduler_attached",
"write_database",
"writes_executed",
"would_write_database",
)
def _as_dict(value):
return value if isinstance(value, dict) else {}
def _as_list(value):
if value is None:
return []
if isinstance(value, (list, tuple, set)):
return list(value)
return [value]
def _safe_int(value):
try:
return int(value or 0)
except (TypeError, ValueError):
return 0
def _has_text(value):
return bool(isinstance(value, str) and value.strip())
def _safe_text(value, limit=500):
if value is None:
return None
text = str(value).strip()
return text[:limit] if text else None
def _contains_forbidden_key(value, forbidden_keys, *, safe_keys=None):
safe_keys = safe_keys or set()
if isinstance(value, dict):
for key, nested in value.items():
normalized_key = str(key).lower()
if normalized_key in safe_keys and isinstance(nested, bool):
continue
if any(forbidden_key in normalized_key for forbidden_key in forbidden_keys):
return True
if _contains_forbidden_key(nested, forbidden_keys, safe_keys=safe_keys):
return True
elif isinstance(value, list):
return any(
_contains_forbidden_key(item, forbidden_keys, safe_keys=safe_keys)
for item in value
)
return False
def _blocked_side_effects(payload):
found = []
def visit(value, path):
if isinstance(value, dict):
for key, item in value.items():
normalized_key = str(key).lower()
key_path = f"{path}.{key}" if path else key
if normalized_key in _BLOCKED_SIDE_EFFECT_KEYS and bool(item):
found.append(key_path)
visit(item, key_path)
elif isinstance(value, list):
for index, item in enumerate(value):
visit(item, f"{path}[{index}]")
visit(payload, "")
return found
def _handoff_review_from_inputs(handoff_review_package, handoff_review_result, phase):
if isinstance(handoff_review_result, dict) and handoff_review_result:
return handoff_review_result
handoff_review_package = _as_dict(handoff_review_package)
return build_mcp_fetch_candidate_handoff_review_preview(
parser_review_package=(
handoff_review_package.get("parser_review_package")
or handoff_review_package.get("parser_review")
or handoff_review_package.get("result_parser_review")
),
parser_review_result=(
handoff_review_package.get("parser_review_result")
or handoff_review_package.get("mcp_fetch_result_parser_review")
),
candidate_handoff=(
handoff_review_package.get("candidate_handoff")
or handoff_review_package.get("handoff")
or handoff_review_package.get("handoff_payload")
),
phase=phase,
)
def _candidate_key(candidate, candidate_type):
candidate = _as_dict(candidate)
if candidate_type == "campaign":
return candidate.get("campaign_key")
return candidate.get("platform_product_id")
def _sample_candidate_queue_review_package():
handoff_preview = build_mcp_fetch_candidate_handoff_review_preview()
handoff_review_package = handoff_preview["sample_candidate_handoff_package"]
handoff_review_result = build_mcp_fetch_candidate_handoff_review_preview(
parser_review_package=handoff_review_package["parser_review_package"],
parser_review_result=handoff_review_package["parser_review_result"],
candidate_handoff=handoff_review_package["candidate_handoff"],
)
handoff_summary = handoff_review_result["candidate_handoff_summary"]
review_items = []
for candidate_type, candidates in (
("campaign", handoff_summary.get("campaign_candidates", [])),
("product", handoff_summary.get("product_candidates", [])),
):
for item in candidates:
item = _as_dict(item)
candidate_key = _candidate_key(item, candidate_type)
review_items.append(
{
"candidate_type": candidate_type,
"platform_code": item.get("platform_code"),
"source_key": item.get("source_key"),
"candidate_key": candidate_key,
"candidate_name": item.get("campaign_name") or item.get("name"),
"candidate_url": item.get("campaign_url") or item.get("product_url"),
"review_state": "needs_review",
"priority_lane": "normal",
"allowed_actions": sorted(SAFE_ALLOWED_ACTIONS),
"evidence_ref": "parser_summary",
"queue_write_status": "not_persisted",
}
)
return {
"handoff_review_package": handoff_review_package,
"handoff_review_result": handoff_review_result,
"candidate_queue_review": {
"review_id": "market-intel-candidate-queue-review-sample",
"review_mode": "operator_queue_review_preview",
"target_queue": "manual_candidate_review",
"operator_confirmed_no_database_write": True,
"operator_confirmed_no_external_network": True,
"operator_confirmed_no_scheduler_attach": True,
"operator_confirmed_no_persistence": True,
"operator_confirmed_manual_review_required": True,
"operator_confirmed_no_secret_payload": True,
"summary": {
"campaign_candidate_count": handoff_summary.get(
"campaign_candidate_count",
0,
),
"product_candidate_count": handoff_summary.get(
"product_candidate_count",
0,
),
"review_item_count": len(review_items),
},
"review_items": review_items,
},
}
def _handoff_summary(handoff_review):
handoff_review = _as_dict(handoff_review)
side_effects_clear = bool(
not handoff_review.get("manual_fetch_gate_opened_by_api")
and not handoff_review.get("network_request_allowed")
and not handoff_review.get("api_executes_cli")
and not handoff_review.get("api_opens_database_connection")
and not handoff_review.get("api_writes_database")
and not handoff_review.get("api_uses_external_network")
and not handoff_review.get("database_write_executed")
and not handoff_review.get("fetch_executed")
and not handoff_review.get("cli_executed")
and not handoff_review.get("file_written")
and not handoff_review.get("scheduler_attached")
and not handoff_review.get("candidate_queue_created")
and not handoff_review.get("candidate_queue_persisted")
)
summary = _as_dict(handoff_review.get("candidate_handoff_summary"))
return {
"mode": handoff_review.get("mode"),
"accepted": bool(
handoff_review.get("mcp_fetch_candidate_handoff_review_accepted")
),
"ready_for_manual_candidate_queue_review": bool(
handoff_review.get("ready_for_manual_candidate_queue_review")
),
"source_count": _safe_int(handoff_review.get("handoff_source_count")),
"campaign_candidate_count": _safe_int(
handoff_review.get("campaign_candidate_count")
),
"product_candidate_count": _safe_int(
handoff_review.get("product_candidate_count")
),
"candidate_count": _safe_int(handoff_review.get("candidate_count")),
"source_keys": sorted(
tuple(item)
for item in _as_list(summary.get("source_keys"))
if isinstance(item, (list, tuple)) and len(item) == 2
),
"campaign_candidate_keys": sorted(
key for key in _as_list(summary.get("campaign_candidate_keys")) if key
),
"product_candidate_keys": sorted(
key for key in _as_list(summary.get("product_candidate_keys")) if key
),
"side_effects_clear": side_effects_clear,
"blocked_reasons": handoff_review.get("blocked_reasons", []),
}
def _review_item_summary(item):
item = _as_dict(item)
allowed_actions = sorted(
action for action in _as_list(item.get("allowed_actions")) if action
)
return {
"candidate_type": _safe_text(item.get("candidate_type"), 40),
"platform_code": _safe_text(item.get("platform_code"), 80),
"source_key": _safe_text(item.get("source_key"), 160),
"candidate_key": _safe_text(item.get("candidate_key"), 240),
"candidate_name": _safe_text(item.get("candidate_name"), 300),
"candidate_url": _safe_text(item.get("candidate_url"), 500),
"review_state": _safe_text(item.get("review_state"), 80),
"priority_lane": _safe_text(item.get("priority_lane"), 80),
"allowed_actions": allowed_actions,
"evidence_ref": _safe_text(item.get("evidence_ref"), 160),
"queue_write_status": _safe_text(item.get("queue_write_status"), 80),
"required_fields_present": bool(
item.get("candidate_type") in {"campaign", "product"}
and _has_text(item.get("platform_code"))
and _has_text(item.get("source_key"))
and _has_text(item.get("candidate_key"))
and _has_text(item.get("candidate_name"))
and _has_text(item.get("candidate_url"))
),
"allowed_actions_safe": bool(
allowed_actions
and set(allowed_actions).issubset(SAFE_ALLOWED_ACTIONS)
),
}
def _queue_review_summary(candidate_queue_review):
candidate_queue_review = _as_dict(candidate_queue_review)
items = [
_review_item_summary(item)
for item in _as_list(candidate_queue_review.get("review_items"))
]
campaign_items = [item for item in items if item["candidate_type"] == "campaign"]
product_items = [item for item in items if item["candidate_type"] == "product"]
summary = _as_dict(candidate_queue_review.get("summary"))
return {
"provided_keys": sorted(candidate_queue_review.keys()),
"review_id": _safe_text(candidate_queue_review.get("review_id"), 160),
"review_mode": _safe_text(candidate_queue_review.get("review_mode"), 120),
"target_queue": _safe_text(candidate_queue_review.get("target_queue"), 160),
"operator_confirmed_no_database_write": bool(
candidate_queue_review.get("operator_confirmed_no_database_write")
),
"operator_confirmed_no_external_network": bool(
candidate_queue_review.get("operator_confirmed_no_external_network")
),
"operator_confirmed_no_scheduler_attach": bool(
candidate_queue_review.get("operator_confirmed_no_scheduler_attach")
),
"operator_confirmed_no_persistence": bool(
candidate_queue_review.get("operator_confirmed_no_persistence")
),
"operator_confirmed_manual_review_required": bool(
candidate_queue_review.get("operator_confirmed_manual_review_required")
),
"operator_confirmed_no_secret_payload": bool(
candidate_queue_review.get("operator_confirmed_no_secret_payload")
),
"review_item_count": len(items),
"summary_review_item_count": _safe_int(summary.get("review_item_count")),
"campaign_candidate_count": len(campaign_items),
"summary_campaign_candidate_count": _safe_int(
summary.get("campaign_candidate_count")
),
"product_candidate_count": len(product_items),
"summary_product_candidate_count": _safe_int(
summary.get("product_candidate_count")
),
"campaign_candidate_keys": sorted(
item["candidate_key"] for item in campaign_items if item["candidate_key"]
),
"product_candidate_keys": sorted(
item["candidate_key"] for item in product_items if item["candidate_key"]
),
"review_items": items,
"all_review_states_safe": bool(
items and all(item["review_state"] in SAFE_REVIEW_STATES for item in items)
),
"all_required_fields_present": bool(
items and all(item["required_fields_present"] for item in items)
),
"all_allowed_actions_safe": bool(
items and all(item["allowed_actions_safe"] for item in items)
),
"all_queue_write_status_preview": bool(
items and all(item["queue_write_status"] == "not_persisted" for item in items)
),
"summary_counts_match_items": bool(
_safe_int(summary.get("review_item_count")) == len(items)
and _safe_int(summary.get("campaign_candidate_count")) == len(campaign_items)
and _safe_int(summary.get("product_candidate_count")) == len(product_items)
),
"raw_payload_submitted_to_api": _contains_forbidden_key(
candidate_queue_review,
FORBIDDEN_RAW_PAYLOAD_KEYS,
),
"secret_or_token_submitted_to_api": _contains_forbidden_key(
candidate_queue_review,
FORBIDDEN_SECRET_KEYS,
safe_keys=SAFE_SECRET_METADATA_KEYS,
),
"blocked_side_effects": _blocked_side_effects(candidate_queue_review),
}
def _queue_review_gates(
*,
handoff_review_received,
queue_review_payload_received,
queue_review_valid_object,
handoff,
queue_review,
):
handoff_campaign_keys = set(handoff["campaign_candidate_keys"])
queue_campaign_keys = set(queue_review["campaign_candidate_keys"])
handoff_product_keys = set(handoff["product_candidate_keys"])
queue_product_keys = set(queue_review["product_candidate_keys"])
operator_boundaries_confirmed = bool(
queue_review["operator_confirmed_no_database_write"]
and queue_review["operator_confirmed_no_external_network"]
and queue_review["operator_confirmed_no_scheduler_attach"]
and queue_review["operator_confirmed_no_persistence"]
and queue_review["operator_confirmed_manual_review_required"]
and queue_review["operator_confirmed_no_secret_payload"]
)
return [
{
"key": "handoff_review_payload_or_result_received",
"label": "已提供 candidate handoff review package 或已審核結果",
"passed": handoff_review_received,
},
{
"key": "handoff_review_accepted",
"label": "candidate handoff review gate 必須已通過",
"passed": handoff["accepted"],
},
{
"key": "handoff_ready_for_queue_review",
"label": "handoff review 只放行到人工 candidate queue review",
"passed": handoff["ready_for_manual_candidate_queue_review"],
},
{
"key": "handoff_side_effect_free",
"label": "handoff review 未顯示 API 執行、連外、寫檔、寫 DB 或掛 scheduler",
"passed": handoff["side_effects_clear"],
},
{
"key": "queue_review_payload_received",
"label": "已提供 candidate queue review 草案",
"passed": queue_review_payload_received,
},
{
"key": "queue_review_valid_object",
"label": "candidate queue review payload 必須是 JSON object",
"passed": queue_review_valid_object,
},
{
"key": "queue_review_identity_recorded",
"label": "candidate queue review 必須記錄 review_id",
"passed": bool(queue_review["review_id"]),
},
{
"key": "queue_review_mode_preview_only",
"label": "candidate queue review 必須維持 operator_queue_review_preview",
"passed": queue_review["review_mode"] in SAFE_REVIEW_MODES,
},
{
"key": "queue_review_target_manual_only",
"label": "target queue 只能是 manual_candidate_review",
"passed": queue_review["target_queue"] in SAFE_TARGET_QUEUES,
},
{
"key": "queue_review_candidates_match_handoff",
"label": "queue review 候選 key 必須完全對齊 handoff review",
"passed": bool(
handoff_campaign_keys
and handoff_product_keys
and queue_campaign_keys == handoff_campaign_keys
and queue_product_keys == handoff_product_keys
),
},
{
"key": "queue_review_item_count_within_limit",
"label": "queue review item 數需維持小批次上限",
"passed": bool(0 < queue_review["review_item_count"] <= MAX_QUEUE_REVIEW_ITEMS),
},
{
"key": "queue_review_counts_match_summary",
"label": "summary count 必須與 review_items 數量一致",
"passed": queue_review["summary_counts_match_items"],
},
{
"key": "queue_review_required_fields_present",
"label": "每個 review item 必須具備候選識別欄位",
"passed": queue_review["all_required_fields_present"],
},
{
"key": "queue_review_state_needs_review_only",
"label": "review_state 必須停在 needs_review",
"passed": queue_review["all_review_states_safe"],
},
{
"key": "queue_review_allowed_actions_safe",
"label": "allowed actions 必須限定人工確認/否決/延後",
"passed": queue_review["all_allowed_actions_safe"],
},
{
"key": "queue_review_not_persisted",
"label": "queue_write_status 必須是 not_persisted",
"passed": queue_review["all_queue_write_status_preview"],
},
{
"key": "queue_review_operator_boundaries_confirmed",
"label": "操作員確認無寫入、無連外、無排程、無保存且需人工覆核",
"passed": operator_boundaries_confirmed,
},
{
"key": "queue_review_no_raw_payload",
"label": "queue review payload 不得回貼 raw HTML/body/snapshot",
"passed": not queue_review["raw_payload_submitted_to_api"],
},
{
"key": "queue_review_no_secret_or_token_key",
"label": "queue review payload 不得包含 secret、cookie、password 或 token key",
"passed": not queue_review["secret_or_token_submitted_to_api"],
},
{
"key": "queue_review_side_effect_free",
"label": "queue review payload 不得要求 API 執行、連外、寫檔、寫 DB 或掛 scheduler",
"passed": not queue_review["blocked_side_effects"],
},
]
def build_mcp_fetch_candidate_queue_review_preview(
*,
handoff_review_package=None,
handoff_review_result=None,
candidate_queue_review=None,
phase=None,
):
"""建立 fetch candidate queue review不建立 queue 或寫入。"""
handoff_review_package = _as_dict(handoff_review_package)
handoff_review_result_received = bool(
isinstance(handoff_review_result, dict) and handoff_review_result
)
queue_review_valid_object = isinstance(candidate_queue_review, dict) if (
candidate_queue_review is not None
) else True
queue_review_payload = _as_dict(candidate_queue_review)
handoff_review = _handoff_review_from_inputs(
handoff_review_package,
handoff_review_result,
phase,
)
handoff_review_received = bool(
handoff_review_package or handoff_review_result_received
)
payload_received = bool(
handoff_review_received
or queue_review_payload
or candidate_queue_review is not None
)
queue_review_payload_received = bool(queue_review_payload)
handoff = _handoff_summary(handoff_review)
queue_review = _queue_review_summary(queue_review_payload)
gates = _queue_review_gates(
handoff_review_received=handoff_review_received,
queue_review_payload_received=queue_review_payload_received,
queue_review_valid_object=queue_review_valid_object,
handoff=handoff,
queue_review=queue_review,
)
blocked_reasons = [gate["key"] for gate in gates if not gate["passed"]]
accepted = bool(payload_received and not blocked_reasons)
return {
"mode": (
"mcp_fetch_candidate_queue_review"
if payload_received
else "mcp_fetch_candidate_queue_review_preview"
),
"phase": phase,
"candidate_queue_review_payload_received": payload_received,
"handoff_review_received": handoff_review_received,
"queue_review_payload_received": queue_review_payload_received,
"queue_review_valid_object": queue_review_valid_object,
"handoff_review_accepted": handoff["accepted"],
"mcp_fetch_candidate_queue_review_accepted": accepted,
"candidate_queue_review_ready": accepted,
"ready_for_candidate_queue_writer_preflight": accepted,
"ready_for_api_database_write": False,
"ready_for_scheduler_attach": False,
"network_request_allowed": False,
"fetch_executed": False,
"cli_executed": False,
"candidate_queue_created": False,
"candidate_queue_persisted": False,
"candidate_review_state_updated": False,
"handoff_source_count": handoff["source_count"],
"review_item_count": queue_review["review_item_count"],
"campaign_candidate_count": queue_review["campaign_candidate_count"],
"product_candidate_count": queue_review["product_candidate_count"],
"gate_count": len(gates),
"passed_gate_count": sum(1 for gate in gates if gate["passed"]),
"blocked_reasons": blocked_reasons,
"gates": gates,
"handoff_review_summary": handoff,
"candidate_queue_review_summary": queue_review,
"sample_candidate_queue_review_package": (
_sample_candidate_queue_review_package()
),
"next_operator_steps": [
"queue review 通過後,只代表可進 writer preflight不代表可寫 market_*",
"candidate queue writer、DB import、scheduler attach、AI/Telegram 摘要都必須另開獨立 gate",
"API/UI 仍不得建立 queue、不得讀 artifact、不得抓外站、不得寫 DB、不得掛 scheduler",
],
"payload_persisted": False,
"api_executes_health_check": False,
"api_executes_docker": False,
"api_executes_ssh": False,
"api_executes_cli": False,
"api_opens_database_connection": False,
"api_writes_database": False,
"api_uses_external_network": False,
"database_session_created": False,
"database_commit_executed": False,
"database_write_executed": False,
"external_network_executed": False,
"file_written": False,
"writes_executed": False,
"would_write_database": False,
"scheduler_attached": False,
}

View File

@@ -1,3 +1,3 @@
"""市場情報 rollout phase 單一來源。"""
MARKET_INTEL_PHASE = "phase_126_market_intel_mcp_fetch_candidate_handoff_review"
MARKET_INTEL_PHASE = "phase_127_market_intel_mcp_fetch_candidate_queue_review"

View File

@@ -812,6 +812,32 @@
</div>
</div>
<div class="market-intel-panel" data-market-intel-mcp-fetch-candidate-queue-review>
<div class="market-intel-preview-head">
<div>
<p class="market-intel-muted momo-mono mb-1">MCP / FETCH CANDIDATE QUEUE</p>
<h2 class="market-intel-preview-title">MCP Fetch Candidate Queue 審核</h2>
</div>
<button class="market-intel-icon-button" type="button" title="重新整理 MCP Fetch Candidate Queue" data-market-intel-mcp-fetch-candidate-queue-review-refresh>
<i class="fas fa-rotate-right" aria-hidden="true"></i>
</button>
</div>
<div class="market-intel-preview-meta" data-market-intel-mcp-fetch-candidate-queue-review-meta>
<span class="market-intel-pill">loading</span>
</div>
<div data-market-intel-mcp-fetch-candidate-queue-review-body>
<div class="market-intel-empty">讀取 MCP Fetch Candidate Queue 審核中...</div>
</div>
<div class="market-intel-control-row mt-3">
<textarea class="market-intel-json-input" rows="9" spellcheck="false" data-market-intel-mcp-fetch-candidate-queue-review-input placeholder="handoff review and candidate queue review JSON"></textarea>
<div class="market-intel-control-actions">
<button class="market-intel-icon-button" type="button" title="審核 MCP Fetch Candidate Queue JSON" data-market-intel-mcp-fetch-candidate-queue-review-review>
<i class="fas fa-check" aria-hidden="true"></i>
</button>
</div>
</div>
</div>
<div class="market-intel-panel" data-market-intel-manual-sample>
<div class="market-intel-preview-head">
<div>
@@ -1326,6 +1352,7 @@
const mcpFetchRunReceiptRoot = document.querySelector('[data-market-intel-mcp-fetch-run-receipt]');
const mcpFetchResultParserReviewRoot = document.querySelector('[data-market-intel-mcp-fetch-result-parser-review]');
const mcpFetchCandidateHandoffReviewRoot = document.querySelector('[data-market-intel-mcp-fetch-candidate-handoff-review]');
const mcpFetchCandidateQueueReviewRoot = document.querySelector('[data-market-intel-mcp-fetch-candidate-queue-review]');
const manualSampleRoot = document.querySelector('[data-market-intel-manual-sample]');
const sampleAcceptanceRoot = document.querySelector('[data-market-intel-sample-acceptance]');
const sampleReviewRoot = document.querySelector('[data-market-intel-sample-review]');
@@ -1342,7 +1369,7 @@
const liveInventoryRoot = document.querySelector('[data-market-intel-live-inventory]');
const approvalRoot = document.querySelector('[data-market-intel-approval]');
const deployRoot = document.querySelector('[data-market-intel-deploy]');
if (!root && !writerRoot && !cliRoot && !dbProbeRoot && !seedDiffRoot && !legacyBridgeRoot && !mcpReadinessRoot && !mcpPreflightRoot && !mcpActivationRoot && !mcpFetchGateRoot && !mcpCompletionRoot && !mcpActivationEvidenceRoot && !mcpRuntimeSmokeRoot && !mcpRuntimePromotionRoot && !mcpManualFetchHandoffRoot && !mcpFetchTargetReviewRoot && !mcpFetchRunPackageRoot && !mcpFetchRunReadinessRoot && !mcpFetchRunReceiptRoot && !mcpFetchResultParserReviewRoot && !mcpFetchCandidateHandoffReviewRoot && !manualSampleRoot && !sampleAcceptanceRoot && !sampleReviewRoot && !schedulerRoot && !matchReviewRoot && !opportunityRoot && !opportunityScoringRoot && !opportunityEvidenceRoot && !opportunityAlertRoot && !migrationRoot && !migrationDrillRoot && !catalogReviewRoot && !liveSmokeRoot && !liveInventoryRoot && !approvalRoot && !deployRoot) return;
if (!root && !writerRoot && !cliRoot && !dbProbeRoot && !seedDiffRoot && !legacyBridgeRoot && !mcpReadinessRoot && !mcpPreflightRoot && !mcpActivationRoot && !mcpFetchGateRoot && !mcpCompletionRoot && !mcpActivationEvidenceRoot && !mcpRuntimeSmokeRoot && !mcpRuntimePromotionRoot && !mcpManualFetchHandoffRoot && !mcpFetchTargetReviewRoot && !mcpFetchRunPackageRoot && !mcpFetchRunReadinessRoot && !mcpFetchRunReceiptRoot && !mcpFetchResultParserReviewRoot && !mcpFetchCandidateHandoffReviewRoot && !mcpFetchCandidateQueueReviewRoot && !manualSampleRoot && !sampleAcceptanceRoot && !sampleReviewRoot && !schedulerRoot && !matchReviewRoot && !opportunityRoot && !opportunityScoringRoot && !opportunityEvidenceRoot && !opportunityAlertRoot && !migrationRoot && !migrationDrillRoot && !catalogReviewRoot && !liveSmokeRoot && !liveInventoryRoot && !approvalRoot && !deployRoot) return;
const meta = root ? root.querySelector('[data-market-intel-preview-meta]') : null;
const body = root ? root.querySelector('[data-market-intel-preview-body]') : null;
@@ -1449,6 +1476,12 @@
const mcpFetchCandidateHandoffReviewReview = mcpFetchCandidateHandoffReviewRoot ? mcpFetchCandidateHandoffReviewRoot.querySelector('[data-market-intel-mcp-fetch-candidate-handoff-review-review]') : null;
const mcpFetchCandidateHandoffReviewRefresh = mcpFetchCandidateHandoffReviewRoot ? mcpFetchCandidateHandoffReviewRoot.querySelector('[data-market-intel-mcp-fetch-candidate-handoff-review-refresh]') : null;
const mcpFetchCandidateHandoffReviewEndpoint = "{{ url_for('market_intel.market_intel_mcp_fetch_candidate_handoff_review') }}";
const mcpFetchCandidateQueueReviewMeta = mcpFetchCandidateQueueReviewRoot ? mcpFetchCandidateQueueReviewRoot.querySelector('[data-market-intel-mcp-fetch-candidate-queue-review-meta]') : null;
const mcpFetchCandidateQueueReviewBody = mcpFetchCandidateQueueReviewRoot ? mcpFetchCandidateQueueReviewRoot.querySelector('[data-market-intel-mcp-fetch-candidate-queue-review-body]') : null;
const mcpFetchCandidateQueueReviewInput = mcpFetchCandidateQueueReviewRoot ? mcpFetchCandidateQueueReviewRoot.querySelector('[data-market-intel-mcp-fetch-candidate-queue-review-input]') : null;
const mcpFetchCandidateQueueReviewReview = mcpFetchCandidateQueueReviewRoot ? mcpFetchCandidateQueueReviewRoot.querySelector('[data-market-intel-mcp-fetch-candidate-queue-review-review]') : null;
const mcpFetchCandidateQueueReviewRefresh = mcpFetchCandidateQueueReviewRoot ? mcpFetchCandidateQueueReviewRoot.querySelector('[data-market-intel-mcp-fetch-candidate-queue-review-refresh]') : null;
const mcpFetchCandidateQueueReviewEndpoint = "{{ url_for('market_intel.market_intel_mcp_fetch_candidate_queue_review') }}";
const manualSampleMeta = manualSampleRoot ? manualSampleRoot.querySelector('[data-market-intel-manual-sample-meta]') : null;
const manualSampleBody = manualSampleRoot ? manualSampleRoot.querySelector('[data-market-intel-manual-sample-body]') : null;
const manualSampleRefresh = manualSampleRoot ? manualSampleRoot.querySelector('[data-market-intel-manual-sample-refresh]') : null;
@@ -3658,6 +3691,134 @@
}
};
const renderMcpFetchCandidateQueueReviewMeta = data => {
mcpFetchCandidateQueueReviewMeta.innerHTML = [
`mode=${data.mode || 'unknown'}`,
`accepted=${data.mcp_fetch_candidate_queue_review_accepted ? 'yes' : 'no'}`,
`gates=${data.passed_gate_count || 0}/${data.gate_count || 0}`,
`items=${data.review_item_count || 0}`,
`campaigns=${data.campaign_candidate_count || 0}`,
`products=${data.product_candidate_count || 0}`
].map(item => `<span class="market-intel-pill">${escapeHtml(item)}</span>`).join('');
};
const renderMcpFetchCandidateQueueReviewBody = data => {
const blockers = (data.blocked_reasons || []).join(' / ');
const gates = data.gates || [];
const handoff = data.handoff_review_summary || {};
const review = data.candidate_queue_review_summary || {};
const items = review.review_items || [];
const steps = data.next_operator_steps || [];
const renderCheck = (key, label, status) => `
<div class="market-intel-check">
<div>
<strong>${escapeHtml(key)}</strong>
<small>${escapeHtml(label || '')}</small>
</div>
<span>${escapeHtml(status)}</span>
</div>
`;
mcpFetchCandidateQueueReviewBody.innerHTML = `
<div class="market-intel-empty mb-3">此 queue review 只審核人工覆核草案API 不建立 queue、不更新 review_state、不抓外站、不寫 DB、不掛 scheduler。${blockers ? `阻擋:${escapeHtml(blockers)}` : ''}</div>
<div class="market-intel-deploy-grid">
<div data-market-intel-mcp-fetch-candidate-queue-review-gates>
<p class="market-intel-deploy-section-title">QUEUE REVIEW GATES</p>
<div class="market-intel-check-list">${
gates.length
? gates.map(item => renderCheck(item.key, item.label, item.passed ? 'PASS' : 'BLOCK')).join('')
: '<div class="market-intel-empty">尚未提供 queue review gates。</div>'
}</div>
</div>
<div data-market-intel-mcp-fetch-candidate-queue-review-handoff>
<p class="market-intel-deploy-section-title">HANDOFF LINK</p>
<div class="market-intel-check-list">
${renderCheck('handoff_review', `${handoff.accepted ? 'accepted' : 'pending'} / candidates=${handoff.candidate_count || 0}`, handoff.accepted ? 'ACCEPTED' : 'PENDING')}
${renderCheck('handoff_boundary', 'no queue create / no DB / no scheduler', handoff.side_effects_clear ? 'CLOSED' : 'BLOCK')}
</div>
</div>
<div data-market-intel-mcp-fetch-candidate-queue-review-items>
<p class="market-intel-deploy-section-title">REVIEW ITEMS</p>
<div class="market-intel-check-list">${
items.slice(0, 10).map(item => renderCheck(
item.candidate_key || 'candidate',
`${item.candidate_type || 'candidate'} / ${item.review_state || 'missing'} / ${item.queue_write_status || 'unknown'}`,
item.review_state === 'needs_review' ? 'NEEDS REVIEW' : 'BLOCK'
)).join('') || '<div class="market-intel-empty">尚未提供 candidate queue review items。</div>'
}</div>
</div>
<div data-market-intel-mcp-fetch-candidate-queue-review-policy>
<p class="market-intel-deploy-section-title">REVIEW POLICY</p>
<div class="market-intel-check-list">
${renderCheck('target_queue', review.target_queue || 'missing', review.target_queue === 'manual_candidate_review' ? 'MANUAL' : 'BLOCK')}
${renderCheck('review_mode', review.review_mode || 'missing', review.review_mode === 'operator_queue_review_preview' ? 'PREVIEW' : 'BLOCK')}
${renderCheck('review_state', 'all items must stay needs_review', review.all_review_states_safe ? 'SAFE' : 'BLOCK')}
${renderCheck('queue_write', 'all items must stay not_persisted', review.all_queue_write_status_preview ? 'NO WRITE' : 'BLOCK')}
</div>
</div>
<div data-market-intel-mcp-fetch-candidate-queue-review-next>
<p class="market-intel-deploy-section-title">BOUNDARY / NEXT</p>
<div class="market-intel-check-list">
${renderCheck(
'api_boundary',
'no queue create / no review_state update / no DB / no scheduler',
data.api_writes_database ? 'BLOCK' : 'CLOSED'
)}
${steps.map((item, index) => renderCheck(`step_${index + 1}`, item, 'NEXT')).join('')}
</div>
</div>
</div>
`;
if (mcpFetchCandidateQueueReviewInput && !mcpFetchCandidateQueueReviewInput.value.trim() && data.sample_candidate_queue_review_package) {
mcpFetchCandidateQueueReviewInput.value = JSON.stringify(data.sample_candidate_queue_review_package, null, 2);
}
};
const loadMcpFetchCandidateQueueReview = async () => {
if (!mcpFetchCandidateQueueReviewMeta || !mcpFetchCandidateQueueReviewBody) return;
mcpFetchCandidateQueueReviewBody.innerHTML = '<div class="market-intel-empty">讀取 MCP Fetch Candidate Queue 審核中...</div>';
try {
const response = await fetch(mcpFetchCandidateQueueReviewEndpoint, { credentials: 'same-origin' });
if (!response.ok) throw new Error(`HTTP ${response.status}`);
const data = await response.json();
renderMcpFetchCandidateQueueReviewMeta(data);
renderMcpFetchCandidateQueueReviewBody(data);
} catch (error) {
mcpFetchCandidateQueueReviewMeta.innerHTML = '<span class="market-intel-pill">error</span>';
mcpFetchCandidateQueueReviewBody.innerHTML = `<div class="market-intel-empty">MCP Fetch Candidate Queue 審核讀取失敗:${escapeHtml(error.message)}</div>`;
}
};
const reviewMcpFetchCandidateQueueReview = async () => {
if (!mcpFetchCandidateQueueReviewMeta || !mcpFetchCandidateQueueReviewBody || !mcpFetchCandidateQueueReviewInput) return;
let parsed;
try {
parsed = JSON.parse(mcpFetchCandidateQueueReviewInput.value || '{}');
} catch (error) {
mcpFetchCandidateQueueReviewMeta.innerHTML = '<span class="market-intel-pill">json_error</span>';
mcpFetchCandidateQueueReviewBody.innerHTML = `<div class="market-intel-empty">JSON 格式錯誤:${escapeHtml(error.message)}</div>`;
return;
}
mcpFetchCandidateQueueReviewBody.innerHTML = '<div class="market-intel-empty">審核 MCP Fetch Candidate Queue 中...</div>';
try {
const response = await fetch(mcpFetchCandidateQueueReviewEndpoint, {
method: 'POST',
credentials: 'same-origin',
headers: {
'Content-Type': 'application/json',
'X-CSRFToken': csrfToken
},
body: JSON.stringify({ candidate_queue_review: parsed })
});
const data = await response.json();
if (!response.ok && !data.mode) throw new Error(`HTTP ${response.status}`);
renderMcpFetchCandidateQueueReviewMeta(data);
renderMcpFetchCandidateQueueReviewBody(data);
} catch (error) {
mcpFetchCandidateQueueReviewMeta.innerHTML = '<span class="market-intel-pill">error</span>';
mcpFetchCandidateQueueReviewBody.innerHTML = `<div class="market-intel-empty">MCP Fetch Candidate Queue 審核失敗:${escapeHtml(error.message)}</div>`;
}
};
const renderManualSampleMeta = data => {
manualSampleMeta.innerHTML = [
`mode=${data.mode || 'unknown'}`,
@@ -13121,6 +13282,12 @@
if (mcpFetchCandidateHandoffReviewReview) {
mcpFetchCandidateHandoffReviewReview.addEventListener('click', reviewMcpFetchCandidateHandoffReview);
}
if (mcpFetchCandidateQueueReviewRefresh) {
mcpFetchCandidateQueueReviewRefresh.addEventListener('click', loadMcpFetchCandidateQueueReview);
}
if (mcpFetchCandidateQueueReviewReview) {
mcpFetchCandidateQueueReviewReview.addEventListener('click', reviewMcpFetchCandidateQueueReview);
}
if (manualSampleRefresh) {
manualSampleRefresh.addEventListener('click', loadManualSample);
}
@@ -13385,6 +13552,7 @@
loadMcpFetchRunReceipt();
loadMcpFetchResultParserReview();
loadMcpFetchCandidateHandoffReview();
loadMcpFetchCandidateQueueReview();
loadManualSample();
loadSampleAcceptance();
loadSampleReview();

File diff suppressed because it is too large Load Diff