diff --git a/TODO_NEXT_STEPS.txt b/TODO_NEXT_STEPS.txt
index 964bdf4..57a6645 100644
--- a/TODO_NEXT_STEPS.txt
+++ b/TODO_NEXT_STEPS.txt
@@ -4,6 +4,7 @@
================================================================================
【已完成】
+ - V10.351 補市場情報 candidate queue review AI summary Telegram dispatch report catalog record run receipt:新增 read-only report catalog record run receipt builder、POST endpoint、UI 按鈕與 deployment readiness smoke target,在 run readiness 後審核外部 CLI writer output、catalog record key/hash、DB commit receipt 與 post-write smoke;API/UI 不讀 approval/Telegram token、不呼叫 LLM、不補產報表、不派送 Telegram、不開 DB、不寫檔、不執行 CLI、不寫 catalog record、不 commit、不更新 review_state、不掛 scheduler。
- V10.348 補市場情報 candidate queue review AI summary Telegram dispatch report catalog record run readiness:新增 read-only report catalog record run readiness builder、POST endpoint、UI 按鈕與 deployment readiness smoke target,在 run package 後檢查 payload manifest、manual CLI command、backup/dry-run、run receipt 與 postwrite smoke 條件;API/UI 不讀 approval/Telegram token、不呼叫 LLM、不補產報表、不派送 Telegram、不開 DB、不寫檔、不執行 CLI、不寫 catalog record、不更新 review_state、不掛 scheduler。
- V10.347 補市場情報 candidate queue review AI summary Telegram dispatch report catalog record run package:新增 read-only report catalog record run package builder、POST endpoint、UI 按鈕與 deployment readiness smoke target,在 record write gate 後整理 payload manifest、CLI command bundle、backup/dry-run trace 與後續 run readiness separate gate;API/UI 不讀 approval/Telegram token、不呼叫 LLM、不補產報表、不派送 Telegram、不開 DB、不寫檔、不執行 CLI、不寫 catalog record、不更新 review_state、不掛 scheduler。
- V10.344 補市場情報 candidate queue review AI summary Telegram dispatch report catalog record write gate:新增 read-only report catalog record write builder、POST endpoint、UI 按鈕與 deployment readiness smoke target,在 write preflight 後檢查 catalog record key/schema/hash trace、operator dry-run、backup 與 commit separate gate;API/UI 不讀 approval/Telegram token、不呼叫 LLM、不補產報表、不派送 Telegram、不開 DB、不寫檔、不寫 catalog record、不更新 review_state、不掛 scheduler。
diff --git a/config.py b/config.py
index e58ae9b..cdd3163 100644
--- a/config.py
+++ b/config.py
@@ -323,7 +323,7 @@ YOUTUBE_API_KEY = os.getenv('YOUTUBE_API_KEY', '')
# ==========================================
# 系統版本與路徑
# ==========================================
-SYSTEM_VERSION = "V10.350"
+SYSTEM_VERSION = "V10.351"
LOG_FILE_PATH = os.path.join(BASE_DIR, 'logs/system.log')
public_url = PUBLIC_URL # 用於模板顯示
diff --git a/docs/memory/history_logs.md b/docs/memory/history_logs.md
index 6789bcc..d534849 100644
--- a/docs/memory/history_logs.md
+++ b/docs/memory/history_logs.md
@@ -12,6 +12,10 @@
## 📅 詳細更新日誌 (考古存檔)
+### 2026-05-20:市場情報 Telegram dispatch report catalog record run receipt
+- **V10.351 report catalog record run receipt**: 新增 `candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_receipt` service、POST endpoint、UI 按鈕與 deployment readiness smoke target,在 run readiness 後審核外部 CLI writer output、catalog record key/hash、DB commit receipt 與 post-write smoke。
+- **只讀安全邊界**: 本階段只放行到後續 report catalog record commit gate;API/UI 不讀 approval/Telegram token、不呼叫 LLM、不補產報表、不派送 Telegram、不開 DB、不寫檔、不執行 CLI、不寫 catalog record、不 commit、不更新 `review_state`、不掛 scheduler。
+
### 2026-05-20:市場情報 Telegram dispatch report catalog record run readiness
- **V10.348 report catalog record run readiness**: 新增 `candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_readiness` service、POST endpoint、UI 按鈕與 deployment readiness smoke target,在 run package 後檢查 payload manifest、manual CLI command、backup/dry-run、run receipt 與 postwrite smoke 條件。
- **只讀安全邊界**: 本階段只放行到後續 report catalog record run receipt gate;API/UI 不讀 approval/Telegram token、不呼叫 LLM、不補產報表、不派送 Telegram、不開 DB、不寫檔、不執行 CLI、不寫 catalog record、不更新 `review_state`、不掛 scheduler。
diff --git a/routes/market_intel_review_report_routes.py b/routes/market_intel_review_report_routes.py
index 3a3f38d..209fb0e 100644
--- a/routes/market_intel_review_report_routes.py
+++ b/routes/market_intel_review_report_routes.py
@@ -40,6 +40,9 @@ from services.market_intel.candidate_queue_review_ai_summary_persistence_telegra
from services.market_intel.candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_readiness import (
build_candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_readiness,
)
+from services.market_intel.candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_receipt import (
+ build_candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_receipt,
+)
from services.market_intel.candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_closeout import (
build_candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_closeout,
)
@@ -484,6 +487,105 @@ def _build_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_pa
)
+def _build_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_readiness(
+ *,
+ service,
+ sample_result,
+ operator_evidence,
+ writer_output,
+ smoke_result,
+ payload_error,
+ limit,
+ execute_requested,
+ apply_real_write,
+):
+ report_catalog_record_run_package = _build_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_package(
+ service=service,
+ sample_result=sample_result,
+ operator_evidence=operator_evidence,
+ writer_output=writer_output,
+ smoke_result=smoke_result,
+ payload_error=payload_error,
+ limit=limit,
+ execute_requested=execute_requested,
+ apply_real_write=apply_real_write,
+ )
+ return build_candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_readiness(
+ telegram_dispatch_report_catalog_record_run_package=report_catalog_record_run_package,
+ operator_evidence=operator_evidence,
+ execute_requested=execute_requested,
+ apply_real_write=apply_real_write,
+ )
+
+
+def _extract_report_catalog_record_run_receipt_payload(
+ sample_result,
+ operator_evidence,
+ writer_output,
+ smoke_result,
+):
+ operator_evidence = _as_dict(operator_evidence)
+ sample_result = _as_dict(sample_result)
+ payload = _as_dict(
+ operator_evidence.get("market_intel_report_catalog_record_run_receipt")
+ or operator_evidence.get("telegram_dispatch_report_catalog_record_run_receipt")
+ or operator_evidence.get("report_catalog_record_run_receipt")
+ or operator_evidence.get("catalog_record_run_receipt")
+ or sample_result.get("market_intel_report_catalog_record_run_receipt")
+ or sample_result.get("telegram_dispatch_report_catalog_record_run_receipt")
+ or sample_result.get("report_catalog_record_run_receipt")
+ or sample_result.get("catalog_record_run_receipt")
+ )
+ if payload:
+ return payload
+ writer_output = _as_dict(writer_output)
+ smoke_result = _as_dict(smoke_result)
+ if writer_output or smoke_result:
+ return {
+ "mode": "manual_market_intel_report_catalog_record_run_receipt",
+ "writer_output": writer_output,
+ "postwrite_smoke_result": smoke_result,
+ }
+ return {}
+
+
+def _build_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_receipt(
+ *,
+ service,
+ sample_result,
+ operator_evidence,
+ writer_output,
+ smoke_result,
+ payload_error,
+ limit,
+ execute_requested,
+ apply_real_write,
+):
+ report_catalog_record_run_readiness = _build_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_readiness(
+ service=service,
+ sample_result=sample_result,
+ operator_evidence=operator_evidence,
+ writer_output=writer_output,
+ smoke_result=smoke_result,
+ payload_error=payload_error,
+ limit=limit,
+ execute_requested=execute_requested,
+ apply_real_write=apply_real_write,
+ )
+ return build_candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_receipt(
+ telegram_dispatch_report_catalog_record_run_readiness=report_catalog_record_run_readiness,
+ catalog_record_run_receipt=_extract_report_catalog_record_run_receipt_payload(
+ sample_result,
+ operator_evidence,
+ writer_output,
+ smoke_result,
+ ),
+ operator_evidence=operator_evidence,
+ execute_requested=execute_requested,
+ apply_real_write=apply_real_write,
+ )
+
+
@market_intel_review_bp.route(
"/api/market_intel/manual_sample_review/"
"candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_input",
@@ -845,7 +947,7 @@ def market_intel_manual_sample_candidate_queue_review_ai_summary_persistence_tel
sample_result, operator_evidence, writer_output, smoke_result, payload_error, limit = (
_extract_run_payload()
)
- report_catalog_record_run_package = _build_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_package(
+ data = _build_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_readiness(
service=service,
sample_result=sample_result,
operator_evidence=operator_evidence,
@@ -856,9 +958,31 @@ def market_intel_manual_sample_candidate_queue_review_ai_summary_persistence_tel
execute_requested=execute_requested,
apply_real_write=apply_real_write,
)
- data = build_candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_readiness(
- telegram_dispatch_report_catalog_record_run_package=report_catalog_record_run_package,
+ data["phase"] = service.phase
+ return jsonify(data), 400 if payload_error else 200
+
+
+@market_intel_review_bp.route(
+ "/api/market_intel/manual_sample_review/"
+ "candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_receipt",
+ methods=["POST"],
+)
+@login_required
+def market_intel_manual_sample_candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_receipt():
+ service = MarketIntelService()
+ execute_requested = request.args.get("execute", "false").lower() == "true"
+ apply_real_write = request.args.get("apply_real_write", "false").lower() == "true"
+ sample_result, operator_evidence, writer_output, smoke_result, payload_error, limit = (
+ _extract_run_payload()
+ )
+ data = _build_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_receipt(
+ service=service,
+ sample_result=sample_result,
operator_evidence=operator_evidence,
+ writer_output=writer_output,
+ smoke_result=smoke_result,
+ payload_error=payload_error,
+ limit=limit,
execute_requested=execute_requested,
apply_real_write=apply_real_write,
)
diff --git a/services/market_intel/candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_receipt.py b/services/market_intel/candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_receipt.py
new file mode 100644
index 0000000..73d2fe6
--- /dev/null
+++ b/services/market_intel/candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_receipt.py
@@ -0,0 +1,846 @@
+"""候選審核 queue AI summary Telegram dispatch report catalog record run receipt。
+
+本模組只審核外部 CLI 回貼的 catalog record 寫入 receipt 與 post-write
+smoke;不讀 approval 或 Telegram token、不呼叫 LLM、不派送 Telegram、
+不開 DB、不寫檔、不執行 CLI、不補寫 catalog record、不 commit、
+不更新 review_state、不掛 scheduler。
+"""
+
+from services.market_intel.candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_archive_summary import (
+ SAFE_APPROVAL_ENV_VAR,
+ SAFE_TOKEN_METADATA_KEYS,
+ TARGET_COLUMN,
+ TARGET_TABLE,
+ _as_dict,
+ _as_list,
+ _contains_forbidden_token_key,
+ _has_text,
+ _safe_int,
+ _safe_text,
+ _strip_safe_token_boolean_keys,
+)
+from services.market_intel.candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_readiness import (
+ FALSE_RESPONSE_KEYS as REPORT_CATALOG_RECORD_RUN_READINESS_FALSE_RESPONSE_KEYS,
+)
+
+
+EXPECTED_RECEIPT_MODES = {
+ "market_intel_report_catalog_record_run_receipt",
+ "manual_market_intel_report_catalog_record_run_receipt",
+ "market_intel_report_catalog_record_cli_run_receipt",
+ "catalog_record_writer_cli_receipt",
+}
+EXPECTED_SMOKE_MODES = {
+ "market_intel_report_catalog_record_postwrite_smoke_read_only",
+ "catalog_record_postwrite_smoke_read_only",
+ "report_catalog_record_postwrite_smoke_read_only",
+}
+REQUIRED_RUN_READINESS_BOUNDARIES = {
+ "do_not_read_approval_token_from_report_catalog_record_run_readiness_api",
+ "do_not_read_telegram_token_from_report_catalog_record_run_readiness_api",
+ "do_not_call_llm_from_report_catalog_record_run_readiness",
+ "do_not_generate_report_from_report_catalog_record_run_readiness_api",
+ "do_not_write_report_catalog_record_run_readiness_artifact_from_api",
+ "do_not_execute_catalog_record_cli_from_report_catalog_record_run_readiness_api",
+ "do_not_write_report_catalog_record_from_report_catalog_record_run_readiness_api",
+ "do_not_dispatch_telegram_from_report_catalog_record_run_readiness_api",
+ "do_not_open_database_connection_from_report_catalog_record_run_readiness",
+ "do_not_update_review_state_from_report_catalog_record_run_readiness",
+ "do_not_attach_scheduler_from_report_catalog_record_run_readiness",
+ "future_market_intel_report_catalog_record_run_receipt_must_use_separate_gate",
+}
+FALSE_RESPONSE_KEYS = tuple(
+ dict.fromkeys(
+ REPORT_CATALOG_RECORD_RUN_READINESS_FALSE_RESPONSE_KEYS
+ + (
+ "telegram_dispatch_report_catalog_record_run_receipt_gate_file_written",
+ "report_catalog_record_run_receipt_gate_file_written",
+ "catalog_record_run_receipt_gate_file_written",
+ "catalog_record_run_receipt_executed",
+ "catalog_record_commit_gate_file_written",
+ "catalog_record_commit_executed",
+ )
+ )
+)
+
+
+def _safe_bool(value):
+ return bool(value)
+
+
+def _hash_matches_expected(expected_hash, observed_hash):
+ return bool(expected_hash and observed_hash and str(expected_hash) == str(observed_hash))
+
+
+def _readiness_summary(telegram_dispatch_report_catalog_record_run_readiness):
+ readiness = _as_dict(telegram_dispatch_report_catalog_record_run_readiness)
+ manifest = _as_dict(readiness.get("catalog_record_run_readiness_manifest"))
+ command = _as_dict(manifest.get("manual_cli_run_command"))
+ boundaries = _as_dict(manifest.get("execution_boundaries"))
+ source_artifacts = _as_dict(manifest.get("source_artifacts"))
+ run_package = _as_dict(readiness.get("telegram_dispatch_report_catalog_record_run_package"))
+ promotion = _as_dict(readiness.get("promotion_gate"))
+ sections = _as_list(
+ readiness.get("telegram_dispatch_report_catalog_record_run_readiness_sections")
+ )
+ safe_boundaries = set(str(item) for item in _as_list(readiness.get("safe_boundaries")))
+ missing_boundaries = sorted(REQUIRED_RUN_READINESS_BOUNDARIES - safe_boundaries)
+ section_keys = sorted(
+ str(item.get("key"))
+ for item in sections
+ if isinstance(item, dict) and item.get("key")
+ )
+
+ return {
+ "provided": bool(readiness),
+ "mode": readiness.get("mode"),
+ "target_operation": readiness.get("target_operation"),
+ "run_readiness_passed": bool(
+ readiness.get("telegram_dispatch_report_catalog_record_run_readiness_passed")
+ or readiness.get(
+ "summary_persistence_telegram_dispatch_report_catalog_record_run_readiness_passed"
+ )
+ or readiness.get("report_catalog_record_run_readiness_passed")
+ ),
+ "ready_for_next_manual_phase": bool(readiness.get("ready_for_next_manual_phase")),
+ "ready_for_cli_operator_run": bool(readiness.get("ready_for_cli_operator_run")),
+ "ready_for_market_intel_report_catalog_record_cli_operator_run": bool(
+ readiness.get("ready_for_market_intel_report_catalog_record_cli_operator_run")
+ ),
+ "ready_for_market_intel_report_catalog_record_run_receipt": bool(
+ readiness.get("ready_for_market_intel_report_catalog_record_run_receipt")
+ ),
+ "ready_for_market_intel_report_catalog_record_cli_run": bool(
+ readiness.get("ready_for_market_intel_report_catalog_record_cli_run")
+ ),
+ "ready_for_market_intel_report_catalog_record_commit": bool(
+ readiness.get("ready_for_market_intel_report_catalog_record_commit")
+ ),
+ "statement_count": _safe_int(readiness.get("statement_count")),
+ "expected_summary_payload_hash": _safe_text(
+ readiness.get("expected_summary_payload_hash")
+ or manifest.get("expected_summary_payload_hash"),
+ 80,
+ ),
+ "manifest_version": manifest.get("manifest_version"),
+ "source_mode": manifest.get("source_mode"),
+ "source_operation": manifest.get("source_operation"),
+ "target_table": manifest.get("target_table"),
+ "target_column": manifest.get("target_column"),
+ "record_family": manifest.get("record_family"),
+ "report_output_hash": manifest.get("report_output_hash"),
+ "manifest_statement_count": _safe_int(manifest.get("statement_count")),
+ "source_record_write_gate": bool(source_artifacts.get("record_write_gate")),
+ "source_write_preflight": bool(source_artifacts.get("write_preflight")),
+ "source_report_output": bool(source_artifacts.get("report_output")),
+ "source_backup": bool(source_artifacts.get("backup")),
+ "source_dry_run": bool(source_artifacts.get("dry_run")),
+ "command_script_path": command.get("script_path"),
+ "command_dry_run": command.get("dry_run_command"),
+ "command_executed_by_api": bool(command.get("executed_by_api")),
+ "command_writes_catalog_record": bool(command.get("writes_catalog_record")),
+ "command_approval_env_var": command.get("requires_approval_env_var"),
+ "api_writes_file": bool(boundaries.get("api_writes_file")),
+ "api_executes_cli": bool(boundaries.get("api_executes_cli")),
+ "api_writes_catalog_record": bool(boundaries.get("api_writes_catalog_record")),
+ "api_writes_database": bool(boundaries.get("api_writes_database")),
+ "api_dispatches_telegram": bool(boundaries.get("api_dispatches_telegram")),
+ "api_calls_llm": bool(boundaries.get("api_calls_llm")),
+ "api_attaches_scheduler": bool(boundaries.get("api_attaches_scheduler")),
+ "run_package_mode": run_package.get("mode"),
+ "run_package_passed": bool(run_package.get("catalog_record_run_package_passed")),
+ "run_package_record_family": run_package.get("record_family"),
+ "run_package_report_output_hash": run_package.get("report_output_hash"),
+ "run_package_expected_hash": run_package.get("expected_summary_payload_hash"),
+ "promotion_allowed": bool(promotion.get("allowed")),
+ "promotion_next_manual_phase": promotion.get("next_manual_phase"),
+ "promotion_requires_real_db_write": bool(promotion.get("requires_real_db_write")),
+ "promotion_requires_cli_run": bool(promotion.get("requires_cli_run")),
+ "promotion_requires_postwrite_smoke": bool(
+ promotion.get("requires_postwrite_smoke")
+ ),
+ "promotion_receipt_separate_gate": bool(
+ promotion.get("report_catalog_record_run_receipt_requires_separate_gate")
+ ),
+ "promotion_commit_separate_gate": bool(
+ promotion.get("report_catalog_record_commit_requires_separate_gate")
+ ),
+ "safe_boundaries_complete": not missing_boundaries,
+ "missing_safe_boundaries": missing_boundaries,
+ "section_keys": section_keys,
+ "section_count": len(section_keys),
+ "forbidden_token_key_detected": _contains_forbidden_token_key(
+ _strip_safe_token_boolean_keys(readiness)
+ ),
+ "blocked_count": len(_as_list(readiness.get("blocked_reasons"))),
+ **{key: bool(readiness.get(key)) for key in FALSE_RESPONSE_KEYS},
+ }
+
+
+def _receipt_payload(operator_evidence, catalog_record_run_receipt):
+ operator_evidence = _as_dict(operator_evidence)
+ embedded = _as_dict(
+ operator_evidence.get("market_intel_report_catalog_record_run_receipt")
+ or operator_evidence.get("telegram_dispatch_report_catalog_record_run_receipt")
+ or operator_evidence.get("report_catalog_record_run_receipt")
+ or operator_evidence.get("catalog_record_run_receipt")
+ )
+ return _as_dict(catalog_record_run_receipt) or embedded
+
+
+def _writer_summary(receipt_payload, readiness):
+ receipt = _as_dict(receipt_payload)
+ writer = _as_dict(
+ receipt.get("writer_output")
+ or receipt.get("catalog_record_writer_output")
+ or receipt.get("execution_result")
+ )
+ source = writer or receipt
+ target_table = source.get("target_table") or receipt.get("target_table")
+ target_column = source.get("target_column") or receipt.get("target_column")
+ record_family = (
+ source.get("record_family")
+ or source.get("target_report_family")
+ or receipt.get("record_family")
+ or receipt.get("target_report_family")
+ )
+ observed_summary_hash = (
+ source.get("expected_summary_payload_hash")
+ or source.get("summary_payload_hash")
+ or receipt.get("expected_summary_payload_hash")
+ or receipt.get("summary_payload_hash")
+ )
+ observed_report_hash = (
+ source.get("report_output_hash")
+ or source.get("report_hash")
+ or receipt.get("report_output_hash")
+ or receipt.get("report_hash")
+ )
+ return {
+ "provided": bool(source),
+ "mode": source.get("mode") or receipt.get("mode"),
+ "exit_code": source.get("exit_code"),
+ "catalog_record_cli_executed": bool(
+ source.get("catalog_record_cli_executed")
+ or source.get("catalog_record_writer_cli_executed")
+ or source.get("cli_executed")
+ or source.get("writes_executed")
+ ),
+ "catalog_record_written": bool(
+ source.get("catalog_record_written")
+ or source.get("report_catalog_record_written")
+ or source.get("record_written")
+ ),
+ "target_table": _safe_text(target_table, 120),
+ "target_column": _safe_text(target_column, 120),
+ "record_family": _safe_text(record_family, 180),
+ "catalog_record_key": _safe_text(
+ source.get("catalog_record_key")
+ or source.get("record_key")
+ or receipt.get("catalog_record_key")
+ or receipt.get("record_key"),
+ 240,
+ ),
+ "statement_count": _safe_int(
+ source.get("statement_count")
+ or source.get("affected_statement_count")
+ or receipt.get("statement_count")
+ ),
+ "affected_count": _safe_int(
+ source.get("affected_count")
+ or source.get("updated_count")
+ or source.get("inserted_count")
+ or receipt.get("affected_count")
+ ),
+ "database_connection_opened": bool(source.get("database_connection_opened")),
+ "explicit_transaction_opened": bool(source.get("explicit_transaction_opened")),
+ "database_write_executed": bool(source.get("database_write_executed")),
+ "database_commit_executed": bool(source.get("database_commit_executed")),
+ "database_rollback_executed": bool(source.get("database_rollback_executed")),
+ "executed_by_api": bool(source.get("executed_by_api") or source.get("api_executed")),
+ "api_writes_file": bool(source.get("api_writes_file")),
+ "api_executes_cli": bool(source.get("api_executes_cli")),
+ "api_writes_database": bool(source.get("api_writes_database")),
+ "api_dispatches_telegram": bool(source.get("api_dispatches_telegram")),
+ "api_calls_llm": bool(source.get("api_calls_llm")),
+ "telegram_dispatched": bool(source.get("telegram_dispatched")),
+ "llm_call_executed": bool(source.get("llm_call_executed")),
+ "external_network_executed": bool(source.get("external_network_executed")),
+ "scheduler_attached": bool(source.get("scheduler_attached")),
+ "review_state_update_executed": bool(source.get("review_state_update_executed")),
+ "expected_summary_payload_hash": readiness["expected_summary_payload_hash"],
+ "observed_summary_payload_hash": _safe_text(observed_summary_hash, 80),
+ "summary_payload_hash_matches_expected": _hash_matches_expected(
+ readiness["expected_summary_payload_hash"],
+ observed_summary_hash,
+ ),
+ "expected_report_output_hash": readiness["report_output_hash"],
+ "observed_report_output_hash": _safe_text(observed_report_hash, 120),
+ "report_output_hash_matches_expected": _hash_matches_expected(
+ readiness["report_output_hash"],
+ observed_report_hash,
+ ),
+ "approval_or_telegram_token_key_detected": _contains_forbidden_token_key(
+ _strip_safe_token_boolean_keys(source)
+ )
+ or _contains_forbidden_token_key(_strip_safe_token_boolean_keys(receipt)),
+ }
+
+
+def _smoke_summary(receipt_payload, readiness):
+ receipt = _as_dict(receipt_payload)
+ smoke = _as_dict(
+ receipt.get("postwrite_smoke_result")
+ or receipt.get("catalog_record_postwrite_smoke")
+ or receipt.get("postwrite_smoke")
+ )
+ source = smoke
+ found_key = (
+ source.get("catalog_record_key")
+ or source.get("record_key")
+ or source.get("found_catalog_record_key")
+ or source.get("found_record_key")
+ )
+ observed_hash = (
+ source.get("expected_summary_payload_hash")
+ or source.get("summary_payload_hash")
+ or source.get("metadata_json_summary_payload_hash")
+ )
+ return {
+ "provided": bool(source),
+ "mode": source.get("mode"),
+ "postwrite_smoke_passed": bool(
+ source.get("postwrite_smoke_passed")
+ or source.get("catalog_record_postwrite_smoke_passed")
+ ),
+ "read_only_query_executed": bool(source.get("read_only_query_executed")),
+ "catalog_record_found": bool(
+ source.get("catalog_record_found")
+ or source.get("record_found")
+ or source.get("found")
+ ),
+ "target_table": _safe_text(source.get("target_table"), 120),
+ "target_column": _safe_text(source.get("target_column"), 120),
+ "record_family": _safe_text(
+ source.get("record_family") or source.get("target_report_family"),
+ 180,
+ ),
+ "catalog_record_key": _safe_text(found_key, 240),
+ "database_connection_opened": bool(source.get("database_connection_opened")),
+ "database_write_executed": bool(source.get("database_write_executed")),
+ "database_commit_executed": bool(source.get("database_commit_executed")),
+ "database_rollback_executed": bool(source.get("database_rollback_executed")),
+ "external_network_executed": bool(source.get("external_network_executed")),
+ "scheduler_attached": bool(source.get("scheduler_attached")),
+ "expected_summary_payload_hash": readiness["expected_summary_payload_hash"],
+ "observed_summary_payload_hash": _safe_text(observed_hash, 80),
+ "summary_payload_hash_matches_expected": _hash_matches_expected(
+ readiness["expected_summary_payload_hash"],
+ observed_hash,
+ ),
+ "approval_or_telegram_token_key_detected": _contains_forbidden_token_key(
+ _strip_safe_token_boolean_keys(source)
+ ),
+ }
+
+
+def _operator_summary(operator_evidence):
+ operator_evidence = _as_dict(operator_evidence)
+ return {
+ "provided_keys": sorted(operator_evidence.keys()),
+ "report_catalog_record_run_receipt_artifact_path": _safe_text(
+ operator_evidence.get("report_catalog_record_run_receipt_artifact_path")
+ or operator_evidence.get(
+ "telegram_dispatch_report_catalog_record_run_receipt_artifact_path"
+ )
+ or operator_evidence.get(
+ "market_intel_report_catalog_record_run_receipt_artifact_path"
+ )
+ ),
+ "report_catalog_record_run_readiness_artifact_path_recorded": _has_text(
+ operator_evidence.get("report_catalog_record_run_readiness_artifact_path")
+ or operator_evidence.get(
+ "telegram_dispatch_report_catalog_record_run_readiness_artifact_path"
+ )
+ ),
+ "report_catalog_record_run_package_artifact_path_recorded": _has_text(
+ operator_evidence.get("report_catalog_record_run_package_artifact_path")
+ or operator_evidence.get(
+ "telegram_dispatch_report_catalog_record_run_package_artifact_path"
+ )
+ ),
+ "catalog_record_payload_manifest_path_recorded": _has_text(
+ operator_evidence.get("catalog_record_payload_manifest_path")
+ or operator_evidence.get("report_catalog_record_payload_manifest_path")
+ ),
+ "catalog_record_backup_artifact_path_recorded": _has_text(
+ operator_evidence.get("catalog_record_backup_artifact_path")
+ or operator_evidence.get("report_catalog_backup_artifact_path")
+ or operator_evidence.get("metadata_json_backup_artifact_path")
+ ),
+ "catalog_record_write_dry_run_artifact_path_recorded": _has_text(
+ operator_evidence.get("catalog_record_write_dry_run_artifact_path")
+ or operator_evidence.get("report_catalog_record_dry_run_artifact_path")
+ ),
+ "catalog_record_writer_output_artifact_path_recorded": _has_text(
+ operator_evidence.get("catalog_record_writer_output_artifact_path")
+ or operator_evidence.get("report_catalog_record_writer_output_artifact_path")
+ or operator_evidence.get("writer_output_json_path")
+ ),
+ "catalog_record_postwrite_smoke_artifact_path_recorded": _has_text(
+ operator_evidence.get("catalog_record_postwrite_smoke_artifact_path")
+ or operator_evidence.get("report_catalog_record_postwrite_smoke_artifact_path")
+ or operator_evidence.get("postwrite_smoke_json_path")
+ ),
+ "catalog_record_cli_command_recorded": _has_text(
+ operator_evidence.get("catalog_record_cli_command")
+ or operator_evidence.get("report_catalog_record_cli_command")
+ ),
+ "operator_confirmed_report_catalog_record_run_receipt": bool(
+ operator_evidence.get("operator_confirmed_report_catalog_record_run_receipt")
+ or operator_evidence.get(
+ "operator_confirmed_market_intel_report_catalog_record_run_receipt"
+ )
+ ),
+ "operator_confirmed_catalog_record_run_readiness_reviewed": bool(
+ operator_evidence.get(
+ "operator_confirmed_catalog_record_run_readiness_reviewed"
+ )
+ or operator_evidence.get(
+ "operator_confirmed_report_catalog_record_run_readiness_reviewed"
+ )
+ ),
+ "operator_confirmed_catalog_record_cli_output_reviewed": bool(
+ operator_evidence.get("operator_confirmed_catalog_record_cli_output_reviewed")
+ or operator_evidence.get("operator_confirmed_writer_output_reviewed")
+ ),
+ "operator_confirmed_catalog_record_postwrite_smoke_passed": bool(
+ operator_evidence.get(
+ "operator_confirmed_catalog_record_postwrite_smoke_passed"
+ )
+ or operator_evidence.get("operator_confirmed_postwrite_smoke_passed")
+ ),
+ "operator_confirmed_catalog_record_backup_available": bool(
+ operator_evidence.get("operator_confirmed_catalog_record_backup_available")
+ or operator_evidence.get("operator_confirmed_catalog_record_backup_required")
+ ),
+ "operator_confirmed_catalog_record_commit_requires_separate_gate": bool(
+ operator_evidence.get(
+ "operator_confirmed_report_catalog_record_commit_requires_separate_gate"
+ )
+ or operator_evidence.get(
+ "operator_confirmed_catalog_record_commit_requires_separate_gate"
+ )
+ ),
+ "operator_confirmed_no_token_in_report_catalog_record_run_receipt": bool(
+ operator_evidence.get(
+ "operator_confirmed_no_token_in_report_catalog_record_run_receipt"
+ )
+ or operator_evidence.get("operator_confirmed_no_token_in_artifacts")
+ ),
+ "operator_confirmed_no_api_file_write": bool(
+ operator_evidence.get("operator_confirmed_no_api_file_write")
+ ),
+ "operator_confirmed_no_api_cli_execution": bool(
+ operator_evidence.get("operator_confirmed_no_api_cli_execution")
+ or operator_evidence.get("operator_confirmed_no_api_execute_cli")
+ ),
+ "operator_confirmed_no_api_catalog_record_write": bool(
+ operator_evidence.get("operator_confirmed_no_api_catalog_record_write")
+ or operator_evidence.get("operator_confirmed_no_api_db_write")
+ ),
+ "operator_confirmed_no_api_telegram_dispatch": bool(
+ operator_evidence.get("operator_confirmed_no_api_telegram_dispatch")
+ ),
+ "operator_confirmed_no_api_db_write": bool(
+ operator_evidence.get("operator_confirmed_no_api_db_write")
+ ),
+ "operator_confirmed_no_llm_call": bool(
+ operator_evidence.get("operator_confirmed_no_llm_call")
+ ),
+ "operator_confirmed_no_scheduler_attach": bool(
+ operator_evidence.get("operator_confirmed_no_scheduler_attach")
+ ),
+ "catalog_record_run_receipt_notes_recorded": _has_text(
+ operator_evidence.get("report_catalog_record_run_receipt_notes")
+ or operator_evidence.get("catalog_record_run_receipt_notes")
+ ),
+ "safe_token_metadata_only": all(
+ key in SAFE_TOKEN_METADATA_KEYS or key == "approval_env_var"
+ for key in operator_evidence
+ if "token" in str(key).lower() or str(key).lower() == "approval_env_var"
+ )
+ and operator_evidence.get("approval_env_var", SAFE_APPROVAL_ENV_VAR)
+ == SAFE_APPROVAL_ENV_VAR,
+ "forbidden_token_submitted_to_api": _contains_forbidden_token_key(
+ operator_evidence
+ ),
+ }
+
+
+def _run_receipt_sections(readiness, writer, smoke, operator):
+ return [
+ {
+ "key": "catalog_record_run_receipt_identity",
+ "title": "Catalog record run receipt identity",
+ "facts": [
+ f"family={readiness['record_family'] or 'missing'}",
+ f"writer_mode={writer['mode'] or 'missing'}",
+ f"statement_count={writer['statement_count']}",
+ ],
+ },
+ {
+ "key": "catalog_record_run_receipt_cli_result",
+ "title": "Catalog record CLI result",
+ "facts": [
+ f"cli_executed={writer['catalog_record_cli_executed']}",
+ f"record_written={writer['catalog_record_written']}",
+ f"committed={writer['database_commit_executed']}",
+ ],
+ },
+ {
+ "key": "catalog_record_run_receipt_postwrite_smoke",
+ "title": "Catalog record post-write smoke",
+ "facts": [
+ f"smoke_mode={smoke['mode'] or 'missing'}",
+ f"record_found={smoke['catalog_record_found']}",
+ f"hash_match={smoke['summary_payload_hash_matches_expected']}",
+ ],
+ },
+ {
+ "key": "catalog_record_run_receipt_safety",
+ "title": "Catalog record run receipt safety",
+ "facts": [
+ f"receipt_path={bool(operator['report_catalog_record_run_receipt_artifact_path'])}",
+ "api_cli_executed=false",
+ "api_database_write_executed=false",
+ "scheduler_attached=false",
+ ],
+ },
+ ]
+
+
+def _run_receipt_gates(readiness, writer, smoke, operator, apply_real_write):
+ required_sections = {
+ "catalog_record_run_readiness_identity",
+ "catalog_record_run_readiness_artifacts",
+ "catalog_record_run_readiness_command",
+ "catalog_record_run_readiness_safety",
+ }
+ section_keys = set(readiness["section_keys"])
+ return [
+ {
+ "key": "report_catalog_record_run_readiness_preview_provided",
+ "label": "必須提供上一階段 report catalog record run readiness preview",
+ "passed": bool(
+ readiness["provided"]
+ and readiness["mode"]
+ == "candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_readiness_preview"
+ ),
+ },
+ {
+ "key": "report_catalog_record_run_readiness_passed",
+ "label": "report catalog record run readiness gate 必須已通過",
+ "passed": readiness["run_readiness_passed"],
+ },
+ {
+ "key": "report_catalog_record_run_readiness_ready_for_receipt",
+ "label": "run readiness 只能放行到 catalog record run receipt",
+ "passed": bool(
+ readiness["ready_for_next_manual_phase"]
+ and readiness["ready_for_cli_operator_run"]
+ and readiness[
+ "ready_for_market_intel_report_catalog_record_cli_operator_run"
+ ]
+ and readiness["ready_for_market_intel_report_catalog_record_run_receipt"]
+ and not readiness["ready_for_market_intel_report_catalog_record_cli_run"]
+ and not readiness["ready_for_market_intel_report_catalog_record_commit"]
+ and readiness["promotion_allowed"]
+ and readiness["promotion_next_manual_phase"]
+ == "market_intel_report_catalog_record_run_receipt"
+ and readiness["promotion_requires_real_db_write"]
+ and readiness["promotion_requires_cli_run"]
+ and readiness["promotion_requires_postwrite_smoke"]
+ and readiness["promotion_receipt_separate_gate"]
+ and readiness["promotion_commit_separate_gate"]
+ ),
+ },
+ {
+ "key": "catalog_record_run_readiness_manifest_complete",
+ "label": "run readiness manifest 必須保留 target、record family、hash、source artifacts 與 CLI command",
+ "passed": bool(
+ readiness["manifest_version"]
+ == "market_intel_report_catalog_record_run_readiness_v1"
+ and readiness["target_table"] == TARGET_TABLE
+ and readiness["target_column"] == TARGET_COLUMN
+ and readiness["record_family"]
+ and readiness["report_output_hash"]
+ and readiness["expected_summary_payload_hash"]
+ and readiness["statement_count"] > 0
+ and readiness["manifest_statement_count"] == readiness["statement_count"]
+ and readiness["source_record_write_gate"]
+ and readiness["source_write_preflight"]
+ and readiness["source_report_output"]
+ and readiness["source_backup"]
+ and readiness["source_dry_run"]
+ and readiness["command_script_path"]
+ == "scripts/market_intel/write_report_catalog_record.py"
+ and readiness["command_dry_run"]
+ and not readiness["command_executed_by_api"]
+ and readiness["command_writes_catalog_record"]
+ and readiness["command_approval_env_var"] == SAFE_APPROVAL_ENV_VAR
+ ),
+ },
+ {
+ "key": "catalog_record_run_readiness_sections_complete",
+ "label": "run readiness sections 必須完整",
+ "passed": required_sections.issubset(section_keys),
+ },
+ {
+ "key": "report_catalog_record_run_readiness_runtime_boundaries_clear",
+ "label": "run readiness payload 不得顯示 API 寫檔、DB、Telegram、LLM、CLI 或 scheduler 副作用",
+ "passed": bool(
+ all(not readiness[key] for key in FALSE_RESPONSE_KEYS)
+ and not readiness["api_writes_file"]
+ and not readiness["api_executes_cli"]
+ and not readiness["api_writes_catalog_record"]
+ and not readiness["api_writes_database"]
+ and not readiness["api_dispatches_telegram"]
+ and not readiness["api_calls_llm"]
+ and not readiness["api_attaches_scheduler"]
+ ),
+ },
+ {
+ "key": "report_catalog_record_run_readiness_safe_boundaries_complete",
+ "label": "run readiness 必須保留 run receipt separate gate 與 runtime 安全邊界",
+ "passed": readiness["safe_boundaries_complete"],
+ },
+ {
+ "key": "catalog_record_run_receipt_writer_output_provided",
+ "label": "操作員需貼回 catalog record CLI writer output",
+ "passed": writer["provided"],
+ },
+ {
+ "key": "catalog_record_run_receipt_writer_committed",
+ "label": "writer output 必須顯示 CLI 已寫入 catalog record 並 commit",
+ "passed": bool(
+ writer["mode"] in EXPECTED_RECEIPT_MODES
+ and writer["catalog_record_cli_executed"]
+ and writer["catalog_record_written"]
+ and writer["target_table"] == TARGET_TABLE
+ and writer["target_column"] == TARGET_COLUMN
+ and writer["record_family"] == readiness["record_family"]
+ and writer["statement_count"] == readiness["statement_count"]
+ and writer["statement_count"] > 0
+ and writer["database_connection_opened"]
+ and writer["database_write_executed"]
+ and writer["database_commit_executed"]
+ and not writer["database_rollback_executed"]
+ and writer["summary_payload_hash_matches_expected"]
+ and writer["report_output_hash_matches_expected"]
+ ),
+ },
+ {
+ "key": "catalog_record_run_receipt_writer_runtime_boundaries",
+ "label": "writer output 不得顯示 API 執行、Telegram、LLM、scheduler 或 review_state 副作用",
+ "passed": bool(
+ not writer["executed_by_api"]
+ and not writer["api_writes_file"]
+ and not writer["api_executes_cli"]
+ and not writer["api_writes_database"]
+ and not writer["api_dispatches_telegram"]
+ and not writer["api_calls_llm"]
+ and not writer["telegram_dispatched"]
+ and not writer["llm_call_executed"]
+ and not writer["external_network_executed"]
+ and not writer["scheduler_attached"]
+ and not writer["review_state_update_executed"]
+ ),
+ },
+ {
+ "key": "catalog_record_run_receipt_postwrite_smoke_provided",
+ "label": "操作員需貼回 catalog record post-write smoke 結果",
+ "passed": smoke["provided"],
+ },
+ {
+ "key": "catalog_record_run_receipt_postwrite_smoke_passed",
+ "label": "post-write smoke 必須是只讀查詢、找到 catalog record,且 hash 符合",
+ "passed": bool(
+ smoke["mode"] in EXPECTED_SMOKE_MODES
+ and smoke["postwrite_smoke_passed"]
+ and smoke["read_only_query_executed"]
+ and smoke["catalog_record_found"]
+ and smoke["target_table"] == TARGET_TABLE
+ and smoke["target_column"] == TARGET_COLUMN
+ and smoke["record_family"] == readiness["record_family"]
+ and smoke["summary_payload_hash_matches_expected"]
+ ),
+ },
+ {
+ "key": "catalog_record_run_receipt_postwrite_smoke_no_db_write",
+ "label": "post-write smoke 不得寫 DB、commit、rollback、連外或掛 scheduler",
+ "passed": bool(
+ not smoke["database_write_executed"]
+ and not smoke["database_commit_executed"]
+ and not smoke["database_rollback_executed"]
+ and not smoke["external_network_executed"]
+ and not smoke["scheduler_attached"]
+ ),
+ },
+ {
+ "key": "operator_confirmed_report_catalog_record_run_receipt",
+ "label": "操作員確認 receipt、readiness、writer output、backup、CLI、postwrite smoke 與 commit separate gate",
+ "passed": bool(
+ operator["report_catalog_record_run_receipt_artifact_path"]
+ and operator["report_catalog_record_run_readiness_artifact_path_recorded"]
+ and operator["report_catalog_record_run_package_artifact_path_recorded"]
+ and operator["catalog_record_payload_manifest_path_recorded"]
+ and operator["catalog_record_backup_artifact_path_recorded"]
+ and operator["catalog_record_write_dry_run_artifact_path_recorded"]
+ and operator["catalog_record_writer_output_artifact_path_recorded"]
+ and operator["catalog_record_postwrite_smoke_artifact_path_recorded"]
+ and operator["catalog_record_cli_command_recorded"]
+ and operator["operator_confirmed_report_catalog_record_run_receipt"]
+ and operator["operator_confirmed_catalog_record_run_readiness_reviewed"]
+ and operator["operator_confirmed_catalog_record_cli_output_reviewed"]
+ and operator["operator_confirmed_catalog_record_postwrite_smoke_passed"]
+ and operator["operator_confirmed_catalog_record_backup_available"]
+ and operator["operator_confirmed_catalog_record_commit_requires_separate_gate"]
+ ),
+ },
+ {
+ "key": "operator_confirmed_report_catalog_record_run_receipt_runtime_boundaries",
+ "label": "操作員確認本 API 不寫檔、不執行 CLI、不寫 DB、不派送 Telegram、不呼叫 LLM、不掛 scheduler",
+ "passed": bool(
+ operator[
+ "operator_confirmed_no_token_in_report_catalog_record_run_receipt"
+ ]
+ and operator["operator_confirmed_no_api_file_write"]
+ and operator["operator_confirmed_no_api_cli_execution"]
+ and operator["operator_confirmed_no_api_catalog_record_write"]
+ and operator["operator_confirmed_no_api_telegram_dispatch"]
+ and operator["operator_confirmed_no_api_db_write"]
+ and operator["operator_confirmed_no_llm_call"]
+ and operator["operator_confirmed_no_scheduler_attach"]
+ ),
+ },
+ {
+ "key": "report_catalog_record_run_receipt_no_token_submitted_to_api",
+ "label": "catalog record run receipt payload 不得包含 approval 或 Telegram token key",
+ "passed": bool(
+ not operator["forbidden_token_submitted_to_api"]
+ and not readiness["forbidden_token_key_detected"]
+ and not writer["approval_or_telegram_token_key_detected"]
+ and not smoke["approval_or_telegram_token_key_detected"]
+ ),
+ },
+ {
+ "key": "report_catalog_record_run_receipt_apply_real_write_not_requested_from_api",
+ "label": "API/UI report catalog record run receipt 不接受 apply_real_write",
+ "passed": not apply_real_write,
+ },
+ ]
+
+
+def build_candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_receipt(
+ *,
+ telegram_dispatch_report_catalog_record_run_readiness,
+ catalog_record_run_receipt=None,
+ operator_evidence=None,
+ execute_requested=False,
+ apply_real_write=False,
+):
+ """建立 Telegram dispatch report catalog record run receipt;不執行副作用。"""
+ readiness = _readiness_summary(telegram_dispatch_report_catalog_record_run_readiness)
+ receipt_payload = _receipt_payload(operator_evidence, catalog_record_run_receipt)
+ writer = _writer_summary(receipt_payload, readiness)
+ smoke = _smoke_summary(receipt_payload, readiness)
+ operator = _operator_summary(operator_evidence)
+ sections = _run_receipt_sections(readiness, writer, smoke, operator)
+ gates = _run_receipt_gates(readiness, writer, smoke, operator, bool(apply_real_write))
+ blocked_reasons = [gate["key"] for gate in gates if not gate["passed"]]
+ receipt_passed = bool(not blocked_reasons)
+
+ return {
+ "mode": "candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_receipt_preview",
+ "target_table": TARGET_TABLE,
+ "target_column": TARGET_COLUMN,
+ "target_json_path": [
+ "market_intel_ai_summary",
+ "telegram_dispatch_report_catalog_record_run_receipt",
+ ],
+ "target_operation": "review_manual_market_intel_report_catalog_record_run_receipt",
+ "execute_requested": bool(execute_requested),
+ "apply_real_write_requested": bool(apply_real_write),
+ "report_catalog_record_run_receipt_reviewed": True,
+ "telegram_dispatch_report_catalog_record_run_receipt_passed": receipt_passed,
+ "summary_persistence_telegram_dispatch_report_catalog_record_run_receipt_passed": receipt_passed,
+ "report_catalog_record_run_receipt_passed": receipt_passed,
+ "ready_for_next_manual_phase": receipt_passed,
+ "ready_for_market_intel_report_catalog_record_commit": receipt_passed,
+ "ready_for_market_intel_report_catalog_record_commit_gate": receipt_passed,
+ "ready_for_market_intel_report_catalog_record_cli_run": False,
+ "ready_for_cli_operator_run": False,
+ "ready_for_real_write": False,
+ "ready_for_report_generation": False,
+ "ready_for_telegram_dispatch": False,
+ "ready_for_api_database_write": False,
+ "ready_for_scheduler_attach": False,
+ "ready_for_llm_call": False,
+ "api_executes_cli": False,
+ "api_executes_llm": False,
+ "api_dispatches_telegram": False,
+ "api_reads_approval_token": False,
+ "api_writes_file": False,
+ "api_writes_database": False,
+ "api_updates_review_state": False,
+ **{key: False for key in FALSE_RESPONSE_KEYS},
+ "statement_count": readiness["statement_count"],
+ "expected_summary_payload_hash": readiness["expected_summary_payload_hash"],
+ "blocked_reasons": blocked_reasons,
+ "gates": gates,
+ "telegram_dispatch_report_catalog_record_run_readiness": readiness,
+ "telegram_dispatch_report_catalog_record_run_receipt_writer": writer,
+ "telegram_dispatch_report_catalog_record_run_receipt_postwrite_smoke": smoke,
+ "telegram_dispatch_report_catalog_record_run_receipt_sections": sections,
+ "operator_telegram_dispatch_report_catalog_record_run_receipt": operator,
+ "promotion_gate": {
+ "allowed": receipt_passed,
+ "next_manual_phase": "market_intel_report_catalog_record_commit",
+ "requires_real_db_write": False,
+ "requires_cli_run": False,
+ "requires_scheduler_attach": False,
+ "requires_operator_approval": True,
+ "requires_postwrite_smoke": False,
+ "api_must_not_generate_report": True,
+ "api_must_not_write_file": True,
+ "api_must_not_execute_cli": True,
+ "api_must_not_write_catalog_record": True,
+ "api_must_not_write_database": True,
+ "api_must_not_dispatch_telegram": True,
+ "report_catalog_record_commit_requires_separate_gate": True,
+ },
+ "next_operator_steps": [
+ "保存 report catalog record run receipt artifact path",
+ "人工確認 CLI writer output、catalog record key、hash 與 post-write smoke",
+ "只在 run receipt 通過後進入 report catalog record commit gate",
+ "不得從 API/UI 補寫 catalog record、重跑 CLI、commit 或掛 scheduler",
+ ],
+ "safe_boundaries": [
+ "do_not_read_approval_token_from_report_catalog_record_run_receipt_api",
+ "do_not_read_telegram_token_from_report_catalog_record_run_receipt_api",
+ "do_not_call_llm_from_report_catalog_record_run_receipt",
+ "do_not_generate_report_from_report_catalog_record_run_receipt_api",
+ "do_not_write_report_catalog_record_run_receipt_artifact_from_api",
+ "do_not_execute_catalog_record_cli_from_report_catalog_record_run_receipt_api",
+ "do_not_write_report_catalog_record_from_report_catalog_record_run_receipt_api",
+ "do_not_dispatch_telegram_from_report_catalog_record_run_receipt_api",
+ "do_not_open_database_connection_from_report_catalog_record_run_receipt",
+ "do_not_update_review_state_from_report_catalog_record_run_receipt",
+ "do_not_attach_scheduler_from_report_catalog_record_run_receipt",
+ "future_market_intel_report_catalog_record_commit_must_use_separate_gate",
+ "report_catalog_record_run_receipt_preview_only",
+ "no_remove_orphans",
+ "no_momo_db_lifecycle_change",
+ ],
+ }
diff --git a/services/market_intel/deployment_readiness.py b/services/market_intel/deployment_readiness.py
index 79e4a45..016a2d4 100644
--- a/services/market_intel/deployment_readiness.py
+++ b/services/market_intel/deployment_readiness.py
@@ -50,6 +50,7 @@ from services.market_intel.candidate_queue_review_ai_summary_persistence_telegra
from services.market_intel.candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_write import build_candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_write
from services.market_intel.candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_package import build_candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_package
from services.market_intel.candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_readiness import build_candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_readiness
+from services.market_intel.candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_receipt import build_candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_receipt
from services.market_intel.candidate_queue_review_ai_summary_persistence_transaction import build_candidate_queue_review_ai_summary_persistence_transaction
from services.market_intel.candidate_queue_review_ai_summary_persistence_writer_preflight import build_candidate_queue_review_ai_summary_persistence_writer_preflight
from services.market_intel.candidate_queue_review_ai_summary_run_package import build_candidate_queue_review_ai_summary_run_package
@@ -80,7 +81,7 @@ BLOCKED_RUN_REVIEW_KEYS = (
"summary_persistence_run_readiness_file_written", "run_readiness_file_written", "summary_persistence_run_receipt_file_written", "run_receipt_file_written", "receipt_file_written", "summary_persistence_closeout_file_written", "closeout_file_written", "telegram_dispatch_gate_file_written", "telegram_dispatch_run_package_file_written", "telegram_dispatch_run_readiness_file_written", "telegram_dispatch_run_receipt_file_written", "telegram_dispatch_receipt_file_written", "telegram_dispatch_closeout_file_written", "telegram_dispatch_archive_file_written", "telegram_dispatch_archive_summary_file_written", "telegram_dispatch_report_input_file_written", "telegram_dispatch_report_run_package_file_written", "telegram_dispatch_report_run_readiness_file_written", "telegram_dispatch_report_run_receipt_file_written", "telegram_message_file_written", "archive_file_written", "archive_record_written", "archive_manifest_written", "archive_summary_file_written", "report_input_file_written", "report_run_package_file_written", "report_run_readiness_file_written", "report_run_receipt_file_written", "report_receipt_file_written", "report_output_file_written", "report_file_written", "report_record_written", "report_manifest_written",
"telegram_dispatch_report_closeout_file_written",
"report_closeout_file_written",
- "telegram_dispatch_report_archive_file_written", "report_archive_file_written", "report_archive_record_written", "report_archive_manifest_written", "telegram_dispatch_report_archive_summary_file_written", "report_archive_summary_file_written", "report_summary_input_file_written", "telegram_dispatch_report_catalog_handoff_file_written", "report_catalog_handoff_file_written", "catalog_handoff_file_written", "telegram_dispatch_report_catalog_index_file_written", "report_catalog_index_file_written", "catalog_index_file_written", "report_catalog_index_manifest_written", "catalog_index_manifest_written", "catalog_index_written", "catalog_manifest_written", "catalog_record_written", "report_catalog_record_written", "report_archive_summary_catalog_handoff_file_written", "telegram_dispatch_report_catalog_write_preflight_file_written", "report_catalog_write_preflight_file_written", "catalog_write_preflight_file_written", "catalog_record_write_preflight_file_written", "telegram_dispatch_report_catalog_record_write_file_written", "report_catalog_record_write_file_written", "catalog_record_write_file_written", "catalog_record_write_executed", "report_catalog_record_write_executed", "telegram_dispatch_report_catalog_record_write_gate_file_written", "report_catalog_record_write_gate_file_written", "catalog_record_write_gate_file_written", "telegram_dispatch_report_catalog_record_run_package_file_written", "report_catalog_record_run_package_file_written", "catalog_record_run_package_file_written", "catalog_record_write_receipt_file_written", "catalog_record_postwrite_smoke_executed", "catalog_record_commit_executed", "telegram_dispatch_report_catalog_record_run_readiness_file_written", "report_catalog_record_run_readiness_file_written", "catalog_record_run_readiness_file_written", "telegram_dispatch_report_catalog_record_run_readiness_gate_file_written", "report_catalog_record_run_readiness_gate_file_written", "catalog_record_run_readiness_gate_file_written", "catalog_record_run_readiness_executed", "catalog_record_run_receipt_written", "catalog_record_postwrite_smoke_written", "telegram_dispatch_report_catalog_record_run_receipt_file_written", "report_catalog_record_run_receipt_file_written", "catalog_record_run_receipt_file_written", "catalog_record_cli_executed", "catalog_record_writer_cli_executed", "catalog_record_postwrite_smoke_file_written",
+ "telegram_dispatch_report_archive_file_written", "report_archive_file_written", "report_archive_record_written", "report_archive_manifest_written", "telegram_dispatch_report_archive_summary_file_written", "report_archive_summary_file_written", "report_summary_input_file_written", "telegram_dispatch_report_catalog_handoff_file_written", "report_catalog_handoff_file_written", "catalog_handoff_file_written", "telegram_dispatch_report_catalog_index_file_written", "report_catalog_index_file_written", "catalog_index_file_written", "report_catalog_index_manifest_written", "catalog_index_manifest_written", "catalog_index_written", "catalog_manifest_written", "catalog_record_written", "report_catalog_record_written", "report_archive_summary_catalog_handoff_file_written", "telegram_dispatch_report_catalog_write_preflight_file_written", "report_catalog_write_preflight_file_written", "catalog_write_preflight_file_written", "catalog_record_write_preflight_file_written", "telegram_dispatch_report_catalog_record_write_file_written", "report_catalog_record_write_file_written", "catalog_record_write_file_written", "catalog_record_write_executed", "report_catalog_record_write_executed", "telegram_dispatch_report_catalog_record_write_gate_file_written", "report_catalog_record_write_gate_file_written", "catalog_record_write_gate_file_written", "telegram_dispatch_report_catalog_record_run_package_file_written", "report_catalog_record_run_package_file_written", "catalog_record_run_package_file_written", "catalog_record_write_receipt_file_written", "catalog_record_postwrite_smoke_executed", "catalog_record_commit_executed", "telegram_dispatch_report_catalog_record_run_readiness_file_written", "report_catalog_record_run_readiness_file_written", "catalog_record_run_readiness_file_written", "telegram_dispatch_report_catalog_record_run_readiness_gate_file_written", "report_catalog_record_run_readiness_gate_file_written", "catalog_record_run_readiness_gate_file_written", "catalog_record_run_readiness_executed", "catalog_record_run_receipt_written", "catalog_record_postwrite_smoke_written", "telegram_dispatch_report_catalog_record_run_receipt_gate_file_written", "report_catalog_record_run_receipt_gate_file_written", "catalog_record_run_receipt_gate_file_written", "catalog_record_run_receipt_executed", "catalog_record_commit_gate_file_written", "telegram_dispatch_report_catalog_record_run_receipt_file_written", "report_catalog_record_run_receipt_file_written", "catalog_record_run_receipt_file_written", "catalog_record_cli_executed", "catalog_record_writer_cli_executed", "catalog_record_postwrite_smoke_file_written",
"summary_persistence_record_written",
"metadata_patch_written",
"transaction_file_written",
@@ -104,7 +105,7 @@ BLOCKED_RUN_REVIEW_KEYS = (
"writes_executed",
"would_write_database",
)
-PRODUCTION_SMOKE_TARGETS = ("/health", "/market_intel", "/api/market_intel/status", "/api/market_intel/deployment_readiness", "/api/market_intel/schema_smoke", "/api/market_intel/schema_db_probe", "/api/market_intel/platform_seed_db_diff", "/api/market_intel/legacy_source_bridge", "/api/market_intel/mcp_readiness", "/api/market_intel/mcp_tool_contract", "/api/market_intel/mcp_deploy_preflight", "/api/market_intel/mcp_activation_runbook", "/api/market_intel/mcp_fetch_gate", "/api/market_intel/scheduler_plan", "/api/market_intel/manual_sample_plan", "/api/market_intel/manual_sample_acceptance", "/api/market_intel/manual_sample_review", "/api/market_intel/match_review_plan", "/api/market_intel/opportunity_plan", "/api/market_intel/opportunity_scoring_plan", "/api/market_intel/opportunity_evidence_plan", "/api/market_intel/opportunity_alert_plan", "/api/market_intel/migration_apply_drill", "/api/market_intel/migration_catalog_review", "/api/market_intel/migration_live_smoke", "/api/market_intel/live_db_inventory", "/api/market_intel/manual_sample_review/candidate_queue_writer_postwrite_smoke", "/api/market_intel/manual_sample_review/candidate_queue_writer_operator_drill", "/api/market_intel/manual_sample_review/candidate_queue_writer_run_package", "/api/market_intel/manual_sample_review/candidate_queue_writer_run_readiness", "/api/market_intel/manual_sample_review/candidate_queue_writer_run_receipt", "/api/market_intel/manual_sample_review/candidate_queue_writer_run_closeout", "/api/market_intel/manual_sample_review/candidate_queue_review_handoff", "/api/market_intel/manual_sample_review/candidate_queue_review_inventory", "/api/market_intel/manual_sample_review/candidate_queue_review_decision", "/api/market_intel/manual_sample_review/candidate_queue_review_decision_approval", "/api/market_intel/manual_sample_review/candidate_queue_review_decision_transaction", "/api/market_intel/manual_sample_review/candidate_queue_review_decision_writer_preflight", "/api/market_intel/manual_sample_review/candidate_queue_review_decision_writer_postwrite_smoke", "/api/market_intel/manual_sample_review/candidate_queue_review_decision_writer_operator_drill", "/api/market_intel/manual_sample_review/candidate_queue_review_decision_writer_run_package", "/api/market_intel/manual_sample_review/candidate_queue_review_decision_writer_run_readiness", "/api/market_intel/manual_sample_review/candidate_queue_review_decision_writer_run_receipt", "/api/market_intel/manual_sample_review/candidate_queue_review_decision_writer_run_closeout", "/api/market_intel/manual_sample_review/candidate_queue_review_decision_post_closeout_inventory", "/api/market_intel/manual_sample_review/candidate_queue_review_completion_archive", "/api/market_intel/manual_sample_review/candidate_queue_review_archive_summary", "/api/market_intel/manual_sample_review/candidate_queue_review_ai_summary_preflight", "/api/market_intel/manual_sample_review/candidate_queue_review_ai_summary_run_package", "/api/market_intel/manual_sample_review/candidate_queue_review_ai_summary_output_receipt", "/api/market_intel/manual_sample_review/candidate_queue_review_ai_summary_persistence_preflight", "/api/market_intel/manual_sample_review/candidate_queue_review_ai_summary_persistence_transaction", "/api/market_intel/manual_sample_review/candidate_queue_review_ai_summary_persistence_writer_preflight", "/api/market_intel/manual_sample_review/candidate_queue_review_ai_summary_persistence_run_package", "/api/market_intel/manual_sample_review/candidate_queue_review_ai_summary_persistence_run_readiness", "/api/market_intel/manual_sample_review/candidate_queue_review_ai_summary_persistence_run_receipt", "/api/market_intel/manual_sample_review/candidate_queue_review_ai_summary_persistence_run_closeout", "/api/market_intel/manual_sample_review/candidate_queue_review_ai_summary_persistence_telegram_dispatch_gate", "/api/market_intel/manual_sample_review/candidate_queue_review_ai_summary_persistence_telegram_dispatch_run_package", "/api/market_intel/manual_sample_review/candidate_queue_review_ai_summary_persistence_telegram_dispatch_run_readiness", "/api/market_intel/manual_sample_review/candidate_queue_review_ai_summary_persistence_telegram_dispatch_run_receipt", "/api/market_intel/manual_sample_review/candidate_queue_review_ai_summary_persistence_telegram_dispatch_closeout", "/api/market_intel/manual_sample_review/candidate_queue_review_ai_summary_persistence_telegram_dispatch_archive", "/api/market_intel/manual_sample_review/candidate_queue_review_ai_summary_persistence_telegram_dispatch_archive_summary", "/api/market_intel/manual_sample_review/candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_input", "/api/market_intel/manual_sample_review/candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_run_package", "/api/market_intel/manual_sample_review/candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_run_readiness", "/api/market_intel/manual_sample_review/candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_run_receipt", "/api/market_intel/manual_sample_review/candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_closeout", "/api/market_intel/manual_sample_review/candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_archive", "/api/market_intel/manual_sample_review/candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_archive_summary", "/api/market_intel/manual_sample_review/candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_handoff", "/api/market_intel/manual_sample_review/candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_index", "/api/market_intel/manual_sample_review/candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_write_preflight", "/api/market_intel/manual_sample_review/candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_write", "/api/market_intel/manual_sample_review/candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_package", "/api/market_intel/manual_sample_review/candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_readiness", "/api/market_intel/manual_sample_review/candidate_queue_review_decision_writer_status")
+PRODUCTION_SMOKE_TARGETS = ("/health", "/market_intel", "/api/market_intel/status", "/api/market_intel/deployment_readiness", "/api/market_intel/schema_smoke", "/api/market_intel/schema_db_probe", "/api/market_intel/platform_seed_db_diff", "/api/market_intel/legacy_source_bridge", "/api/market_intel/mcp_readiness", "/api/market_intel/mcp_tool_contract", "/api/market_intel/mcp_deploy_preflight", "/api/market_intel/mcp_activation_runbook", "/api/market_intel/mcp_fetch_gate", "/api/market_intel/scheduler_plan", "/api/market_intel/manual_sample_plan", "/api/market_intel/manual_sample_acceptance", "/api/market_intel/manual_sample_review", "/api/market_intel/match_review_plan", "/api/market_intel/opportunity_plan", "/api/market_intel/opportunity_scoring_plan", "/api/market_intel/opportunity_evidence_plan", "/api/market_intel/opportunity_alert_plan", "/api/market_intel/migration_apply_drill", "/api/market_intel/migration_catalog_review", "/api/market_intel/migration_live_smoke", "/api/market_intel/live_db_inventory", "/api/market_intel/manual_sample_review/candidate_queue_writer_postwrite_smoke", "/api/market_intel/manual_sample_review/candidate_queue_writer_operator_drill", "/api/market_intel/manual_sample_review/candidate_queue_writer_run_package", "/api/market_intel/manual_sample_review/candidate_queue_writer_run_readiness", "/api/market_intel/manual_sample_review/candidate_queue_writer_run_receipt", "/api/market_intel/manual_sample_review/candidate_queue_writer_run_closeout", "/api/market_intel/manual_sample_review/candidate_queue_review_handoff", "/api/market_intel/manual_sample_review/candidate_queue_review_inventory", "/api/market_intel/manual_sample_review/candidate_queue_review_decision", "/api/market_intel/manual_sample_review/candidate_queue_review_decision_approval", "/api/market_intel/manual_sample_review/candidate_queue_review_decision_transaction", "/api/market_intel/manual_sample_review/candidate_queue_review_decision_writer_preflight", "/api/market_intel/manual_sample_review/candidate_queue_review_decision_writer_postwrite_smoke", "/api/market_intel/manual_sample_review/candidate_queue_review_decision_writer_operator_drill", "/api/market_intel/manual_sample_review/candidate_queue_review_decision_writer_run_package", "/api/market_intel/manual_sample_review/candidate_queue_review_decision_writer_run_readiness", "/api/market_intel/manual_sample_review/candidate_queue_review_decision_writer_run_receipt", "/api/market_intel/manual_sample_review/candidate_queue_review_decision_writer_run_closeout", "/api/market_intel/manual_sample_review/candidate_queue_review_decision_post_closeout_inventory", "/api/market_intel/manual_sample_review/candidate_queue_review_completion_archive", "/api/market_intel/manual_sample_review/candidate_queue_review_archive_summary", "/api/market_intel/manual_sample_review/candidate_queue_review_ai_summary_preflight", "/api/market_intel/manual_sample_review/candidate_queue_review_ai_summary_run_package", "/api/market_intel/manual_sample_review/candidate_queue_review_ai_summary_output_receipt", "/api/market_intel/manual_sample_review/candidate_queue_review_ai_summary_persistence_preflight", "/api/market_intel/manual_sample_review/candidate_queue_review_ai_summary_persistence_transaction", "/api/market_intel/manual_sample_review/candidate_queue_review_ai_summary_persistence_writer_preflight", "/api/market_intel/manual_sample_review/candidate_queue_review_ai_summary_persistence_run_package", "/api/market_intel/manual_sample_review/candidate_queue_review_ai_summary_persistence_run_readiness", "/api/market_intel/manual_sample_review/candidate_queue_review_ai_summary_persistence_run_receipt", "/api/market_intel/manual_sample_review/candidate_queue_review_ai_summary_persistence_run_closeout", "/api/market_intel/manual_sample_review/candidate_queue_review_ai_summary_persistence_telegram_dispatch_gate", "/api/market_intel/manual_sample_review/candidate_queue_review_ai_summary_persistence_telegram_dispatch_run_package", "/api/market_intel/manual_sample_review/candidate_queue_review_ai_summary_persistence_telegram_dispatch_run_readiness", "/api/market_intel/manual_sample_review/candidate_queue_review_ai_summary_persistence_telegram_dispatch_run_receipt", "/api/market_intel/manual_sample_review/candidate_queue_review_ai_summary_persistence_telegram_dispatch_closeout", "/api/market_intel/manual_sample_review/candidate_queue_review_ai_summary_persistence_telegram_dispatch_archive", "/api/market_intel/manual_sample_review/candidate_queue_review_ai_summary_persistence_telegram_dispatch_archive_summary", "/api/market_intel/manual_sample_review/candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_input", "/api/market_intel/manual_sample_review/candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_run_package", "/api/market_intel/manual_sample_review/candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_run_readiness", "/api/market_intel/manual_sample_review/candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_run_receipt", "/api/market_intel/manual_sample_review/candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_closeout", "/api/market_intel/manual_sample_review/candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_archive", "/api/market_intel/manual_sample_review/candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_archive_summary", "/api/market_intel/manual_sample_review/candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_handoff", "/api/market_intel/manual_sample_review/candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_index", "/api/market_intel/manual_sample_review/candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_write_preflight", "/api/market_intel/manual_sample_review/candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_write", "/api/market_intel/manual_sample_review/candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_package", "/api/market_intel/manual_sample_review/candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_readiness", "/api/market_intel/manual_sample_review/candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_receipt", "/api/market_intel/manual_sample_review/candidate_queue_review_decision_writer_status")
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):
@@ -280,6 +281,7 @@ def build_deployment_readiness_preview(*, service, market_intel_tables, schema_s
candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_write = build_candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_write(telegram_dispatch_report_catalog_write_preflight=candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_write_preflight)
candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_package = build_candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_package(telegram_dispatch_report_catalog_record_write=candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_write)
candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_readiness = build_candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_readiness(telegram_dispatch_report_catalog_record_run_package=candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_package)
+ candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_receipt = build_candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_receipt(telegram_dispatch_report_catalog_record_run_readiness=candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_readiness)
checks = {
"schema_smoke_passed": bool(schema_smoke["passed"]),
"feature_flags_default_safe": bool(
@@ -537,6 +539,7 @@ def build_deployment_readiness_preview(*, service, market_intel_tables, schema_s
"candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_write_preview_safe": _run_review_preview_safe(candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_write, "candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_write_preview"),
"candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_package_preview_safe": _run_review_preview_safe(candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_package, "candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_package_preview"),
"candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_readiness_preview_safe": _run_review_preview_safe(candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_readiness, "candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_readiness_preview"),
+ "candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_receipt_preview_safe": _run_review_preview_safe(candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_receipt, "candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_receipt_preview"),
"candidate_queue_review_decision_writer_cli_status_safe": _run_review_preview_safe(candidate_queue_review_decision_writer_status, "candidate_queue_review_decision_writer_cli_blocked"),
"match_review_plan_preview_safe": bool(
match_review_plan["mode"] == "match_review_plan_preview"
@@ -810,6 +813,7 @@ def build_deployment_readiness_preview(*, service, market_intel_tables, schema_s
"candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_write": candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_write,
"candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_package": candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_package,
"candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_readiness": candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_readiness,
+ "candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_receipt": candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_receipt,
"candidate_queue_review_decision_writer_status": candidate_queue_review_decision_writer_status,
"match_review_plan": match_review_plan,
"opportunity_plan": opportunity_plan,
diff --git a/services/market_intel/phase.py b/services/market_intel/phase.py
index c373d2f..8712c9b 100644
--- a/services/market_intel/phase.py
+++ b/services/market_intel/phase.py
@@ -1,3 +1,3 @@
"""市場情報 rollout phase 單一來源。"""
-MARKET_INTEL_PHASE = "phase_109_candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_readiness"
+MARKET_INTEL_PHASE = "phase_110_candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_receipt"
diff --git a/templates/market_intel/disabled.html b/templates/market_intel/disabled.html
index 22e9f6a..bb14df6 100644
--- a/templates/market_intel/disabled.html
+++ b/templates/market_intel/disabled.html
@@ -766,6 +766,9 @@
+
@@ -1149,6 +1152,7 @@
const sampleCandidateQueueReviewAiSummaryPersistenceTelegramDispatchReportCatalogRecordWrite = sampleReviewRoot ? sampleReviewRoot.querySelector('[data-market-intel-sample-candidate-queue-review-ai-summary-persistence-telegram-dispatch-report-catalog-record-write]') : null;
const sampleCandidateQueueReviewAiSummaryPersistenceTelegramDispatchReportCatalogRecordRunPackage = sampleReviewRoot ? sampleReviewRoot.querySelector('[data-market-intel-sample-candidate-queue-review-ai-summary-persistence-telegram-dispatch-report-catalog-record-run-package]') : null;
const sampleCandidateQueueReviewAiSummaryPersistenceTelegramDispatchReportCatalogRecordRunReadiness = sampleReviewRoot ? sampleReviewRoot.querySelector('[data-market-intel-sample-candidate-queue-review-ai-summary-persistence-telegram-dispatch-report-catalog-record-run-readiness]') : null;
+ const sampleCandidateQueueReviewAiSummaryPersistenceTelegramDispatchReportCatalogRecordRunReceipt = sampleReviewRoot ? sampleReviewRoot.querySelector('[data-market-intel-sample-candidate-queue-review-ai-summary-persistence-telegram-dispatch-report-catalog-record-run-receipt]') : null;
const sampleReviewEndpoint = "{{ url_for('market_intel.market_intel_manual_sample_review') }}";
const sampleReviewEvaluateEndpoint = "{{ url_for('market_intel.market_intel_manual_sample_review_evaluate') }}";
const sampleCandidateHandoffEndpoint = "{{ url_for('market_intel.market_intel_manual_sample_candidate_handoff') }}";
@@ -1209,6 +1213,7 @@
const sampleCandidateQueueReviewAiSummaryPersistenceTelegramDispatchReportCatalogRecordWriteEndpoint = "{{ url_for('market_intel_review.market_intel_manual_sample_candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_write') }}";
const sampleCandidateQueueReviewAiSummaryPersistenceTelegramDispatchReportCatalogRecordRunPackageEndpoint = "{{ url_for('market_intel_review.market_intel_manual_sample_candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_package') }}";
const sampleCandidateQueueReviewAiSummaryPersistenceTelegramDispatchReportCatalogRecordRunReadinessEndpoint = "{{ url_for('market_intel_review.market_intel_manual_sample_candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_readiness') }}";
+ const sampleCandidateQueueReviewAiSummaryPersistenceTelegramDispatchReportCatalogRecordRunReceiptEndpoint = "{{ url_for('market_intel_review.market_intel_manual_sample_candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_receipt') }}";
const schedulerMeta = schedulerRoot ? schedulerRoot.querySelector('[data-market-intel-scheduler-meta]') : null;
const schedulerBody = schedulerRoot ? schedulerRoot.querySelector('[data-market-intel-scheduler-body]') : null;
const schedulerRefresh = schedulerRoot ? schedulerRoot.querySelector('[data-market-intel-scheduler-refresh]') : null;
@@ -8863,6 +8868,162 @@
}
};
+ const renderCandidateQueueReviewAiSummaryPersistenceTelegramDispatchReportCatalogRecordRunReceipt = data => {
+ const blockers = (data.blocked_reasons || []).join(' / ');
+ const gates = data.gates || [];
+ const readiness = data.telegram_dispatch_report_catalog_record_run_readiness || {};
+ const writer = data.telegram_dispatch_report_catalog_record_run_receipt_writer || {};
+ const smoke = data.telegram_dispatch_report_catalog_record_run_receipt_postwrite_smoke || {};
+ const operator = data.operator_telegram_dispatch_report_catalog_record_run_receipt || {};
+ const promotion = data.promotion_gate || {};
+ sampleReviewMeta.innerHTML = [
+ `mode=${data.mode || 'unknown'}`,
+ `receipt=${data.telegram_dispatch_report_catalog_record_run_receipt_passed ? 'pass' : 'blocked'}`,
+ `writer=${writer.catalog_record_written ? 'written' : 'blocked'}`,
+ `smoke=${smoke.postwrite_smoke_passed ? 'pass' : 'blocked'}`,
+ `api_db=${data.api_writes_database ? 'written' : 'blocked'}`
+ ].map(item => `${escapeHtml(item)}`).join('');
+ sampleReviewBody.innerHTML = `
+
此卡只審核外部 CLI 回貼的 Telegram dispatch report catalog record run receipt;API/UI 不讀 token、不寫檔、不執行 CLI、不開 DB、不寫 catalog record、不掛 scheduler。${blockers ? `阻擋:${escapeHtml(blockers)}` : ''}
+
+
+
RECEIPT GATES
+
${
+ gates.map(item => `
+
+
+ ${escapeHtml(item.key)}
+ ${escapeHtml(item.label)}
+
+
${item.passed ? 'PASS' : 'BLOCK'}
+
+ `).join('') || '
尚未提供 run receipt gate。
'
+ }
+
+
+
READINESS
+
+ ${[
+ ['provided', readiness.provided],
+ ['mode', readiness.mode || 'missing'],
+ ['passed', readiness.run_readiness_passed],
+ ['record_family', readiness.record_family || 'missing'],
+ ['statement_count', readiness.statement_count || 0],
+ ['expected_hash', readiness.expected_summary_payload_hash || 'missing']
+ ].map(([key, value]) => `
+
+
${escapeHtml(key)}
+
${escapeHtml(String(value))}
+
+ `).join('')}
+
+
+
+
WRITER OUTPUT
+
+ ${[
+ ['mode', writer.mode || 'missing'],
+ ['cli_executed', writer.catalog_record_cli_executed],
+ ['record_written', writer.catalog_record_written],
+ ['db_write', writer.database_write_executed],
+ ['commit', writer.database_commit_executed],
+ ['hash_match', writer.summary_payload_hash_matches_expected]
+ ].map(([key, value]) => `
+
+
${escapeHtml(key)}
+
${escapeHtml(String(value))}
+
+ `).join('')}
+
+
+
+
POSTWRITE SMOKE
+
+ ${[
+ ['mode', smoke.mode || 'missing'],
+ ['read_only', smoke.read_only_query_executed],
+ ['smoke_passed', smoke.postwrite_smoke_passed],
+ ['record_found', smoke.catalog_record_found],
+ ['hash_match', smoke.summary_payload_hash_matches_expected],
+ ['db_write', smoke.database_write_executed]
+ ].map(([key, value]) => `
+
+
${escapeHtml(key)}
+
${escapeHtml(String(value))}
+
+ `).join('')}
+
+
+
+
OPERATOR
+
+ ${[
+ ['receipt_path', operator.report_catalog_record_run_receipt_artifact_path || 'missing'],
+ ['readiness_path', operator.report_catalog_record_run_readiness_artifact_path_recorded],
+ ['writer_path', operator.catalog_record_writer_output_artifact_path_recorded],
+ ['smoke_path', operator.catalog_record_postwrite_smoke_artifact_path_recorded],
+ ['commit_gate', operator.operator_confirmed_catalog_record_commit_requires_separate_gate],
+ ['runtime_clear', operator.operator_confirmed_no_token_in_report_catalog_record_run_receipt && operator.operator_confirmed_no_api_file_write && operator.operator_confirmed_no_api_cli_execution && operator.operator_confirmed_no_api_catalog_record_write && operator.operator_confirmed_no_api_telegram_dispatch && operator.operator_confirmed_no_api_db_write && operator.operator_confirmed_no_llm_call && operator.operator_confirmed_no_scheduler_attach]
+ ].map(([key, value]) => `
+
+
${escapeHtml(key)}
+
${escapeHtml(String(value))}
+
+ `).join('')}
+
+
+
+
PROMOTION
+
+ ${[
+ ['next_phase', promotion.next_manual_phase || 'missing'],
+ ['commit_gate', promotion.report_catalog_record_commit_requires_separate_gate],
+ ['api_write', promotion.api_must_not_write_database],
+ ['api_cli', promotion.api_must_not_execute_cli],
+ ['api_catalog_record', promotion.api_must_not_write_catalog_record]
+ ].map(([key, value]) => `
+
+
${escapeHtml(key)}
+
${escapeHtml(String(value))}
+
+ `).join('')}
+
+
+
+ `;
+ };
+
+ const loadCandidateQueueReviewAiSummaryPersistenceTelegramDispatchReportCatalogRecordRunReceipt = async () => {
+ if (!sampleReviewMeta || !sampleReviewBody || !sampleReviewInput) return;
+ let parsed;
+ try {
+ parsed = JSON.parse(sampleReviewInput.value || '{}');
+ } catch (error) {
+ sampleReviewMeta.innerHTML = 'json_error';
+ sampleReviewBody.innerHTML = `JSON 格式錯誤:${escapeHtml(error.message)}
`;
+ return;
+ }
+ const body = parsed && parsed.sample_result ? parsed : { sample_result: parsed };
+ sampleReviewBody.innerHTML = '審核 queue review AI summary Telegram dispatch report catalog record run receipt 中...
';
+ try {
+ const response = await fetch(`${sampleCandidateQueueReviewAiSummaryPersistenceTelegramDispatchReportCatalogRecordRunReceiptEndpoint}?execute=false&apply_real_write=false`, {
+ method: 'POST',
+ credentials: 'same-origin',
+ headers: {
+ 'Content-Type': 'application/json',
+ 'X-CSRFToken': csrfToken
+ },
+ body: JSON.stringify(body)
+ });
+ const data = await response.json();
+ if (!response.ok && !data.mode) throw new Error(`HTTP ${response.status}`);
+ renderCandidateQueueReviewAiSummaryPersistenceTelegramDispatchReportCatalogRecordRunReceipt(data);
+ } catch (error) {
+ sampleReviewMeta.innerHTML = 'error';
+ sampleReviewBody.innerHTML = `queue review AI summary Telegram dispatch report catalog record run receipt 失敗:${escapeHtml(error.message)}
`;
+ }
+ };
+
const renderCandidateQueueReviewDecisionWriter = data => {
const blockers = (data.blocked_reasons || []).join(' / ');
const summary = data.statement_summary || {};
@@ -10644,6 +10805,9 @@
if (sampleCandidateQueueReviewAiSummaryPersistenceTelegramDispatchReportCatalogRecordRunReadiness) {
sampleCandidateQueueReviewAiSummaryPersistenceTelegramDispatchReportCatalogRecordRunReadiness.addEventListener('click', loadCandidateQueueReviewAiSummaryPersistenceTelegramDispatchReportCatalogRecordRunReadiness);
}
+ if (sampleCandidateQueueReviewAiSummaryPersistenceTelegramDispatchReportCatalogRecordRunReceipt) {
+ sampleCandidateQueueReviewAiSummaryPersistenceTelegramDispatchReportCatalogRecordRunReceipt.addEventListener('click', loadCandidateQueueReviewAiSummaryPersistenceTelegramDispatchReportCatalogRecordRunReceipt);
+ }
if (schedulerRefresh) {
schedulerRefresh.addEventListener('click', loadScheduler);
}
diff --git a/tests/test_market_intel_skeleton.py b/tests/test_market_intel_skeleton.py
index 924b97d..99bc529 100644
--- a/tests/test_market_intel_skeleton.py
+++ b/tests/test_market_intel_skeleton.py
@@ -1323,6 +1323,14 @@ def test_market_intel_preview_template_uses_safe_fetch_false_endpoint():
"data-market-intel-sample-candidate-queue-review-ai-summary-persistence-telegram-dispatch-report-catalog-record-run-readiness"
in template
)
+ assert (
+ "market_intel_review.market_intel_manual_sample_candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_receipt"
+ in template
+ )
+ assert (
+ "data-market-intel-sample-candidate-queue-review-ai-summary-persistence-telegram-dispatch-report-catalog-record-run-receipt"
+ in template
+ )
assert "X-CSRFToken" in template
assert "market_intel.market_intel_scheduler_plan" in template
assert "market_intel.market_intel_match_review_plan" in template
@@ -1359,7 +1367,7 @@ def test_legacy_source_bridge_default_is_planned_only():
bridge = MarketIntelService().build_legacy_source_bridge()
assert bridge["mode"] == "legacy_source_bridge_planned"
- assert bridge["phase"] == "phase_109_candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_readiness"
+ assert bridge["phase"] == "phase_110_candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_receipt"
assert bridge["execute_requested"] is False
assert bridge["read_only_query_executed"] is False
assert bridge["database_connection_opened"] is False
@@ -1517,7 +1525,7 @@ def test_mcp_tool_contract_preview_is_read_only_and_whitelisted():
contract = MarketIntelService().build_mcp_tool_contract()
assert contract["mode"] == "mcp_tool_contract_preview"
- assert contract["phase"] == "phase_109_candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_readiness"
+ assert contract["phase"] == "phase_110_candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_receipt"
assert contract["caller"] == "market_intel"
assert contract["contract_ready"] is True
assert contract["blocked_reasons"] == []
@@ -1650,7 +1658,7 @@ def test_mcp_activation_runbook_route_is_preview_only():
assert response.status_code == 200
assert data["mode"] == "mcp_activation_runbook_preview"
- assert data["phase"] == "phase_109_candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_readiness"
+ assert data["phase"] == "phase_110_candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_receipt"
assert data["deployment_actions_executed"] is False
assert data["docker_command_executed"] is False
assert data["ssh_command_executed"] is False
@@ -1663,7 +1671,7 @@ def test_mcp_fetch_gate_default_blocks_external_fetch():
gate = MarketIntelService().build_mcp_fetch_gate(fetch_requested=True)
assert gate["mode"] == "mcp_fetch_gate_planned"
- assert gate["phase"] == "phase_109_candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_readiness"
+ assert gate["phase"] == "phase_110_candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_receipt"
assert gate["fetch_requested"] is True
assert gate["manual_fetch_gate_open"] is False
assert gate["network_request_allowed"] is False
@@ -1733,7 +1741,7 @@ def test_mcp_fetch_gate_route_is_preview_only():
assert response.status_code == 200
assert data["mode"] == "mcp_fetch_gate_planned"
- assert data["phase"] == "phase_109_candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_readiness"
+ assert data["phase"] == "phase_110_candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_receipt"
assert data["fetch_requested"] is False
assert data["network_request_allowed"] is False
assert data["external_network_executed"] is False
@@ -1745,7 +1753,7 @@ def test_manual_sample_plan_preview_blocks_fetch_and_write():
plan = MarketIntelService().build_manual_sample_plan()
assert plan["mode"] == "manual_sample_fetch_plan_preview"
- assert plan["phase"] == "phase_109_candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_readiness"
+ assert plan["phase"] == "phase_110_candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_receipt"
assert plan["ready_for_manual_sample_fetch"] is False
assert plan["sample_fetch_executed"] is False
assert plan["external_network_executed"] is False
@@ -1793,7 +1801,7 @@ def test_manual_sample_plan_route_is_preview_only():
assert response.status_code == 200
assert data["mode"] == "manual_sample_fetch_plan_preview"
- assert data["phase"] == "phase_109_candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_readiness"
+ assert data["phase"] == "phase_110_candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_receipt"
assert data["sample_fetch_executed"] is False
assert data["external_network_executed"] is False
assert data["database_write_executed"] is False
@@ -1804,7 +1812,7 @@ def test_manual_sample_acceptance_preview_blocks_candidate_import():
acceptance = MarketIntelService().build_manual_sample_acceptance()
assert acceptance["mode"] == "manual_sample_acceptance_preview"
- assert acceptance["phase"] == "phase_109_candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_readiness"
+ assert acceptance["phase"] == "phase_110_candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_receipt"
assert acceptance["contract_ready"] is True
assert acceptance["sample_result_loaded"] is False
assert acceptance["sample_result_accepted"] is False
@@ -1846,7 +1854,7 @@ def test_manual_sample_acceptance_route_is_preview_only():
assert response.status_code == 200
assert data["mode"] == "manual_sample_acceptance_preview"
- assert data["phase"] == "phase_109_candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_readiness"
+ assert data["phase"] == "phase_110_candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_receipt"
assert data["sample_result_loaded"] is False
assert data["candidate_import_allowed"] is False
assert data["external_network_executed"] is False
@@ -1858,7 +1866,7 @@ def test_manual_sample_review_preview_is_planned_until_result_loaded():
review = MarketIntelService().build_manual_sample_review()
assert review["mode"] == "manual_sample_review_preview"
- assert review["phase"] == "phase_109_candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_readiness"
+ assert review["phase"] == "phase_110_candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_receipt"
assert review["contract_ready"] is True
assert review["sample_result_loaded"] is False
assert review["sample_result_reviewed"] is False
@@ -1969,7 +1977,7 @@ def test_manual_sample_review_route_is_preview_only():
assert response.status_code == 200
assert data["mode"] == "manual_sample_review_preview"
- assert data["phase"] == "phase_109_candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_readiness"
+ assert data["phase"] == "phase_110_candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_receipt"
assert data["sample_result_loaded"] is False
assert data["sample_result_reviewed"] is False
assert data["candidate_import_allowed"] is False
@@ -2008,7 +2016,7 @@ def test_manual_sample_review_evaluation_preview_accepts_payload_without_persist
)
assert review["mode"] == "manual_sample_review_evaluation_preview"
- assert review["phase"] == "phase_109_candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_readiness"
+ assert review["phase"] == "phase_110_candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_receipt"
assert review["review_request_type"] == "operator_posted_json"
assert review["payload_received"] is True
assert review["payload_valid_json_object"] is True
@@ -2070,7 +2078,7 @@ def test_manual_sample_review_evaluate_route_is_post_only_and_no_write():
assert response.status_code == 200
assert data["mode"] == "manual_sample_review_evaluation_preview"
- assert data["phase"] == "phase_109_candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_readiness"
+ assert data["phase"] == "phase_110_candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_receipt"
assert data["payload_received"] is True
assert data["payload_valid_json_object"] is True
assert data["payload_persisted"] is False
@@ -2150,7 +2158,7 @@ def test_manual_sample_candidate_handoff_preview_creates_candidates_without_pers
)
assert handoff["mode"] == "manual_sample_candidate_handoff_preview"
- assert handoff["phase"] == "phase_109_candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_readiness"
+ assert handoff["phase"] == "phase_110_candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_receipt"
assert handoff["payload_received"] is True
assert handoff["payload_valid_json_object"] is True
assert handoff["payload_persisted"] is False
@@ -2214,7 +2222,7 @@ def test_manual_sample_candidate_handoff_route_is_post_only_and_no_write():
assert response.status_code == 200
assert data["mode"] == "manual_sample_candidate_handoff_preview"
- assert data["phase"] == "phase_109_candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_readiness"
+ assert data["phase"] == "phase_110_candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_receipt"
assert data["payload_received"] is True
assert data["handoff_ready"] is True
assert data["candidate_handoff_created"] is True
@@ -2273,7 +2281,7 @@ def test_manual_sample_candidate_queue_draft_preview_builds_review_items_without
)
assert queue_draft["mode"] == "manual_sample_candidate_queue_draft_preview"
- assert queue_draft["phase"] == "phase_109_candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_readiness"
+ assert queue_draft["phase"] == "phase_110_candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_receipt"
assert queue_draft["payload_received"] is True
assert queue_draft["payload_valid_json_object"] is True
assert queue_draft["payload_persisted"] is False
@@ -2347,7 +2355,7 @@ def test_manual_sample_candidate_queue_draft_route_is_post_only_and_no_write():
assert response.status_code == 200
assert data["mode"] == "manual_sample_candidate_queue_draft_preview"
- assert data["phase"] == "phase_109_candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_readiness"
+ assert data["phase"] == "phase_110_candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_receipt"
assert data["payload_received"] is True
assert data["handoff_ready"] is True
assert data["queue_draft_ready"] is True
@@ -2410,7 +2418,7 @@ def test_manual_sample_candidate_queue_approval_preview_blocks_write_and_maps_ro
)
assert approval["mode"] == "manual_sample_candidate_queue_approval_preview"
- assert approval["phase"] == "phase_109_candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_readiness"
+ assert approval["phase"] == "phase_110_candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_receipt"
assert approval["payload_received"] is True
assert approval["payload_valid_json_object"] is True
assert approval["payload_persisted"] is False
@@ -2488,7 +2496,7 @@ def test_manual_sample_candidate_queue_approval_route_is_post_only_and_no_write(
assert response.status_code == 200
assert data["mode"] == "manual_sample_candidate_queue_approval_preview"
- assert data["phase"] == "phase_109_candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_readiness"
+ assert data["phase"] == "phase_110_candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_receipt"
assert data["payload_received"] is True
assert data["approval_preview_created"] is True
assert data["approval_request_created"] is False
@@ -2551,7 +2559,7 @@ def test_manual_sample_candidate_queue_transaction_preview_blocks_execution():
)
assert transaction["mode"] == "manual_sample_candidate_queue_transaction_preview"
- assert transaction["phase"] == "phase_109_candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_readiness"
+ assert transaction["phase"] == "phase_110_candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_receipt"
assert transaction["payload_received"] is True
assert transaction["payload_valid_json_object"] is True
assert transaction["payload_persisted"] is False
@@ -2631,7 +2639,7 @@ def test_manual_sample_candidate_queue_transaction_route_is_post_only_and_no_wri
assert response.status_code == 200
assert data["mode"] == "manual_sample_candidate_queue_transaction_preview"
- assert data["phase"] == "phase_109_candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_readiness"
+ assert data["phase"] == "phase_110_candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_receipt"
assert data["payload_received"] is True
assert data["transaction_preview_created"] is True
assert data["transaction_ready"] is False
@@ -8342,7 +8350,7 @@ def test_candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_
"candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_input_preview"
)
assert data["phase"] == (
- "phase_109_candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_readiness"
+ "phase_110_candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_receipt"
)
assert data["telegram_dispatch_report_input_ready"] is False
assert data["summary_persistence_telegram_dispatch_report_input_ready"] is False
@@ -8417,7 +8425,7 @@ def test_candidate_queue_review_ai_summary_persistence_telegram_dispatch_archive
"candidate_queue_review_ai_summary_persistence_telegram_dispatch_archive_summary_preview"
)
assert data["phase"] == (
- "phase_109_candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_readiness"
+ "phase_110_candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_receipt"
)
assert data["telegram_dispatch_archive_summary_ready"] is False
assert (
@@ -8689,7 +8697,7 @@ def test_candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_
"candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_input_preview"
)
assert data["phase"] == (
- "phase_109_candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_readiness"
+ "phase_110_candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_receipt"
)
assert data["telegram_dispatch_report_input_ready"] is False
assert data["summary_persistence_telegram_dispatch_report_input_ready"] is False
@@ -8977,7 +8985,7 @@ def test_candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_
"candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_run_package_preview"
)
assert data["phase"] == (
- "phase_109_candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_readiness"
+ "phase_110_candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_receipt"
)
assert data["telegram_dispatch_report_run_package_ready"] is False
assert (
@@ -9287,7 +9295,7 @@ def test_candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_
"candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_run_readiness_preview"
)
assert data["phase"] == (
- "phase_109_candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_readiness"
+ "phase_110_candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_receipt"
)
assert data["telegram_dispatch_report_run_readiness_ready"] is False
assert (
@@ -9590,7 +9598,7 @@ def test_candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_
"candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_run_receipt_preview"
)
assert data["phase"] == (
- "phase_109_candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_readiness"
+ "phase_110_candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_receipt"
)
assert data["telegram_dispatch_report_run_receipt_passed"] is False
assert (
@@ -9849,7 +9857,7 @@ def test_candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_
"candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_closeout_preview"
)
assert data["phase"] == (
- "phase_109_candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_readiness"
+ "phase_110_candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_receipt"
)
assert data["telegram_dispatch_report_closeout_passed"] is False
assert (
@@ -10122,7 +10130,7 @@ def test_candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_
"candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_archive_preview"
)
assert data["phase"] == (
- "phase_109_candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_readiness"
+ "phase_110_candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_receipt"
)
assert data["telegram_dispatch_report_archive_passed"] is False
assert (
@@ -10370,7 +10378,7 @@ def test_candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_
"candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_archive_summary_preview"
)
assert data["phase"] == (
- "phase_109_candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_readiness"
+ "phase_110_candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_receipt"
)
assert data["telegram_dispatch_report_archive_summary_passed"] is False
assert (
@@ -10600,7 +10608,7 @@ def test_candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_
"candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_handoff_preview"
)
assert data["phase"] == (
- "phase_109_candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_readiness"
+ "phase_110_candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_receipt"
)
assert data["telegram_dispatch_report_catalog_handoff_passed"] is False
assert data["summary_persistence_telegram_dispatch_report_catalog_handoff_passed"] is False
@@ -10837,7 +10845,7 @@ def test_candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_
"candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_index_preview"
)
assert data["phase"] == (
- "phase_109_candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_readiness"
+ "phase_110_candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_receipt"
)
assert data["telegram_dispatch_report_catalog_index_passed"] is False
assert data["summary_persistence_telegram_dispatch_report_catalog_index_passed"] is False
@@ -11079,7 +11087,7 @@ def test_candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_
"candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_write_preflight_preview"
)
assert data["phase"] == (
- "phase_109_candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_readiness"
+ "phase_110_candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_receipt"
)
assert data["telegram_dispatch_report_catalog_write_preflight_passed"] is False
assert (
@@ -11355,7 +11363,7 @@ def test_candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_
"candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_write_preview"
)
assert data["phase"] == (
- "phase_109_candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_readiness"
+ "phase_110_candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_receipt"
)
assert data["telegram_dispatch_report_catalog_record_write_passed"] is False
assert (
@@ -11632,7 +11640,7 @@ def test_candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_
"candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_package_preview"
)
assert data["phase"] == (
- "phase_109_candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_readiness"
+ "phase_110_candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_receipt"
)
assert data["telegram_dispatch_report_catalog_record_run_package_passed"] is False
assert (
@@ -11911,7 +11919,7 @@ def test_candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_
"candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_readiness_preview"
)
assert data["phase"] == (
- "phase_109_candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_readiness"
+ "phase_110_candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_receipt"
)
assert data["telegram_dispatch_report_catalog_record_run_readiness_passed"] is False
assert (
@@ -11957,6 +11965,332 @@ def test_candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_
assert TEST_APPROVAL_TOKEN not in payload
+def test_candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_receipt_contract_only():
+ from services.market_intel.candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_receipt import (
+ build_candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_receipt,
+ )
+
+ readiness = {
+ "mode": "candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_readiness_preview",
+ "target_operation": "manual_market_intel_report_catalog_record_run_readiness",
+ "telegram_dispatch_report_catalog_record_run_readiness_passed": True,
+ "summary_persistence_telegram_dispatch_report_catalog_record_run_readiness_passed": True,
+ "report_catalog_record_run_readiness_passed": True,
+ "ready_for_next_manual_phase": True,
+ "ready_for_cli_operator_run": True,
+ "ready_for_market_intel_report_catalog_record_cli_operator_run": True,
+ "ready_for_market_intel_report_catalog_record_run_receipt": True,
+ "ready_for_market_intel_report_catalog_record_cli_run": False,
+ "ready_for_market_intel_report_catalog_record_commit": False,
+ "statement_count": 1,
+ "expected_summary_payload_hash": "e" * 64,
+ "catalog_record_run_readiness_manifest": {
+ "manifest_version": "market_intel_report_catalog_record_run_readiness_v1",
+ "source_mode": "candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_package_preview",
+ "source_operation": "package_market_intel_report_catalog_record_run",
+ "target_table": "market_alert_review_queue",
+ "target_column": "metadata_json",
+ "record_family": "market_intel_telegram_dispatch_audit_report",
+ "report_output_hash": "f" * 64,
+ "expected_summary_payload_hash": "e" * 64,
+ "statement_count": 1,
+ "source_artifacts": {
+ "record_write_gate": True,
+ "write_preflight": True,
+ "report_output": True,
+ "backup": True,
+ "dry_run": True,
+ },
+ "manual_cli_run_command": {
+ "script_path": "scripts/market_intel/write_report_catalog_record.py",
+ "dry_run_command": (
+ "python scripts/market_intel/write_report_catalog_record.py --input payload --dry-run"
+ ),
+ "executed_by_api": False,
+ "writes_catalog_record": True,
+ "requires_approval_env_var": "MARKET_INTEL_QUEUE_WRITE_APPROVAL",
+ },
+ "execution_boundaries": {
+ "api_writes_file": False,
+ "api_executes_cli": False,
+ "api_writes_catalog_record": False,
+ "api_writes_database": False,
+ "api_dispatches_telegram": False,
+ "api_calls_llm": False,
+ "api_attaches_scheduler": False,
+ },
+ },
+ "telegram_dispatch_report_catalog_record_run_package": {
+ "mode": "candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_package_preview",
+ "catalog_record_run_package_passed": True,
+ "record_family": "market_intel_telegram_dispatch_audit_report",
+ "report_output_hash": "f" * 64,
+ "expected_summary_payload_hash": "e" * 64,
+ },
+ "promotion_gate": {
+ "allowed": True,
+ "next_manual_phase": "market_intel_report_catalog_record_run_receipt",
+ "requires_real_db_write": True,
+ "requires_cli_run": True,
+ "requires_postwrite_smoke": True,
+ "report_catalog_record_run_receipt_requires_separate_gate": True,
+ "report_catalog_record_commit_requires_separate_gate": True,
+ },
+ "telegram_dispatch_report_catalog_record_run_readiness_sections": [
+ {"key": "catalog_record_run_readiness_identity"},
+ {"key": "catalog_record_run_readiness_artifacts"},
+ {"key": "catalog_record_run_readiness_command"},
+ {"key": "catalog_record_run_readiness_safety"},
+ ],
+ "safe_boundaries": [
+ "do_not_read_approval_token_from_report_catalog_record_run_readiness_api",
+ "do_not_read_telegram_token_from_report_catalog_record_run_readiness_api",
+ "do_not_call_llm_from_report_catalog_record_run_readiness",
+ "do_not_generate_report_from_report_catalog_record_run_readiness_api",
+ "do_not_write_report_catalog_record_run_readiness_artifact_from_api",
+ "do_not_execute_catalog_record_cli_from_report_catalog_record_run_readiness_api",
+ "do_not_write_report_catalog_record_from_report_catalog_record_run_readiness_api",
+ "do_not_dispatch_telegram_from_report_catalog_record_run_readiness_api",
+ "do_not_open_database_connection_from_report_catalog_record_run_readiness",
+ "do_not_update_review_state_from_report_catalog_record_run_readiness",
+ "do_not_attach_scheduler_from_report_catalog_record_run_readiness",
+ "future_market_intel_report_catalog_record_run_receipt_must_use_separate_gate",
+ ],
+ }
+ receipt_payload = {
+ "mode": "manual_market_intel_report_catalog_record_run_receipt",
+ "writer_output": {
+ "mode": "market_intel_report_catalog_record_run_receipt",
+ "catalog_record_cli_executed": True,
+ "catalog_record_written": True,
+ "target_table": "market_alert_review_queue",
+ "target_column": "metadata_json",
+ "record_family": "market_intel_telegram_dispatch_audit_report",
+ "catalog_record_key": "market-intel-report-20260520",
+ "statement_count": 1,
+ "affected_count": 1,
+ "database_connection_opened": True,
+ "explicit_transaction_opened": True,
+ "database_write_executed": True,
+ "database_commit_executed": True,
+ "expected_summary_payload_hash": "e" * 64,
+ "report_output_hash": "f" * 64,
+ },
+ "postwrite_smoke_result": {
+ "mode": "market_intel_report_catalog_record_postwrite_smoke_read_only",
+ "postwrite_smoke_passed": True,
+ "read_only_query_executed": True,
+ "catalog_record_found": True,
+ "target_table": "market_alert_review_queue",
+ "target_column": "metadata_json",
+ "record_family": "market_intel_telegram_dispatch_audit_report",
+ "catalog_record_key": "market-intel-report-20260520",
+ "database_connection_opened": True,
+ "database_write_executed": False,
+ "database_commit_executed": False,
+ "expected_summary_payload_hash": "e" * 64,
+ },
+ }
+ operator_evidence = {
+ "report_catalog_record_run_receipt_artifact_path": (
+ "artifacts/market_intel/catalog-record-run-receipt.json"
+ ),
+ "report_catalog_record_run_readiness_artifact_path": (
+ "artifacts/market_intel/catalog-record-run-readiness.json"
+ ),
+ "report_catalog_record_run_package_artifact_path": (
+ "artifacts/market_intel/catalog-record-run-package.json"
+ ),
+ "catalog_record_payload_manifest_path": (
+ "artifacts/market_intel/catalog-record-payload.json"
+ ),
+ "catalog_record_backup_artifact_path": (
+ "artifacts/market_intel/catalog-record-backup.json"
+ ),
+ "catalog_record_write_dry_run_artifact_path": (
+ "artifacts/market_intel/catalog-record-dry-run.json"
+ ),
+ "catalog_record_writer_output_artifact_path": (
+ "artifacts/market_intel/catalog-record-writer-output.json"
+ ),
+ "catalog_record_postwrite_smoke_artifact_path": (
+ "artifacts/market_intel/catalog-record-postwrite-smoke.json"
+ ),
+ "catalog_record_cli_command": (
+ "python scripts/market_intel/write_report_catalog_record.py --input payload"
+ ),
+ "operator_confirmed_report_catalog_record_run_receipt": True,
+ "operator_confirmed_catalog_record_run_readiness_reviewed": True,
+ "operator_confirmed_catalog_record_cli_output_reviewed": True,
+ "operator_confirmed_catalog_record_postwrite_smoke_passed": True,
+ "operator_confirmed_catalog_record_backup_available": True,
+ "operator_confirmed_catalog_record_commit_requires_separate_gate": True,
+ "operator_confirmed_no_token_in_report_catalog_record_run_receipt": True,
+ "operator_confirmed_no_api_file_write": True,
+ "operator_confirmed_no_api_cli_execution": True,
+ "operator_confirmed_no_api_catalog_record_write": True,
+ "operator_confirmed_no_api_telegram_dispatch": True,
+ "operator_confirmed_no_api_db_write": True,
+ "operator_confirmed_no_llm_call": True,
+ "operator_confirmed_no_scheduler_attach": True,
+ "catalog_record_run_receipt_notes": "manual run receipt reviewed",
+ }
+
+ receipt = build_candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_receipt(
+ telegram_dispatch_report_catalog_record_run_readiness=readiness,
+ catalog_record_run_receipt=receipt_payload,
+ operator_evidence=operator_evidence,
+ execute_requested=True,
+ )
+ token_leak = build_candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_receipt(
+ telegram_dispatch_report_catalog_record_run_readiness=readiness,
+ catalog_record_run_receipt=receipt_payload,
+ operator_evidence={**operator_evidence, "telegram_token": TEST_APPROVAL_TOKEN},
+ )
+ apply_write = build_candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_receipt(
+ telegram_dispatch_report_catalog_record_run_readiness=readiness,
+ catalog_record_run_receipt=receipt_payload,
+ operator_evidence=operator_evidence,
+ apply_real_write=True,
+ )
+
+ assert receipt["mode"] == (
+ "candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_receipt_preview"
+ )
+ assert receipt["target_operation"] == (
+ "review_manual_market_intel_report_catalog_record_run_receipt"
+ )
+ assert receipt["telegram_dispatch_report_catalog_record_run_receipt_passed"] is True
+ assert (
+ receipt[
+ "summary_persistence_telegram_dispatch_report_catalog_record_run_receipt_passed"
+ ]
+ is True
+ )
+ assert receipt["report_catalog_record_run_receipt_passed"] is True
+ assert receipt["ready_for_market_intel_report_catalog_record_commit"] is True
+ assert receipt["ready_for_market_intel_report_catalog_record_cli_run"] is False
+ assert receipt["ready_for_real_write"] is False
+ assert receipt["ready_for_api_database_write"] is False
+ assert receipt["api_executes_cli"] is False
+ assert receipt["api_writes_file"] is False
+ assert receipt["api_writes_database"] is False
+ assert receipt["catalog_record_written"] is False
+ assert receipt["catalog_record_run_receipt_gate_file_written"] is False
+ assert receipt["catalog_record_run_receipt_executed"] is False
+ assert receipt["catalog_record_commit_executed"] is False
+ assert receipt["database_connection_opened"] is False
+ assert receipt["database_write_executed"] is False
+ assert receipt["scheduler_attached"] is False
+ assert receipt["blocked_reasons"] == []
+ assert (
+ receipt["telegram_dispatch_report_catalog_record_run_receipt_writer"][
+ "catalog_record_written"
+ ]
+ is True
+ )
+ assert (
+ receipt["telegram_dispatch_report_catalog_record_run_receipt_postwrite_smoke"][
+ "postwrite_smoke_passed"
+ ]
+ is True
+ )
+ assert receipt["promotion_gate"]["next_manual_phase"] == (
+ "market_intel_report_catalog_record_commit"
+ )
+ assert token_leak["telegram_dispatch_report_catalog_record_run_receipt_passed"] is False
+ assert "report_catalog_record_run_receipt_no_token_submitted_to_api" in token_leak[
+ "blocked_reasons"
+ ]
+ assert (
+ "report_catalog_record_run_receipt_apply_real_write_not_requested_from_api"
+ in apply_write["blocked_reasons"]
+ )
+ assert (
+ "do_not_execute_catalog_record_cli_from_report_catalog_record_run_receipt_api"
+ in receipt["safe_boundaries"]
+ )
+ assert TEST_APPROVAL_TOKEN not in json.dumps(
+ token_leak,
+ ensure_ascii=False,
+ sort_keys=True,
+ )
+
+
+def test_candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_receipt_route_is_post_only_and_no_write():
+ from routes.market_intel_routes import market_intel_bp
+ from routes.market_intel_review_routes import market_intel_review_bp
+
+ app = Flask(__name__)
+ app.secret_key = "test-secret"
+ app.register_blueprint(market_intel_bp)
+ app.register_blueprint(market_intel_review_bp)
+ client = app.test_client()
+ with client.session_transaction() as session:
+ session["logged_in"] = True
+
+ get_response = client.get(
+ "/api/market_intel/manual_sample_review/"
+ "candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_receipt"
+ )
+ response = client.post(
+ "/api/market_intel/manual_sample_review/"
+ "candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_receipt"
+ "?execute=true&apply_real_write=true",
+ json={"sample_result": {}},
+ )
+ data = response.get_json()
+ payload = json.dumps(data, ensure_ascii=False, sort_keys=True)
+
+ assert get_response.status_code == 405
+ assert response.status_code == 200
+ assert data["mode"] == (
+ "candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_receipt_preview"
+ )
+ assert data["phase"] == (
+ "phase_110_candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_receipt"
+ )
+ assert data["telegram_dispatch_report_catalog_record_run_receipt_passed"] is False
+ assert (
+ data[
+ "summary_persistence_telegram_dispatch_report_catalog_record_run_receipt_passed"
+ ]
+ is False
+ )
+ assert data["report_catalog_record_run_receipt_passed"] is False
+ assert data["ready_for_market_intel_report_catalog_record_commit"] is False
+ assert data["ready_for_market_intel_report_catalog_record_cli_run"] is False
+ assert data["ready_for_real_write"] is False
+ assert data["ready_for_api_database_write"] is False
+ assert data["api_executes_cli"] is False
+ assert data["api_writes_file"] is False
+ assert data["api_writes_database"] is False
+ assert data["catalog_record_written"] is False
+ assert data["catalog_record_run_receipt_gate_file_written"] is False
+ assert data["catalog_record_run_receipt_executed"] is False
+ assert data["catalog_record_commit_executed"] is False
+ assert data["database_connection_opened"] is False
+ assert data["database_write_executed"] is False
+ assert data["scheduler_attached"] is False
+ assert data["telegram_dispatch_report_catalog_record_run_readiness"]["mode"] == (
+ "candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_readiness_preview"
+ )
+ assert "report_catalog_record_run_readiness_passed" in data["blocked_reasons"]
+ assert (
+ "catalog_record_run_receipt_writer_output_provided"
+ in data["blocked_reasons"]
+ )
+ assert (
+ "report_catalog_record_run_receipt_apply_real_write_not_requested_from_api"
+ in data["blocked_reasons"]
+ )
+ assert (
+ "do_not_execute_catalog_record_cli_from_report_catalog_record_run_receipt_api"
+ in data["safe_boundaries"]
+ )
+ assert TEST_APPROVAL_TOKEN not in payload
+
+
def test_candidate_queue_writer_preflight_route_is_post_only_and_no_write():
from routes.market_intel_routes import market_intel_bp
@@ -11999,7 +12333,7 @@ def test_candidate_queue_writer_preflight_route_is_post_only_and_no_write():
assert response.status_code == 200
assert data["mode"] == "candidate_queue_writer_preflight_planned"
- assert data["phase"] == "phase_109_candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_readiness"
+ assert data["phase"] == "phase_110_candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_receipt"
assert data["execute_requested"] is False
assert data["read_only_query_executed"] is False
assert data["database_connection_opened"] is False
@@ -12056,7 +12390,7 @@ def test_candidate_queue_writer_status_route_never_leaks_approval_token(monkeypa
assert response.status_code == 200
assert data["mode"] == "candidate_queue_writer_cli_blocked"
- assert data["phase"] == "phase_109_candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_readiness"
+ assert data["phase"] == "phase_110_candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_receipt"
assert data["execute_requested"] is True
assert data["apply_real_write_requested"] is True
assert data["approval_token_present"] is False
@@ -12145,7 +12479,7 @@ def test_candidate_queue_writer_postwrite_smoke_route_is_post_only_and_no_write(
assert response.status_code == 200
assert data["mode"] == "candidate_queue_writer_postwrite_smoke_planned"
- assert data["phase"] == "phase_109_candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_readiness"
+ assert data["phase"] == "phase_110_candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_receipt"
assert data["execute_requested"] is False
assert data["read_only_query_executed"] is False
assert data["database_connection_opened"] is False
@@ -12199,7 +12533,7 @@ def test_candidate_queue_writer_operator_drill_route_is_post_only_and_no_write()
assert response.status_code == 200
assert data["mode"] == "candidate_queue_writer_operator_drill_preview"
- assert data["phase"] == "phase_109_candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_readiness"
+ assert data["phase"] == "phase_110_candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_receipt"
assert data["operator_drill_ready"] is True
assert data["api_executes_cli"] is False
assert data["api_reads_approval_token"] is False
@@ -12255,7 +12589,7 @@ def test_candidate_queue_writer_run_package_route_is_post_only_and_no_write():
assert response.status_code == 200
assert data["mode"] == "candidate_queue_writer_run_package_preview"
- assert data["phase"] == "phase_109_candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_readiness"
+ assert data["phase"] == "phase_110_candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_receipt"
assert data["package_ready"] is True
assert data["package_artifact_created"] is False
assert data["api_writes_file"] is False
@@ -12321,7 +12655,7 @@ def test_candidate_queue_writer_run_readiness_route_is_post_only_and_no_write():
assert response.status_code == 200
assert data["mode"] == "candidate_queue_writer_run_readiness_preview"
- assert data["phase"] == "phase_109_candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_readiness"
+ assert data["phase"] == "phase_110_candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_receipt"
assert data["ready_for_cli_operator_run"] is True
assert data["ready_for_api_database_write"] is False
assert data["api_executes_cli"] is False
@@ -12623,7 +12957,7 @@ def test_candidate_queue_writer_run_receipt_route_accepts_inline_payload_no_writ
assert response.status_code == 200
assert data["mode"] == "candidate_queue_writer_run_receipt_preview"
- assert data["phase"] == "phase_109_candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_readiness"
+ assert data["phase"] == "phase_110_candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_receipt"
assert data["receipt_passed"] is True
assert data["ready_for_api_database_write"] is False
assert data["ready_for_scheduler_attach"] is False
@@ -12671,7 +13005,7 @@ def test_candidate_queue_writer_run_closeout_route_is_post_only_and_no_write():
assert response.status_code == 200
assert data["mode"] == "candidate_queue_writer_run_closeout_preview"
- assert data["phase"] == "phase_109_candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_readiness"
+ assert data["phase"] == "phase_110_candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_receipt"
assert data["closeout_passed"] is True
assert data["ready_for_next_manual_phase"] is True
assert data["ready_for_api_database_write"] is False
@@ -12720,7 +13054,7 @@ def test_candidate_queue_review_handoff_route_is_post_only_and_no_write():
assert response.status_code == 200
assert data["mode"] == "candidate_queue_review_handoff_preview"
- assert data["phase"] == "phase_109_candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_readiness"
+ assert data["phase"] == "phase_110_candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_receipt"
assert data["handoff_ready"] is True
assert data["ready_for_manual_queue_review"] is True
assert data["ready_for_api_database_write"] is False
@@ -12778,7 +13112,7 @@ def test_candidate_queue_review_inventory_route_is_post_only_and_no_write():
assert get_response.status_code == 405
assert response.status_code == 200
assert data["mode"] == "candidate_queue_review_inventory_preview"
- assert data["phase"] == "phase_109_candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_readiness"
+ assert data["phase"] == "phase_110_candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_receipt"
assert data["execute_requested"] is False
assert data["review_inventory_ready"] is False
assert data["ready_for_human_decision_review"] is False
@@ -12844,7 +13178,7 @@ def test_candidate_queue_review_decision_route_is_post_only_and_no_write():
assert get_response.status_code == 405
assert response.status_code == 200
assert data["mode"] == "candidate_queue_review_decision_preview"
- assert data["phase"] == "phase_109_candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_readiness"
+ assert data["phase"] == "phase_110_candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_receipt"
assert data["decision_ready"] is False
assert data["ready_for_human_decision_record"] is False
assert data["ready_for_api_review_state_update"] is False
@@ -12915,7 +13249,7 @@ def test_candidate_queue_review_decision_approval_route_is_post_only_and_no_writ
assert get_response.status_code == 405
assert response.status_code == 200
assert data["mode"] == "candidate_queue_review_decision_approval_preview"
- assert data["phase"] == "phase_109_candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_readiness"
+ assert data["phase"] == "phase_110_candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_receipt"
assert data["approval_ready"] is False
assert data["ready_for_review_state_transaction_preview"] is False
assert data["ready_for_cli_decision_writer"] is False
@@ -12991,7 +13325,7 @@ def test_candidate_queue_review_decision_transaction_route_is_post_only_and_no_w
assert get_response.status_code == 405
assert response.status_code == 200
assert data["mode"] == "candidate_queue_review_decision_transaction_preview"
- assert data["phase"] == "phase_109_candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_readiness"
+ assert data["phase"] == "phase_110_candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_receipt"
assert data["transaction_preview_created"] is False
assert data["transaction_ready"] is False
assert data["ready_for_manual_shell_update_window"] is False
@@ -13073,7 +13407,7 @@ def test_candidate_queue_review_decision_writer_status_route_is_post_only_and_no
assert get_response.status_code == 405
assert response.status_code == 200
assert data["mode"] == "candidate_queue_review_decision_writer_cli_blocked"
- assert data["phase"] == "phase_109_candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_readiness"
+ assert data["phase"] == "phase_110_candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_receipt"
assert data["execute_requested"] is True
assert data["apply_real_write_requested"] is True
assert data["approval_token_present"] is False
@@ -13159,7 +13493,7 @@ def test_candidate_queue_review_decision_writer_preflight_route_is_post_only_and
assert data["mode"] == (
"candidate_queue_review_decision_writer_preflight_preview"
)
- assert data["phase"] == "phase_109_candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_readiness"
+ assert data["phase"] == "phase_110_candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_receipt"
assert data["execute_requested"] is True
assert data["apply_real_write_requested"] is True
assert data["read_only_query_executed"] is False
@@ -13242,7 +13576,7 @@ def test_candidate_queue_review_decision_writer_postwrite_smoke_route_is_post_on
assert data["mode"] == (
"candidate_queue_review_decision_writer_postwrite_smoke_planned"
)
- assert data["phase"] == "phase_109_candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_readiness"
+ assert data["phase"] == "phase_110_candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_receipt"
assert data["execute_requested"] is False
assert data["read_only_query_executed"] is False
assert data["database_connection_opened"] is False
@@ -13325,7 +13659,7 @@ def test_candidate_queue_review_decision_writer_operator_drill_route_is_post_onl
assert data["mode"] == (
"candidate_queue_review_decision_writer_operator_drill_preview"
)
- assert data["phase"] == "phase_109_candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_readiness"
+ assert data["phase"] == "phase_110_candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_receipt"
assert data["operator_drill_ready"] is False
assert data["ready_for_api_review_state_update"] is False
assert data["ready_for_api_database_write"] is False
@@ -13411,7 +13745,7 @@ def test_candidate_queue_review_decision_writer_run_package_route_is_post_only_a
assert data["mode"] == (
"candidate_queue_review_decision_writer_run_package_preview"
)
- assert data["phase"] == "phase_109_candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_readiness"
+ assert data["phase"] == "phase_110_candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_receipt"
assert data["package_ready"] is False
assert data["package_artifact_created"] is False
assert data["ready_for_api_review_state_update"] is False
@@ -13502,7 +13836,7 @@ def test_candidate_queue_review_decision_writer_run_readiness_route_is_post_only
"candidate_queue_review_decision_writer_run_readiness_preview"
)
assert data["phase"] == (
- "phase_109_candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_readiness"
+ "phase_110_candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_receipt"
)
assert data["ready_for_cli_operator_run"] is False
assert data["ready_for_api_review_state_update"] is False
@@ -13612,7 +13946,7 @@ def test_candidate_queue_review_decision_writer_run_receipt_route_is_post_only_a
"candidate_queue_review_decision_writer_run_receipt_preview"
)
assert data["phase"] == (
- "phase_109_candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_readiness"
+ "phase_110_candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_receipt"
)
assert data["receipt_passed"] is False
assert data["ready_for_api_review_state_update"] is False
@@ -13698,7 +14032,7 @@ def test_candidate_queue_review_decision_writer_run_closeout_route_is_post_only_
"candidate_queue_review_decision_writer_run_closeout_preview"
)
assert data["phase"] == (
- "phase_109_candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_readiness"
+ "phase_110_candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_receipt"
)
assert data["closeout_passed"] is False
assert data["ready_for_api_review_state_update"] is False
@@ -13755,7 +14089,7 @@ def test_candidate_queue_review_decision_post_closeout_inventory_route_is_post_o
"candidate_queue_review_decision_post_closeout_inventory_preview"
)
assert data["phase"] == (
- "phase_109_candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_readiness"
+ "phase_110_candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_receipt"
)
assert data["post_closeout_inventory_ready"] is False
assert data["ready_for_api_review_state_update"] is False
@@ -13808,7 +14142,7 @@ def test_candidate_queue_review_completion_archive_route_is_post_only_and_no_wri
assert get_response.status_code == 405
assert response.status_code == 200
assert data["mode"] == "candidate_queue_review_completion_archive_preview"
- assert data["phase"] == "phase_109_candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_readiness"
+ assert data["phase"] == "phase_110_candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_receipt"
assert data["review_completion_archive_ready"] is False
assert data["archive_manifest_ready"] is False
assert data["ready_for_api_review_state_update"] is False
@@ -13861,7 +14195,7 @@ def test_candidate_queue_review_archive_summary_route_is_post_only_and_no_write(
assert get_response.status_code == 405
assert response.status_code == 200
assert data["mode"] == "candidate_queue_review_archive_summary_preview"
- assert data["phase"] == "phase_109_candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_readiness"
+ assert data["phase"] == "phase_110_candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_receipt"
assert data["archive_summary_ready"] is False
assert data["summary_input_ready"] is False
assert data["ready_for_ai_summary_generation"] is False
@@ -13922,7 +14256,7 @@ def test_candidate_queue_review_ai_summary_preflight_route_is_post_only_and_no_w
assert get_response.status_code == 405
assert response.status_code == 200
assert data["mode"] == "candidate_queue_review_ai_summary_preflight_preview"
- assert data["phase"] == "phase_109_candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_readiness"
+ assert data["phase"] == "phase_110_candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_receipt"
assert data["ai_summary_preflight_ready"] is False
assert data["ready_for_manual_ollama_summary_run"] is False
assert data["ready_for_ai_summary_generation"] is False
@@ -13991,7 +14325,7 @@ def test_candidate_queue_review_ai_summary_run_package_route_is_post_only_and_no
assert get_response.status_code == 405
assert response.status_code == 200
assert data["mode"] == "candidate_queue_review_ai_summary_run_package_preview"
- assert data["phase"] == "phase_109_candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_readiness"
+ assert data["phase"] == "phase_110_candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_receipt"
assert data["ai_summary_run_package_ready"] is False
assert data["ready_for_manual_ollama_summary_run"] is False
assert data["ready_for_ai_summary_generation"] is False
@@ -14063,7 +14397,7 @@ def test_candidate_queue_review_ai_summary_output_receipt_route_is_post_only_and
assert get_response.status_code == 405
assert response.status_code == 200
assert data["mode"] == "candidate_queue_review_ai_summary_output_receipt_preview"
- assert data["phase"] == "phase_109_candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_readiness"
+ assert data["phase"] == "phase_110_candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_receipt"
assert data["ai_summary_output_receipt_ready"] is False
assert data["ready_for_summary_persistence_review"] is False
assert data["manual_ai_summary_output_provided"] is False
@@ -14136,7 +14470,7 @@ def test_candidate_queue_review_ai_summary_persistence_preflight_route_is_post_o
"candidate_queue_review_ai_summary_persistence_preflight_preview"
)
assert data["phase"] == (
- "phase_109_candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_readiness"
+ "phase_110_candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_receipt"
)
assert data["summary_persistence_preflight_ready"] is False
assert data["ready_for_summary_transaction_preview"] is False
@@ -14207,7 +14541,7 @@ def test_candidate_queue_review_ai_summary_persistence_transaction_route_is_post
"candidate_queue_review_ai_summary_persistence_transaction_preview"
)
assert data["phase"] == (
- "phase_109_candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_readiness"
+ "phase_110_candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_receipt"
)
assert data["summary_persistence_transaction_ready"] is False
assert data["ready_for_summary_persistence_writer_gate"] is False
@@ -14272,7 +14606,7 @@ def test_candidate_queue_review_ai_summary_persistence_writer_preflight_route_is
"candidate_queue_review_ai_summary_persistence_writer_preflight_preview"
)
assert data["phase"] == (
- "phase_109_candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_readiness"
+ "phase_110_candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_receipt"
)
assert data["summary_persistence_writer_preflight_ready"] is False
assert data["ready_for_summary_persistence_run_package"] is False
@@ -14343,7 +14677,7 @@ def test_candidate_queue_review_ai_summary_persistence_run_package_route_is_post
"candidate_queue_review_ai_summary_persistence_run_package_preview"
)
assert data["phase"] == (
- "phase_109_candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_readiness"
+ "phase_110_candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_receipt"
)
assert data["package_ready"] is False
assert data["ready_for_summary_persistence_run_readiness"] is False
@@ -14416,7 +14750,7 @@ def test_candidate_queue_review_ai_summary_persistence_run_readiness_route_is_po
"candidate_queue_review_ai_summary_persistence_run_readiness_preview"
)
assert data["phase"] == (
- "phase_109_candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_readiness"
+ "phase_110_candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_receipt"
)
assert data["run_readiness_ready"] is False
assert data["summary_persistence_run_readiness_ready"] is False
@@ -14493,7 +14827,7 @@ def test_candidate_queue_review_ai_summary_persistence_run_receipt_route_is_post
"candidate_queue_review_ai_summary_persistence_run_receipt_preview"
)
assert data["phase"] == (
- "phase_109_candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_readiness"
+ "phase_110_candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_receipt"
)
assert data["run_receipt_passed"] is False
assert data["summary_persistence_run_receipt_passed"] is False
@@ -14570,7 +14904,7 @@ def test_candidate_queue_review_ai_summary_persistence_run_closeout_route_is_pos
"candidate_queue_review_ai_summary_persistence_run_closeout_preview"
)
assert data["phase"] == (
- "phase_109_candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_readiness"
+ "phase_110_candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_receipt"
)
assert data["closeout_passed"] is False
assert data["summary_persistence_closeout_passed"] is False
@@ -14647,7 +14981,7 @@ def test_candidate_queue_review_ai_summary_persistence_telegram_dispatch_gate_ro
"candidate_queue_review_ai_summary_persistence_telegram_dispatch_gate_preview"
)
assert data["phase"] == (
- "phase_109_candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_readiness"
+ "phase_110_candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_receipt"
)
assert data["telegram_dispatch_gate_passed"] is False
assert data["summary_persistence_telegram_dispatch_gate_passed"] is False
@@ -14721,7 +15055,7 @@ def test_candidate_queue_review_ai_summary_persistence_telegram_dispatch_run_pac
"candidate_queue_review_ai_summary_persistence_telegram_dispatch_run_package_preview"
)
assert data["phase"] == (
- "phase_109_candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_readiness"
+ "phase_110_candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_receipt"
)
assert data["telegram_dispatch_run_package_ready"] is False
assert data["summary_persistence_telegram_dispatch_run_package_ready"] is False
@@ -14800,7 +15134,7 @@ def test_candidate_queue_review_ai_summary_persistence_telegram_dispatch_run_rea
"candidate_queue_review_ai_summary_persistence_telegram_dispatch_run_readiness_preview"
)
assert data["phase"] == (
- "phase_109_candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_readiness"
+ "phase_110_candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_receipt"
)
assert data["telegram_dispatch_run_readiness_ready"] is False
assert (
@@ -14887,7 +15221,7 @@ def test_candidate_queue_review_ai_summary_persistence_telegram_dispatch_run_rec
"candidate_queue_review_ai_summary_persistence_telegram_dispatch_run_receipt_preview"
)
assert data["phase"] == (
- "phase_109_candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_readiness"
+ "phase_110_candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_receipt"
)
assert data["telegram_dispatch_run_receipt_passed"] is False
assert data["summary_persistence_telegram_dispatch_run_receipt_passed"] is False
@@ -14968,7 +15302,7 @@ def test_candidate_queue_review_ai_summary_persistence_telegram_dispatch_closeou
"candidate_queue_review_ai_summary_persistence_telegram_dispatch_closeout_preview"
)
assert data["phase"] == (
- "phase_109_candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_readiness"
+ "phase_110_candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_receipt"
)
assert data["telegram_dispatch_closeout_passed"] is False
assert data["summary_persistence_telegram_dispatch_closeout_passed"] is False
@@ -15050,7 +15384,7 @@ def test_candidate_queue_review_ai_summary_persistence_telegram_dispatch_archive
"candidate_queue_review_ai_summary_persistence_telegram_dispatch_archive_preview"
)
assert data["phase"] == (
- "phase_109_candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_readiness"
+ "phase_110_candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_receipt"
)
assert data["telegram_dispatch_archive_ready"] is False
assert data["summary_persistence_telegram_dispatch_archive_ready"] is False
@@ -15133,7 +15467,7 @@ def test_candidate_queue_writer_run_receipt_route_is_post_only_and_no_write():
assert response.status_code == 200
assert data["mode"] == "candidate_queue_writer_run_receipt_preview"
- assert data["phase"] == "phase_109_candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_readiness"
+ assert data["phase"] == "phase_110_candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_receipt"
assert data["receipt_passed"] is True
assert data["ready_for_next_manual_review"] is True
assert data["ready_for_api_database_write"] is False
@@ -15158,7 +15492,7 @@ def test_scheduler_plan_preview_blocks_job_attachment():
plan = MarketIntelService().build_scheduler_plan()
assert plan["mode"] == "scheduler_attach_plan_preview"
- assert plan["phase"] == "phase_109_candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_readiness"
+ assert plan["phase"] == "phase_110_candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_receipt"
assert plan["ready_to_attach_scheduler"] is False
assert plan["scheduler_attached"] is False
assert plan["scheduler_registration_executed"] is False
@@ -15196,7 +15530,7 @@ def test_scheduler_plan_route_is_preview_only():
assert response.status_code == 200
assert data["mode"] == "scheduler_attach_plan_preview"
- assert data["phase"] == "phase_109_candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_readiness"
+ assert data["phase"] == "phase_110_candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_receipt"
assert data["scheduler_registration_executed"] is False
assert data["crawler_job_started"] is False
assert data["external_network_executed"] is False
@@ -15207,7 +15541,7 @@ def test_match_review_plan_preview_blocks_auto_confirm():
plan = MarketIntelService().build_match_review_plan()
assert plan["mode"] == "match_review_plan_preview"
- assert plan["phase"] == "phase_109_candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_readiness"
+ assert plan["phase"] == "phase_110_candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_receipt"
assert plan["ready_for_review_queue"] is False
assert plan["review_queue_created"] is False
assert plan["auto_match_executed"] is False
@@ -15243,7 +15577,7 @@ def test_match_review_plan_route_is_preview_only():
assert response.status_code == 200
assert data["mode"] == "match_review_plan_preview"
- assert data["phase"] == "phase_109_candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_readiness"
+ assert data["phase"] == "phase_110_candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_receipt"
assert data["review_queue_created"] is False
assert data["auto_confirm_executed"] is False
assert data["external_network_executed"] is False
@@ -15254,7 +15588,7 @@ def test_opportunity_plan_preview_blocks_alerts_and_ai_summary():
plan = MarketIntelService().build_opportunity_plan()
assert plan["mode"] == "opportunity_plan_preview"
- assert plan["phase"] == "phase_109_candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_readiness"
+ assert plan["phase"] == "phase_110_candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_receipt"
assert plan["ready_for_opportunity_queue"] is False
assert plan["opportunity_queue_created"] is False
assert plan["threat_alert_dispatched"] is False
@@ -15295,7 +15629,7 @@ def test_opportunity_plan_route_is_preview_only():
assert response.status_code == 200
assert data["mode"] == "opportunity_plan_preview"
- assert data["phase"] == "phase_109_candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_readiness"
+ assert data["phase"] == "phase_110_candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_receipt"
assert data["opportunity_queue_created"] is False
assert data["threat_alert_dispatched"] is False
assert data["ai_summary_generated"] is False
@@ -15306,7 +15640,7 @@ def test_opportunity_scoring_plan_preview_blocks_scoring_and_alerts():
plan = MarketIntelService().build_opportunity_scoring_plan()
assert plan["mode"] == "opportunity_scoring_plan_preview"
- assert plan["phase"] == "phase_109_candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_readiness"
+ assert plan["phase"] == "phase_110_candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_receipt"
assert plan["ready_for_scoring_job"] is False
assert plan["scoring_job_created"] is False
assert plan["score_calculation_executed"] is False
@@ -15354,7 +15688,7 @@ def test_opportunity_scoring_plan_route_is_preview_only():
assert response.status_code == 200
assert data["mode"] == "opportunity_scoring_plan_preview"
- assert data["phase"] == "phase_109_candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_readiness"
+ assert data["phase"] == "phase_110_candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_receipt"
assert data["scoring_job_created"] is False
assert data["score_calculation_executed"] is False
assert data["sample_scores_generated"] is False
@@ -15366,7 +15700,7 @@ def test_opportunity_evidence_plan_preview_blocks_queries_and_alerts():
plan = MarketIntelService().build_opportunity_evidence_plan()
assert plan["mode"] == "opportunity_evidence_plan_preview"
- assert plan["phase"] == "phase_109_candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_readiness"
+ assert plan["phase"] == "phase_110_candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_receipt"
assert plan["ready_for_evidence_bundle"] is False
assert plan["evidence_bundle_created"] is False
assert plan["evidence_query_executed"] is False
@@ -15412,7 +15746,7 @@ def test_opportunity_evidence_plan_route_is_preview_only():
assert response.status_code == 200
assert data["mode"] == "opportunity_evidence_plan_preview"
- assert data["phase"] == "phase_109_candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_readiness"
+ assert data["phase"] == "phase_110_candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_receipt"
assert data["evidence_bundle_created"] is False
assert data["evidence_query_executed"] is False
assert data["sample_evidence_generated"] is False
@@ -15425,7 +15759,7 @@ def test_opportunity_alert_plan_preview_blocks_dispatch_and_llm_calls():
plan = MarketIntelService().build_opportunity_alert_plan()
assert plan["mode"] == "opportunity_alert_plan_preview"
- assert plan["phase"] == "phase_109_candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_readiness"
+ assert plan["phase"] == "phase_110_candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_receipt"
assert plan["ready_for_alert_candidates"] is False
assert plan["alert_candidate_created"] is False
assert plan["alert_queue_created"] is False
@@ -15510,7 +15844,7 @@ def test_opportunity_alert_plan_route_is_preview_only():
assert response.status_code == 200
assert data["mode"] == "opportunity_alert_plan_preview"
- assert data["phase"] == "phase_109_candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_readiness"
+ assert data["phase"] == "phase_110_candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_receipt"
assert data["alert_candidate_created"] is False
assert data["alert_queue_created"] is False
assert data["review_queue_created"] is False
@@ -15588,7 +15922,7 @@ def test_mcp_deploy_preflight_route_is_preview_only():
assert response.status_code == 200
assert data["mode"] == "mcp_external_deploy_preflight_preview"
- assert data["phase"] == "phase_109_candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_readiness"
+ assert data["phase"] == "phase_110_candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_receipt"
assert data["deployment_actions_executed"] is False
assert data["docker_command_executed"] is False
assert data["ssh_command_executed"] is False
@@ -15603,7 +15937,7 @@ def test_mcp_readiness_default_is_planned_only(monkeypatch):
readiness = MarketIntelService().build_mcp_readiness()
assert readiness["mode"] == "mcp_readiness_planned"
- assert readiness["phase"] == "phase_109_candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_readiness"
+ assert readiness["phase"] == "phase_110_candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_receipt"
assert readiness["execute_requested"] is False
assert readiness["router_enabled"] is False
assert readiness["external_mcp_complete"] is False
@@ -16289,6 +16623,12 @@ def test_deployment_readiness_reports_app_only_release_gate():
]
is True
)
+ assert (
+ readiness["checks"][
+ "candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_receipt_preview_safe"
+ ]
+ is True
+ )
assert (
readiness["checks"][
"candidate_queue_review_decision_writer_cli_status_safe"
@@ -16583,6 +16923,11 @@ def test_deployment_readiness_reports_app_only_release_gate():
"candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_readiness"
in readiness["production_smoke_targets"]
)
+ assert (
+ "/api/market_intel/manual_sample_review/"
+ "candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_receipt"
+ in readiness["production_smoke_targets"]
+ )
assert (
"/api/market_intel/manual_sample_review/"
"candidate_queue_review_decision_writer_status"
@@ -19836,6 +20181,54 @@ def test_deployment_readiness_reports_app_only_release_gate():
]["scheduler_attached"]
is False
)
+ assert (
+ readiness[
+ "candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_receipt"
+ ]["mode"]
+ == "candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_receipt_preview"
+ )
+ assert (
+ readiness[
+ "candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_receipt"
+ ]["telegram_dispatch_report_catalog_record_run_receipt_passed"]
+ is False
+ )
+ assert (
+ readiness[
+ "candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_receipt"
+ ]["ready_for_market_intel_report_catalog_record_commit"]
+ is False
+ )
+ assert (
+ readiness[
+ "candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_receipt"
+ ]["catalog_record_run_receipt_gate_file_written"]
+ is False
+ )
+ assert (
+ readiness[
+ "candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_receipt"
+ ]["catalog_record_run_receipt_executed"]
+ is False
+ )
+ assert (
+ readiness[
+ "candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_receipt"
+ ]["catalog_record_commit_executed"]
+ is False
+ )
+ assert (
+ readiness[
+ "candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_receipt"
+ ]["database_write_executed"]
+ is False
+ )
+ assert (
+ readiness[
+ "candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_receipt"
+ ]["scheduler_attached"]
+ is False
+ )
assert (
readiness["candidate_queue_review_decision_writer_status"]["mode"]
== "candidate_queue_review_decision_writer_cli_blocked"
@@ -19986,7 +20379,7 @@ def test_migration_apply_drill_planned_is_safe_and_manual_only():
drill = MarketIntelService().build_migration_apply_drill()
assert drill["mode"] == "migration_apply_drill_preview"
- assert drill["phase"] == "phase_109_candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_readiness"
+ assert drill["phase"] == "phase_110_candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_receipt"
assert drill["execute_requested"] is False
assert drill["schema_state"] == "planned_no_db_probe"
assert drill["drill_ready_for_operator_review"] is True
@@ -20101,7 +20494,7 @@ def test_migration_apply_drill_route_is_preview_only():
assert response.status_code == 200
assert data["mode"] == "migration_apply_drill_preview"
- assert data["phase"] == "phase_109_candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_readiness"
+ assert data["phase"] == "phase_110_candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_receipt"
assert data["execute_requested"] is False
assert data["migration_executed"] is False
assert data["rollback_executed"] is False
@@ -20113,7 +20506,7 @@ def test_migration_catalog_review_planned_is_safe_and_diagnostic():
review = MarketIntelService().build_migration_catalog_review()
assert review["mode"] == "migration_catalog_review_preview"
- assert review["phase"] == "phase_109_candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_readiness"
+ assert review["phase"] == "phase_110_candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_receipt"
assert review["execute_requested"] is False
assert review["catalog_state"] == "planned_no_probe"
assert review["seed_state"] == "planned_no_probe"
@@ -20228,7 +20621,7 @@ def test_migration_catalog_review_route_is_preview_only():
assert response.status_code == 200
assert data["mode"] == "migration_catalog_review_preview"
- assert data["phase"] == "phase_109_candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_readiness"
+ assert data["phase"] == "phase_110_candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_receipt"
assert data["execute_requested"] is False
assert data["catalog_state"] == "planned_no_probe"
assert data["migration_executed"] is False
@@ -20241,7 +20634,7 @@ def test_migration_live_smoke_planned_is_preview_only():
smoke = MarketIntelService().build_migration_live_smoke()
assert smoke["mode"] == "migration_live_smoke_preview"
- assert smoke["phase"] == "phase_109_candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_readiness"
+ assert smoke["phase"] == "phase_110_candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_receipt"
assert smoke["execute_requested"] is False
assert smoke["smoke_result"] == "planned_no_execution"
assert smoke["live_smoke_passed"] is False
@@ -20303,7 +20696,7 @@ def test_migration_live_smoke_route_is_preview_only():
assert response.status_code == 200
assert data["mode"] == "migration_live_smoke_preview"
- assert data["phase"] == "phase_109_candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_readiness"
+ assert data["phase"] == "phase_110_candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_receipt"
assert data["execute_requested"] is False
assert data["smoke_result"] == "planned_no_execution"
assert data["migration_executed"] is False
@@ -20316,7 +20709,7 @@ def test_live_db_inventory_planned_is_preview_only():
inventory = MarketIntelService().build_live_db_inventory()
assert inventory["mode"] == "live_db_inventory_planned"
- assert inventory["phase"] == "phase_109_candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_readiness"
+ assert inventory["phase"] == "phase_110_candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_receipt"
assert inventory["execute_requested"] is False
assert inventory["read_only_query_executed"] is False
assert inventory["database_connection_opened"] is False
@@ -20460,7 +20853,7 @@ def test_live_db_inventory_route_is_preview_only():
assert response.status_code == 200
assert data["mode"] == "live_db_inventory_planned"
- assert data["phase"] == "phase_109_candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_readiness"
+ assert data["phase"] == "phase_110_candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_receipt"
assert data["execute_requested"] is False
assert data["read_only_query_executed"] is False
assert data["database_write_executed"] is False
@@ -20696,7 +21089,7 @@ def test_candidate_queue_writer_cli_script_outputs_blocked_gate(tmp_path):
assert result.returncode == 0
assert data["mode"] == "candidate_queue_writer_cli_blocked"
- assert data["phase"] == "phase_109_candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_readiness"
+ assert data["phase"] == "phase_110_candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_receipt"
assert data["execute_requested"] is False
assert data["apply_real_write_requested"] is False
assert data["writes_executed"] is False
@@ -20725,7 +21118,7 @@ def test_review_decision_writer_cli_script_outputs_blocked_gate_without_login_en
assert result.returncode == 0
assert data["mode"] == "candidate_queue_review_decision_writer_cli_blocked"
- assert data["phase"] == "phase_109_candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_readiness"
+ assert data["phase"] == "phase_110_candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_receipt"
assert data["execute_requested"] is False
assert data["apply_real_write_requested"] is False
assert data["approval_token_present"] is False