From f4ec50cd30611141776f07d7779148382a72d6c8 Mon Sep 17 00:00:00 2001 From: OoO Date: Thu, 21 May 2026 11:31:28 +0800 Subject: [PATCH] [V10.358] add market intel MCP activation evidence review --- TODO_NEXT_STEPS.txt | 1 + config.py | 2 +- docs/memory/history_logs.md | 4 + routes/market_intel_routes.py | 16 + services/market_intel/deployment_readiness.py | 16 +- .../market_intel/mcp_activation_evidence.py | 269 ++++++++++++++ services/market_intel/phase.py | 2 +- templates/market_intel/disabled.html | 152 +++++++- tests/test_market_intel_skeleton.py | 330 ++++++++++++------ 9 files changed, 679 insertions(+), 113 deletions(-) create mode 100644 services/market_intel/mcp_activation_evidence.py diff --git a/TODO_NEXT_STEPS.txt b/TODO_NEXT_STEPS.txt index 5a44349..e893358 100644 --- a/TODO_NEXT_STEPS.txt +++ b/TODO_NEXT_STEPS.txt @@ -4,6 +4,7 @@ ================================================================================ 【已完成】 + - V10.358 補市場情報 MCP 啟用證據審核:新增 `mcp_activation_evidence` read-only builder、GET/POST endpoint、UI redacted evidence 審核面板與 deployment readiness smoke target,讓操作員貼上 env/health/router/telemetry/fallback 證據後判斷能否補齊 external/internal MCP runtime 缺口;API/UI 不保存 payload、不打 health、不啟動 MCP、不執行 docker/SSH、不開 DB、不抓外站、不掛 scheduler,且會阻擋真實 secret 字串與任何 DB write/fetch/scheduler 證據。 - V10.357 補市場情報 MCP 完整度稽核:新增 `mcp_completion_audit` read-only builder、GET endpoint、UI 面板與 deployment readiness smoke target,彙整外部 MCP design/runtime、內部 tool contract/runtime、activation runbook 與 fetch gate 狀態;API/UI 不啟動 MCP、不打 health、不執行 docker/SSH、不開 DB、不寫檔、不抓外站、不掛 scheduler。 - V10.356 補市場情報 candidate queue review AI summary Telegram dispatch report catalog record final closeout gate:新增 read-only report catalog record final closeout builder、POST endpoint、UI 按鈕與 deployment readiness smoke target,在 archive summary gate 後覆核 catalog record identity、artifact traceability、sections、DB commit/post-write smoke、pipeline complete 與無後續 follow-up;API/UI 不讀 approval/Telegram token、不呼叫 LLM、不產報表、不派送 Telegram、不開 DB、不寫檔、不執行 CLI、不寫 catalog record、不 commit、不更新 review_state、不掛 scheduler。 - V10.355 補市場情報 candidate queue review AI summary Telegram dispatch report catalog record archive summary gate:新增 read-only report catalog record archive summary builder、POST endpoint、UI 按鈕與 deployment readiness smoke target,在 archive gate 後整理 catalog record identity、artifact traceability、DB commit/post-write smoke、archive manifest/retention policy 與後續 final closeout separate gate;API/UI 不讀 approval/Telegram token、不呼叫 LLM、不產報表、不派送 Telegram、不開 DB、不寫檔、不執行 CLI、不寫 catalog record、不 commit、不更新 review_state、不掛 scheduler。 diff --git a/config.py b/config.py index a8bff74..f4f9336 100644 --- a/config.py +++ b/config.py @@ -323,7 +323,7 @@ YOUTUBE_API_KEY = os.getenv('YOUTUBE_API_KEY', '') # ========================================== # 系統版本與路徑 # ========================================== -SYSTEM_VERSION = "V10.357" +SYSTEM_VERSION = "V10.358" 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 0c5b90c..f4ee879 100644 --- a/docs/memory/history_logs.md +++ b/docs/memory/history_logs.md @@ -12,6 +12,10 @@ ## 📅 詳細更新日誌 (考古存檔) +### 2026-05-21:市場情報 MCP 啟用證據審核 +- **V10.358 MCP activation evidence review**: 新增 `mcp_activation_evidence` read-only builder、GET/POST endpoint、UI redacted evidence 審核面板與 deployment readiness smoke target,讓操作員貼上 env/health/router/telemetry/fallback 證據後判斷能否補齊 external/internal MCP runtime 缺口。 +- **只讀安全邊界**: 本階段不保存 payload、不打 health、不啟動 MCP、不執行 docker/SSH、不開 DB、不抓外站、不掛 scheduler;payload 只允許 redacted/boolean,真實 secret 字串與任何 DB write/fetch/scheduler 證據會被阻擋。 + ### 2026-05-21:市場情報 MCP 完整度稽核 - **V10.357 MCP completion audit**: 新增 `mcp_completion_audit` read-only builder、GET endpoint、UI 面板與 deployment readiness smoke target,彙整外部 MCP design/runtime、內部 tool contract/runtime、activation runbook 與 fetch gate 狀態。 - **只讀安全邊界**: 本階段只做完整度稽核,不啟動 MCP、不打 health、不執行 docker/SSH、不開 DB、不寫檔、不抓外站、不掛 scheduler;外部 MCP runtime complete 仍需 operator 依 runbook 啟用與 health 驗證。 diff --git a/routes/market_intel_routes.py b/routes/market_intel_routes.py index 547c26a..5088859 100644 --- a/routes/market_intel_routes.py +++ b/routes/market_intel_routes.py @@ -18,6 +18,7 @@ from services.market_intel.candidate_queue_writer_run_readiness import build_can from services.market_intel.candidate_queue_writer_run_receipt import build_candidate_queue_writer_run_receipt from services.market_intel.candidate_queue_writer_run_closeout import build_candidate_queue_writer_run_closeout from services.market_intel.candidate_queue_review_handoff import build_candidate_queue_review_handoff +from services.market_intel.mcp_activation_evidence import build_mcp_activation_evidence_preview TAIPEI_TZ = timezone(timedelta(hours=8)) @@ -166,6 +167,21 @@ def market_intel_mcp_completion_audit(): return jsonify(_service().build_mcp_completion_audit()) +@market_intel_bp.route("/api/market_intel/mcp_activation_evidence", methods=["GET", "POST"]) +@login_required +def market_intel_mcp_activation_evidence(): + evidence = {} + if request.method == "POST": + payload = request.get_json(silent=True) or {} + evidence = payload.get("evidence", payload) + return jsonify( + build_mcp_activation_evidence_preview( + evidence=evidence, + phase=_service().phase, + ) + ) + + @market_intel_bp.route("/api/market_intel/scheduler_plan") @login_required def market_intel_scheduler_plan(): diff --git a/services/market_intel/deployment_readiness.py b/services/market_intel/deployment_readiness.py index 3bd36ba..0c5077e 100644 --- a/services/market_intel/deployment_readiness.py +++ b/services/market_intel/deployment_readiness.py @@ -59,6 +59,7 @@ from services.market_intel.candidate_queue_review_ai_summary_persistence_telegra 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 +from services.market_intel.mcp_activation_evidence import build_mcp_activation_evidence_preview BLOCKED_RUN_REVIEW_KEYS = ( "ready_for_api_database_write", "ready_for_scheduler_attach", @@ -111,7 +112,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/mcp_completion_audit", "/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_ai_summary_persistence_telegram_dispatch_report_catalog_record_commit", "/api/market_intel/manual_sample_review/candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_closeout", "/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/mcp_completion_audit", "/api/market_intel/mcp_activation_evidence", "/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_ai_summary_persistence_telegram_dispatch_report_catalog_record_commit", "/api/market_intel/manual_sample_review/candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_closeout", "/api/market_intel/manual_sample_review/candidate_queue_review_decision_writer_status") PRODUCTION_SMOKE_TARGETS = ( PRODUCTION_SMOKE_TARGETS[:-1] + ( @@ -147,6 +148,9 @@ def build_deployment_readiness_preview(*, service, market_intel_tables, schema_s mcp_activation_runbook = service.build_mcp_activation_runbook() mcp_fetch_gate = service.build_mcp_fetch_gate() mcp_completion_audit = service.build_mcp_completion_audit() + mcp_activation_evidence = build_mcp_activation_evidence_preview( + phase=service.phase, + ) scheduler_plan = service.build_scheduler_plan() manual_sample_plan = service.build_manual_sample_plan() manual_sample_acceptance = service.build_manual_sample_acceptance() @@ -371,6 +375,15 @@ def build_deployment_readiness_preview(*, service, market_intel_tables, schema_s and not mcp_completion_audit["api_writes_database"] and not mcp_completion_audit["api_uses_external_network"] ), + "mcp_activation_evidence_preview_safe": bool( + mcp_activation_evidence["mode"] == "mcp_activation_evidence_preview" + and not mcp_activation_evidence["payload_persisted"] + and not mcp_activation_evidence["evidence_persisted"] + and not mcp_activation_evidence["api_executes_health_check"] + and not mcp_activation_evidence["api_opens_database_connection"] + and not mcp_activation_evidence["api_writes_database"] + and not mcp_activation_evidence["api_uses_external_network"] + ), "scheduler_plan_preview_safe": bool( scheduler_plan["mode"] == "scheduler_attach_plan_preview" and not scheduler_plan["scheduler_registration_executed"] @@ -802,6 +815,7 @@ def build_deployment_readiness_preview(*, service, market_intel_tables, schema_s "mcp_activation_runbook": mcp_activation_runbook, "mcp_fetch_gate": mcp_fetch_gate, "mcp_completion_audit": mcp_completion_audit, + "mcp_activation_evidence": mcp_activation_evidence, "scheduler_plan": scheduler_plan, "manual_sample_plan": manual_sample_plan, "manual_sample_acceptance": manual_sample_acceptance, diff --git a/services/market_intel/mcp_activation_evidence.py b/services/market_intel/mcp_activation_evidence.py new file mode 100644 index 0000000..29e950d --- /dev/null +++ b/services/market_intel/mcp_activation_evidence.py @@ -0,0 +1,269 @@ +"""市場情報 MCP 啟用證據審核 preview。 + +操作員可貼上外部 MCP 啟用後的 redacted evidence;本模組只做結構化審核, +不打 health endpoint、不開 DB、不寫入、不啟動 fetch。 +""" + + +REQUIRED_ENV_VARS = ("MCP_POSTGRES_PASSWORD", "TAVILY_API_KEY", "EXA_API_KEY") +EXPECTED_HEALTH_TARGETS = ( + "http://localhost:3001/health", + "http://localhost:3002/health", + "http://localhost:3003/health", + "http://localhost:3004/health", +) +SAFE_LITERAL_VALUES = {"redacted", "***", "set", "present", "true", "yes", "ok"} +PROHIBITED_EVIDENCE_FLAGS = ( + "database_write_executed", + "database_commit_executed", + "scheduler_attached", + "crawler_job_started", + "external_fetch_executed", + "manual_fetch_executed", +) + + +def _truthy(value): + if isinstance(value, bool): + return value + if isinstance(value, (int, float)): + return bool(value) + if isinstance(value, str): + return value.strip().lower() in {"1", "true", "yes", "y", "ok", "pass", "passed", "healthy", "set", "present", "redacted", "***"} + return False + + +def _looks_like_secret_literal(value): + if not isinstance(value, str): + return False + normalized = value.strip().lower() + if not normalized or normalized in SAFE_LITERAL_VALUES: + return False + return len(value.strip()) > 2 + + +def _evidence_map(evidence, key): + value = evidence.get(key) + return value if isinstance(value, dict) else {} + + +def _ack(evidence, key): + acknowledgements = _evidence_map(evidence, "operator_acknowledgements") + return _truthy(evidence.get(key)) or _truthy(acknowledgements.get(key)) + + +def _env_statuses(evidence): + env = _evidence_map(evidence, "required_env_vars") or _evidence_map(evidence, "env") + statuses = [] + for name in REQUIRED_ENV_VARS: + value = env.get(name) + statuses.append( + { + "name": name, + "present": _truthy(value), + "secret_literal_in_payload": _looks_like_secret_literal(value), + "value_redacted": bool(value) and not _looks_like_secret_literal(value), + } + ) + return statuses + + +def _health_item_ok(item): + if isinstance(item, bool): + return item + if isinstance(item, (int, float)): + return int(item) == 200 + if isinstance(item, str): + return item.strip().lower() in {"200", "ok", "pass", "passed", "healthy"} + if isinstance(item, dict): + return ( + int(item.get("status_code") or item.get("code") or 0) == 200 + or _truthy(item.get("healthy")) + or _truthy(item.get("passed")) + or _truthy(item.get("status")) + ) + return False + + +def _health_item_for_target(health, target): + if isinstance(health, dict): + if target in health: + return health[target] + port = target.rsplit(":", 1)[-1].split("/", 1)[0] + return health.get(port) or health.get(f"localhost:{port}") + if isinstance(health, list): + for item in health: + if not isinstance(item, dict): + continue + value = item.get("url") or item.get("target") or item.get("endpoint") + port = str(item.get("port") or "") + if value == target or (port and port in target): + return item + return None + + +def _health_statuses(evidence): + health = evidence.get("health_checks") or evidence.get("health_targets") or {} + return [ + { + "target": target, + "passed": _health_item_ok(_health_item_for_target(health, target)), + } + for target in EXPECTED_HEALTH_TARGETS + ] + + +def _sample_evidence_template(): + return { + "required_env_vars": { + "MCP_POSTGRES_PASSWORD": "redacted", + "TAVILY_API_KEY": "redacted", + "EXA_API_KEY": "redacted", + }, + "health_checks": { + "http://localhost:3001/health": 200, + "http://localhost:3002/health": 200, + "http://localhost:3003/health": 200, + "http://localhost:3004/health": 200, + }, + "mcp_router_enabled": True, + "market_intel_readonly_smoke_passed": True, + "telemetry": { + "mcp_calls_table_exists": True, + "read_only_query_executed": True, + }, + "operator_acknowledgements": { + "no_remove_orphans": True, + "momo_db_preserved": True, + "router_enabled_after_health": True, + "rollback_plan_confirmed": True, + "fetch_gate_left_closed": True, + }, + } + + +def build_mcp_activation_evidence_preview(*, evidence=None, phase=None): + """審核 redacted MCP runtime evidence;不執行任何外部動作。""" + evidence = evidence or {} + payload_received = bool(evidence) + env_statuses = _env_statuses(evidence) + health_statuses = _health_statuses(evidence) + telemetry = _evidence_map(evidence, "telemetry") + prohibited_flags = [ + key for key in PROHIBITED_EVIDENCE_FLAGS + if _truthy(evidence.get(key)) + ] + secret_literal_keys = [ + item["name"] for item in env_statuses + if item["secret_literal_in_payload"] + ] + gates = [ + { + "key": "evidence_payload_received", + "passed": payload_received, + "label": "已提供 redacted MCP runtime evidence payload", + }, + { + "key": "no_secret_literals_in_payload", + "passed": not secret_literal_keys, + "label": "payload 只允許 redacted / boolean,不接受真實 secret 字串", + }, + { + "key": "required_env_vars_acknowledged", + "passed": all(item["present"] for item in env_statuses), + "label": "三個必要 env 已由操作員確認存在", + }, + { + "key": "all_health_targets_passed", + "passed": all(item["passed"] for item in health_statuses), + "label": "四個 localhost MCP health target 全部 200 / healthy", + }, + { + "key": "mcp_router_enabled_after_health", + "passed": _truthy(evidence.get("mcp_router_enabled")) + and _ack(evidence, "router_enabled_after_health"), + "label": "router 僅在 health 全過後啟用", + }, + { + "key": "market_intel_readonly_smoke_passed", + "passed": _truthy(evidence.get("market_intel_readonly_smoke_passed")), + "label": "market_intel MCP read-only smoke 已通過", + }, + { + "key": "telemetry_runtime_confirmed", + "passed": _truthy(telemetry.get("mcp_calls_table_exists")) + and _truthy(telemetry.get("read_only_query_executed")), + "label": "mcp_calls telemetry read-only 查詢鏈路已確認", + }, + { + "key": "operator_fallback_confirmed", + "passed": _ack(evidence, "rollback_plan_confirmed"), + "label": "router kill switch 與 stop MCP stack fallback 已確認", + }, + { + "key": "production_boundaries_acknowledged", + "passed": _ack(evidence, "no_remove_orphans") + and _ack(evidence, "momo_db_preserved"), + "label": "已確認不使用 remove-orphans 且不動 momo-db lifecycle", + }, + { + "key": "manual_fetch_gate_left_closed", + "passed": _ack(evidence, "fetch_gate_left_closed"), + "label": "啟用 MCP 後仍未打開人工 fetch gate", + }, + { + "key": "no_write_or_scheduler_evidence", + "passed": not prohibited_flags, + "label": "證據未宣告 DB write、scheduler 或 fetch 已執行", + }, + ] + blocked_reasons = [ + gate["key"] for gate in gates + if not gate["passed"] + ] + accepted = payload_received and not blocked_reasons + + payload = { + "mode": ( + "mcp_activation_evidence_review" + if payload_received + else "mcp_activation_evidence_preview" + ), + "phase": phase, + "evidence_payload_received": payload_received, + "activation_evidence_accepted": accepted, + "ready_for_runtime_promotion": accepted, + "ready_for_fetch_gate_review": accepted, + "external_mcp_runtime_evidence_complete": accepted, + "internal_mcp_runtime_evidence_complete": accepted, + "gate_count": len(gates), + "passed_gate_count": sum(1 for gate in gates if gate["passed"]), + "blocked_reasons": blocked_reasons, + "gates": gates, + "env_statuses": env_statuses, + "health_statuses": health_statuses, + "secret_literal_keys": secret_literal_keys, + "prohibited_flags": prohibited_flags, + "sample_evidence_template": _sample_evidence_template(), + "next_operator_steps": [ + "若本審核通過,再跑 /api/market_intel/mcp_readiness?execute=true&timeout=3 做只讀 runtime smoke", + "確認 completion audit 的 external/internal runtime 缺口被證據補齊", + "人工 fetch gate 仍需另行審核,不可因 MCP runtime ready 自動抓外站", + ], + "payload_persisted": False, + "evidence_persisted": False, + "api_executes_health_check": False, + "api_executes_docker": False, + "api_executes_ssh": False, + "api_opens_database_connection": False, + "api_writes_database": False, + "api_uses_external_network": False, + "database_session_created": False, + "database_write_executed": False, + "database_commit_executed": False, + "external_network_executed": False, + "scheduler_attached": False, + "writes_executed": False, + "would_write_database": False, + } + return payload diff --git a/services/market_intel/phase.py b/services/market_intel/phase.py index 1afd5e0..345408c 100644 --- a/services/market_intel/phase.py +++ b/services/market_intel/phase.py @@ -1,3 +1,3 @@ """市場情報 rollout phase 單一來源。""" -MARKET_INTEL_PHASE = "phase_116_market_intel_mcp_completion_audit" +MARKET_INTEL_PHASE = "phase_117_market_intel_mcp_activation_evidence" diff --git a/templates/market_intel/disabled.html b/templates/market_intel/disabled.html index e67cc86..cbe0c52 100644 --- a/templates/market_intel/disabled.html +++ b/templates/market_intel/disabled.html @@ -552,6 +552,32 @@ +
+
+
+

MCP / ACTIVATION EVIDENCE

+

MCP 啟用證據審核

+
+ +
+
+ loading +
+
+
讀取 MCP 啟用證據審核中...
+
+
+ +
+ +
+
+
+
@@ -1056,6 +1082,7 @@ const mcpActivationRoot = document.querySelector('[data-market-intel-mcp-activation]'); const mcpFetchGateRoot = document.querySelector('[data-market-intel-mcp-fetch-gate]'); const mcpCompletionRoot = document.querySelector('[data-market-intel-mcp-completion]'); + const mcpActivationEvidenceRoot = document.querySelector('[data-market-intel-mcp-activation-evidence]'); const manualSampleRoot = document.querySelector('[data-market-intel-manual-sample]'); const sampleAcceptanceRoot = document.querySelector('[data-market-intel-sample-acceptance]'); const sampleReviewRoot = document.querySelector('[data-market-intel-sample-review]'); @@ -1072,7 +1099,7 @@ const liveInventoryRoot = document.querySelector('[data-market-intel-live-inventory]'); const approvalRoot = document.querySelector('[data-market-intel-approval]'); const deployRoot = document.querySelector('[data-market-intel-deploy]'); - if (!root && !writerRoot && !cliRoot && !dbProbeRoot && !seedDiffRoot && !legacyBridgeRoot && !mcpReadinessRoot && !mcpPreflightRoot && !mcpActivationRoot && !mcpFetchGateRoot && !mcpCompletionRoot && !manualSampleRoot && !sampleAcceptanceRoot && !sampleReviewRoot && !schedulerRoot && !matchReviewRoot && !opportunityRoot && !opportunityScoringRoot && !opportunityEvidenceRoot && !opportunityAlertRoot && !migrationRoot && !migrationDrillRoot && !catalogReviewRoot && !liveSmokeRoot && !liveInventoryRoot && !approvalRoot && !deployRoot) return; + if (!root && !writerRoot && !cliRoot && !dbProbeRoot && !seedDiffRoot && !legacyBridgeRoot && !mcpReadinessRoot && !mcpPreflightRoot && !mcpActivationRoot && !mcpFetchGateRoot && !mcpCompletionRoot && !mcpActivationEvidenceRoot && !manualSampleRoot && !sampleAcceptanceRoot && !sampleReviewRoot && !schedulerRoot && !matchReviewRoot && !opportunityRoot && !opportunityScoringRoot && !opportunityEvidenceRoot && !opportunityAlertRoot && !migrationRoot && !migrationDrillRoot && !catalogReviewRoot && !liveSmokeRoot && !liveInventoryRoot && !approvalRoot && !deployRoot) return; const meta = root ? root.querySelector('[data-market-intel-preview-meta]') : null; const body = root ? root.querySelector('[data-market-intel-preview-body]') : null; @@ -1119,6 +1146,12 @@ const mcpCompletionBody = mcpCompletionRoot ? mcpCompletionRoot.querySelector('[data-market-intel-mcp-completion-body]') : null; const mcpCompletionRefresh = mcpCompletionRoot ? mcpCompletionRoot.querySelector('[data-market-intel-mcp-completion-refresh]') : null; const mcpCompletionEndpoint = "{{ url_for('market_intel.market_intel_mcp_completion_audit') }}"; + const mcpActivationEvidenceMeta = mcpActivationEvidenceRoot ? mcpActivationEvidenceRoot.querySelector('[data-market-intel-mcp-activation-evidence-meta]') : null; + const mcpActivationEvidenceBody = mcpActivationEvidenceRoot ? mcpActivationEvidenceRoot.querySelector('[data-market-intel-mcp-activation-evidence-body]') : null; + const mcpActivationEvidenceInput = mcpActivationEvidenceRoot ? mcpActivationEvidenceRoot.querySelector('[data-market-intel-mcp-activation-evidence-input]') : null; + const mcpActivationEvidenceReview = mcpActivationEvidenceRoot ? mcpActivationEvidenceRoot.querySelector('[data-market-intel-mcp-activation-evidence-review]') : null; + const mcpActivationEvidenceRefresh = mcpActivationEvidenceRoot ? mcpActivationEvidenceRoot.querySelector('[data-market-intel-mcp-activation-evidence-refresh]') : null; + const mcpActivationEvidenceEndpoint = "{{ url_for('market_intel.market_intel_mcp_activation_evidence') }}"; const manualSampleMeta = manualSampleRoot ? manualSampleRoot.querySelector('[data-market-intel-manual-sample-meta]') : null; const manualSampleBody = manualSampleRoot ? manualSampleRoot.querySelector('[data-market-intel-manual-sample-body]') : null; const manualSampleRefresh = manualSampleRoot ? manualSampleRoot.querySelector('[data-market-intel-manual-sample-refresh]') : null; @@ -2106,6 +2139,116 @@ } }; + const renderMcpActivationEvidenceMeta = data => { + mcpActivationEvidenceMeta.innerHTML = [ + `mode=${data.mode || 'unknown'}`, + `accepted=${data.activation_evidence_accepted ? 'yes' : 'no'}`, + `gates=${data.passed_gate_count || 0}/${data.gate_count || 0}`, + `health=${(data.health_statuses || []).filter(item => item.passed).length}/4`, + `persisted=${data.evidence_persisted ? 'yes' : 'no'}` + ].map(item => `${escapeHtml(item)}`).join(''); + }; + + const renderMcpActivationEvidenceBody = data => { + const blockers = (data.blocked_reasons || []).join(' / '); + const gates = data.gates || []; + const envs = data.env_statuses || []; + const health = data.health_statuses || []; + const steps = data.next_operator_steps || []; + const renderCheck = (key, label, status) => ` +
+
+ ${escapeHtml(key)} + ${escapeHtml(label || '')} +
+ ${escapeHtml(status)} +
+ `; + mcpActivationEvidenceBody.innerHTML = ` +
此審核只檢查操作員貼上的 redacted evidence;不打 health、不開 DB、不保存 payload、不啟動 fetch。${blockers ? `阻擋:${escapeHtml(blockers)}` : ''}
+
+
+

EVIDENCE GATES

+
${ + gates.length + ? gates.map(item => renderCheck(item.key, item.label, item.passed ? 'PASS' : 'BLOCK')).join('') + : '
尚未提供 evidence gates。
' + }
+
+
+

ENV / HEALTH

+
+ ${envs.map(item => renderCheck( + item.name, + item.secret_literal_in_payload ? 'secret literal blocked' : 'redacted or boolean only', + item.present && !item.secret_literal_in_payload ? 'OK' : 'BLOCK' + )).join('')} + ${health.map(item => renderCheck( + item.target, + 'localhost health evidence', + item.passed ? '200' : 'BLOCK' + )).join('')} +
+
+
+

NEXT

+
${ + steps.map((item, index) => renderCheck(`step_${index + 1}`, item, 'MANUAL')).join('') + }
+
+
+ `; + if (mcpActivationEvidenceInput && !mcpActivationEvidenceInput.value.trim() && data.sample_evidence_template) { + mcpActivationEvidenceInput.value = JSON.stringify(data.sample_evidence_template, null, 2); + } + }; + + const loadMcpActivationEvidence = async () => { + if (!mcpActivationEvidenceMeta || !mcpActivationEvidenceBody) return; + mcpActivationEvidenceBody.innerHTML = '
讀取 MCP 啟用證據審核中...
'; + try { + const response = await fetch(mcpActivationEvidenceEndpoint, { credentials: 'same-origin' }); + if (!response.ok) throw new Error(`HTTP ${response.status}`); + const data = await response.json(); + renderMcpActivationEvidenceMeta(data); + renderMcpActivationEvidenceBody(data); + } catch (error) { + mcpActivationEvidenceMeta.innerHTML = 'error'; + mcpActivationEvidenceBody.innerHTML = `
MCP 啟用證據審核讀取失敗:${escapeHtml(error.message)}
`; + } + }; + + const reviewMcpActivationEvidence = async () => { + if (!mcpActivationEvidenceMeta || !mcpActivationEvidenceBody || !mcpActivationEvidenceInput) return; + let parsed; + try { + parsed = JSON.parse(mcpActivationEvidenceInput.value || '{}'); + } catch (error) { + mcpActivationEvidenceMeta.innerHTML = 'json_error'; + mcpActivationEvidenceBody.innerHTML = `
JSON 格式錯誤:${escapeHtml(error.message)}
`; + return; + } + mcpActivationEvidenceBody.innerHTML = '
審核 MCP 啟用證據中...
'; + try { + const response = await fetch(mcpActivationEvidenceEndpoint, { + method: 'POST', + credentials: 'same-origin', + headers: { + 'Content-Type': 'application/json', + 'X-CSRFToken': csrfToken + }, + body: JSON.stringify({ evidence: parsed }) + }); + const data = await response.json(); + if (!response.ok && !data.mode) throw new Error(`HTTP ${response.status}`); + renderMcpActivationEvidenceMeta(data); + renderMcpActivationEvidenceBody(data); + } catch (error) { + mcpActivationEvidenceMeta.innerHTML = 'error'; + mcpActivationEvidenceBody.innerHTML = `
MCP 啟用證據審核失敗:${escapeHtml(error.message)}
`; + } + }; + const renderManualSampleMeta = data => { manualSampleMeta.innerHTML = [ `mode=${data.mode || 'unknown'}`, @@ -11509,6 +11652,12 @@ if (mcpCompletionRefresh) { mcpCompletionRefresh.addEventListener('click', loadMcpCompletion); } + if (mcpActivationEvidenceRefresh) { + mcpActivationEvidenceRefresh.addEventListener('click', loadMcpActivationEvidence); + } + if (mcpActivationEvidenceReview) { + mcpActivationEvidenceReview.addEventListener('click', reviewMcpActivationEvidence); + } if (manualSampleRefresh) { manualSampleRefresh.addEventListener('click', loadManualSample); } @@ -11763,6 +11912,7 @@ loadMcpActivation(); loadMcpFetchGate(); loadMcpCompletion(); + loadMcpActivationEvidence(); loadManualSample(); loadSampleAcceptance(); loadSampleReview(); diff --git a/tests/test_market_intel_skeleton.py b/tests/test_market_intel_skeleton.py index 98eeaba..9128764 100644 --- a/tests/test_market_intel_skeleton.py +++ b/tests/test_market_intel_skeleton.py @@ -14,6 +14,7 @@ from services.market_intel.adapters import get_adapter, get_adapter_summaries from services.market_intel.candidate_preview import build_candidate_preview_from_discovery from services.market_intel.discovery_runner import ManualDiscoveryRunner from services.market_intel.html_diagnostics import parse_html_diagnostics +from services.market_intel.mcp_activation_evidence import build_mcp_activation_evidence_preview from services.market_intel.mcp_activation_runbook import build_mcp_activation_runbook_preview from services.market_intel.mcp_completion_audit import build_mcp_completion_audit_preview from services.market_intel.mcp_contract import build_mcp_tool_contract_preview @@ -940,6 +941,8 @@ def test_market_intel_preview_template_uses_safe_fetch_false_endpoint(): assert "market_intel.market_intel_mcp_fetch_gate" in template assert "market_intel.market_intel_mcp_completion_audit" in template assert "data-market-intel-mcp-completion" in template + assert "market_intel.market_intel_mcp_activation_evidence" in template + assert "data-market-intel-mcp-activation-evidence" in template assert "market_intel.market_intel_manual_sample_plan" in template assert "market_intel.market_intel_manual_sample_acceptance" in template assert "market_intel.market_intel_manual_sample_review" in template @@ -1410,7 +1413,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_116_market_intel_mcp_completion_audit" + assert bridge["phase"] == "phase_117_market_intel_mcp_activation_evidence" assert bridge["execute_requested"] is False assert bridge["read_only_query_executed"] is False assert bridge["database_connection_opened"] is False @@ -1568,7 +1571,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_116_market_intel_mcp_completion_audit" + assert contract["phase"] == "phase_117_market_intel_mcp_activation_evidence" assert contract["caller"] == "market_intel" assert contract["contract_ready"] is True assert contract["blocked_reasons"] == [] @@ -1701,7 +1704,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_116_market_intel_mcp_completion_audit" + assert data["phase"] == "phase_117_market_intel_mcp_activation_evidence" assert data["deployment_actions_executed"] is False assert data["docker_command_executed"] is False assert data["ssh_command_executed"] is False @@ -1714,7 +1717,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_116_market_intel_mcp_completion_audit" + assert gate["phase"] == "phase_117_market_intel_mcp_activation_evidence" assert gate["fetch_requested"] is True assert gate["manual_fetch_gate_open"] is False assert gate["network_request_allowed"] is False @@ -1784,7 +1787,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_116_market_intel_mcp_completion_audit" + assert data["phase"] == "phase_117_market_intel_mcp_activation_evidence" assert data["fetch_requested"] is False assert data["network_request_allowed"] is False assert data["external_network_executed"] is False @@ -1798,7 +1801,7 @@ def test_mcp_completion_audit_summarizes_external_and_internal_state(monkeypatch audit = MarketIntelService().build_mcp_completion_audit() assert audit["mode"] == "mcp_completion_audit_preview" - assert audit["phase"] == "phase_116_market_intel_mcp_completion_audit" + assert audit["phase"] == "phase_117_market_intel_mcp_activation_evidence" assert audit["audit_ready_for_operator_review"] is True assert audit["audit_preview_safe"] is True assert audit["external_mcp_runtime_complete"] is False @@ -1872,7 +1875,7 @@ def test_mcp_completion_audit_route_is_preview_only(): assert response.status_code == 200 assert data["mode"] == "mcp_completion_audit_preview" - assert data["phase"] == "phase_116_market_intel_mcp_completion_audit" + assert data["phase"] == "phase_117_market_intel_mcp_activation_evidence" assert data["audit_preview_safe"] is True assert data["external_mcp_runtime_complete"] is False assert data["internal_mcp_contract_complete"] is True @@ -1887,11 +1890,112 @@ def test_mcp_completion_audit_route_is_preview_only(): assert data["scheduler_attached"] is False +def test_mcp_activation_evidence_preview_is_safe_without_payload(): + evidence = build_mcp_activation_evidence_preview( + phase="phase_117_market_intel_mcp_activation_evidence" + ) + + assert evidence["mode"] == "mcp_activation_evidence_preview" + assert evidence["phase"] == "phase_117_market_intel_mcp_activation_evidence" + assert evidence["evidence_payload_received"] is False + assert evidence["activation_evidence_accepted"] is False + assert evidence["ready_for_runtime_promotion"] is False + assert evidence["payload_persisted"] is False + assert evidence["evidence_persisted"] is False + assert evidence["api_executes_health_check"] is False + assert evidence["api_opens_database_connection"] is False + assert evidence["api_writes_database"] is False + assert evidence["api_uses_external_network"] is False + assert evidence["database_write_executed"] is False + assert evidence["external_network_executed"] is False + assert evidence["scheduler_attached"] is False + assert "evidence_payload_received" in evidence["blocked_reasons"] + assert evidence["sample_evidence_template"]["required_env_vars"][ + "MCP_POSTGRES_PASSWORD" + ] == "redacted" + + +def test_mcp_activation_evidence_accepts_redacted_runtime_evidence(): + sample = build_mcp_activation_evidence_preview()[ + "sample_evidence_template" + ] + evidence = build_mcp_activation_evidence_preview( + evidence=sample, + phase="phase_117_market_intel_mcp_activation_evidence", + ) + + assert evidence["mode"] == "mcp_activation_evidence_review" + assert evidence["activation_evidence_accepted"] is True + assert evidence["ready_for_runtime_promotion"] is True + assert evidence["external_mcp_runtime_evidence_complete"] is True + assert evidence["internal_mcp_runtime_evidence_complete"] is True + assert evidence["blocked_reasons"] == [] + assert evidence["passed_gate_count"] == evidence["gate_count"] + assert all(item["passed"] for item in evidence["health_statuses"]) + assert all(item["present"] for item in evidence["env_statuses"]) + assert evidence["secret_literal_keys"] == [] + assert evidence["prohibited_flags"] == [] + assert evidence["payload_persisted"] is False + assert evidence["api_writes_database"] is False + assert evidence["external_network_executed"] is False + + +def test_mcp_activation_evidence_blocks_secret_literals_and_write_flags(): + sample = build_mcp_activation_evidence_preview()[ + "sample_evidence_template" + ] + sample["required_env_vars"]["TAVILY_API_KEY"] = "real-secret-token" + sample["database_write_executed"] = True + + evidence = build_mcp_activation_evidence_preview(evidence=sample) + + assert evidence["activation_evidence_accepted"] is False + assert "no_secret_literals_in_payload" in evidence["blocked_reasons"] + assert "no_write_or_scheduler_evidence" in evidence["blocked_reasons"] + assert evidence["secret_literal_keys"] == ["TAVILY_API_KEY"] + assert evidence["prohibited_flags"] == ["database_write_executed"] + assert evidence["database_write_executed"] is False + assert evidence["payload_persisted"] is False + + +def test_mcp_activation_evidence_route_get_and_post_are_preview_only(): + from routes.market_intel_routes import market_intel_bp + + app = Flask(__name__) + app.secret_key = "test-secret" + app.register_blueprint(market_intel_bp) + client = app.test_client() + with client.session_transaction() as session: + session["logged_in"] = True + + get_response = client.get("/api/market_intel/mcp_activation_evidence") + get_data = get_response.get_json() + sample = get_data["sample_evidence_template"] + post_response = client.post( + "/api/market_intel/mcp_activation_evidence", + json={"evidence": sample}, + ) + post_data = post_response.get_json() + + assert get_response.status_code == 200 + assert get_data["mode"] == "mcp_activation_evidence_preview" + assert get_data["phase"] == "phase_117_market_intel_mcp_activation_evidence" + assert get_data["api_executes_health_check"] is False + assert get_data["api_writes_database"] is False + assert post_response.status_code == 200 + assert post_data["mode"] == "mcp_activation_evidence_review" + assert post_data["phase"] == "phase_117_market_intel_mcp_activation_evidence" + assert post_data["activation_evidence_accepted"] is True + assert post_data["payload_persisted"] is False + assert post_data["api_opens_database_connection"] is False + assert post_data["api_uses_external_network"] is False + + 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_116_market_intel_mcp_completion_audit" + assert plan["phase"] == "phase_117_market_intel_mcp_activation_evidence" assert plan["ready_for_manual_sample_fetch"] is False assert plan["sample_fetch_executed"] is False assert plan["external_network_executed"] is False @@ -1939,7 +2043,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_116_market_intel_mcp_completion_audit" + assert data["phase"] == "phase_117_market_intel_mcp_activation_evidence" assert data["sample_fetch_executed"] is False assert data["external_network_executed"] is False assert data["database_write_executed"] is False @@ -1950,7 +2054,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_116_market_intel_mcp_completion_audit" + assert acceptance["phase"] == "phase_117_market_intel_mcp_activation_evidence" assert acceptance["contract_ready"] is True assert acceptance["sample_result_loaded"] is False assert acceptance["sample_result_accepted"] is False @@ -1992,7 +2096,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_116_market_intel_mcp_completion_audit" + assert data["phase"] == "phase_117_market_intel_mcp_activation_evidence" assert data["sample_result_loaded"] is False assert data["candidate_import_allowed"] is False assert data["external_network_executed"] is False @@ -2004,7 +2108,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_116_market_intel_mcp_completion_audit" + assert review["phase"] == "phase_117_market_intel_mcp_activation_evidence" assert review["contract_ready"] is True assert review["sample_result_loaded"] is False assert review["sample_result_reviewed"] is False @@ -2115,7 +2219,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_116_market_intel_mcp_completion_audit" + assert data["phase"] == "phase_117_market_intel_mcp_activation_evidence" assert data["sample_result_loaded"] is False assert data["sample_result_reviewed"] is False assert data["candidate_import_allowed"] is False @@ -2154,7 +2258,7 @@ def test_manual_sample_review_evaluation_preview_accepts_payload_without_persist ) assert review["mode"] == "manual_sample_review_evaluation_preview" - assert review["phase"] == "phase_116_market_intel_mcp_completion_audit" + assert review["phase"] == "phase_117_market_intel_mcp_activation_evidence" assert review["review_request_type"] == "operator_posted_json" assert review["payload_received"] is True assert review["payload_valid_json_object"] is True @@ -2216,7 +2320,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_116_market_intel_mcp_completion_audit" + assert data["phase"] == "phase_117_market_intel_mcp_activation_evidence" assert data["payload_received"] is True assert data["payload_valid_json_object"] is True assert data["payload_persisted"] is False @@ -2296,7 +2400,7 @@ def test_manual_sample_candidate_handoff_preview_creates_candidates_without_pers ) assert handoff["mode"] == "manual_sample_candidate_handoff_preview" - assert handoff["phase"] == "phase_116_market_intel_mcp_completion_audit" + assert handoff["phase"] == "phase_117_market_intel_mcp_activation_evidence" assert handoff["payload_received"] is True assert handoff["payload_valid_json_object"] is True assert handoff["payload_persisted"] is False @@ -2360,7 +2464,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_116_market_intel_mcp_completion_audit" + assert data["phase"] == "phase_117_market_intel_mcp_activation_evidence" assert data["payload_received"] is True assert data["handoff_ready"] is True assert data["candidate_handoff_created"] is True @@ -2419,7 +2523,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_116_market_intel_mcp_completion_audit" + assert queue_draft["phase"] == "phase_117_market_intel_mcp_activation_evidence" assert queue_draft["payload_received"] is True assert queue_draft["payload_valid_json_object"] is True assert queue_draft["payload_persisted"] is False @@ -2493,7 +2597,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_116_market_intel_mcp_completion_audit" + assert data["phase"] == "phase_117_market_intel_mcp_activation_evidence" assert data["payload_received"] is True assert data["handoff_ready"] is True assert data["queue_draft_ready"] is True @@ -2556,7 +2660,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_116_market_intel_mcp_completion_audit" + assert approval["phase"] == "phase_117_market_intel_mcp_activation_evidence" assert approval["payload_received"] is True assert approval["payload_valid_json_object"] is True assert approval["payload_persisted"] is False @@ -2634,7 +2738,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_116_market_intel_mcp_completion_audit" + assert data["phase"] == "phase_117_market_intel_mcp_activation_evidence" assert data["payload_received"] is True assert data["approval_preview_created"] is True assert data["approval_request_created"] is False @@ -2697,7 +2801,7 @@ def test_manual_sample_candidate_queue_transaction_preview_blocks_execution(): ) assert transaction["mode"] == "manual_sample_candidate_queue_transaction_preview" - assert transaction["phase"] == "phase_116_market_intel_mcp_completion_audit" + assert transaction["phase"] == "phase_117_market_intel_mcp_activation_evidence" assert transaction["payload_received"] is True assert transaction["payload_valid_json_object"] is True assert transaction["payload_persisted"] is False @@ -2777,7 +2881,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_116_market_intel_mcp_completion_audit" + assert data["phase"] == "phase_117_market_intel_mcp_activation_evidence" assert data["payload_received"] is True assert data["transaction_preview_created"] is True assert data["transaction_ready"] is False @@ -8488,7 +8592,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_116_market_intel_mcp_completion_audit" + "phase_117_market_intel_mcp_activation_evidence" ) assert data["telegram_dispatch_report_input_ready"] is False assert data["summary_persistence_telegram_dispatch_report_input_ready"] is False @@ -8563,7 +8667,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_116_market_intel_mcp_completion_audit" + "phase_117_market_intel_mcp_activation_evidence" ) assert data["telegram_dispatch_archive_summary_ready"] is False assert ( @@ -8835,7 +8939,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_116_market_intel_mcp_completion_audit" + "phase_117_market_intel_mcp_activation_evidence" ) assert data["telegram_dispatch_report_input_ready"] is False assert data["summary_persistence_telegram_dispatch_report_input_ready"] is False @@ -9123,7 +9227,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_116_market_intel_mcp_completion_audit" + "phase_117_market_intel_mcp_activation_evidence" ) assert data["telegram_dispatch_report_run_package_ready"] is False assert ( @@ -9433,7 +9537,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_116_market_intel_mcp_completion_audit" + "phase_117_market_intel_mcp_activation_evidence" ) assert data["telegram_dispatch_report_run_readiness_ready"] is False assert ( @@ -9736,7 +9840,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_116_market_intel_mcp_completion_audit" + "phase_117_market_intel_mcp_activation_evidence" ) assert data["telegram_dispatch_report_run_receipt_passed"] is False assert ( @@ -9995,7 +10099,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_116_market_intel_mcp_completion_audit" + "phase_117_market_intel_mcp_activation_evidence" ) assert data["telegram_dispatch_report_closeout_passed"] is False assert ( @@ -10268,7 +10372,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_116_market_intel_mcp_completion_audit" + "phase_117_market_intel_mcp_activation_evidence" ) assert data["telegram_dispatch_report_archive_passed"] is False assert ( @@ -10516,7 +10620,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_116_market_intel_mcp_completion_audit" + "phase_117_market_intel_mcp_activation_evidence" ) assert data["telegram_dispatch_report_archive_summary_passed"] is False assert ( @@ -10746,7 +10850,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_116_market_intel_mcp_completion_audit" + "phase_117_market_intel_mcp_activation_evidence" ) assert data["telegram_dispatch_report_catalog_handoff_passed"] is False assert data["summary_persistence_telegram_dispatch_report_catalog_handoff_passed"] is False @@ -10983,7 +11087,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_116_market_intel_mcp_completion_audit" + "phase_117_market_intel_mcp_activation_evidence" ) assert data["telegram_dispatch_report_catalog_index_passed"] is False assert data["summary_persistence_telegram_dispatch_report_catalog_index_passed"] is False @@ -11225,7 +11329,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_116_market_intel_mcp_completion_audit" + "phase_117_market_intel_mcp_activation_evidence" ) assert data["telegram_dispatch_report_catalog_write_preflight_passed"] is False assert ( @@ -11501,7 +11605,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_116_market_intel_mcp_completion_audit" + "phase_117_market_intel_mcp_activation_evidence" ) assert data["telegram_dispatch_report_catalog_record_write_passed"] is False assert ( @@ -11778,7 +11882,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_116_market_intel_mcp_completion_audit" + "phase_117_market_intel_mcp_activation_evidence" ) assert data["telegram_dispatch_report_catalog_record_run_package_passed"] is False assert ( @@ -12057,7 +12161,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_116_market_intel_mcp_completion_audit" + "phase_117_market_intel_mcp_activation_evidence" ) assert data["telegram_dispatch_report_catalog_record_run_readiness_passed"] is False assert ( @@ -12386,7 +12490,7 @@ def test_candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_ "candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_run_receipt_preview" ) assert data["phase"] == ( - "phase_116_market_intel_mcp_completion_audit" + "phase_117_market_intel_mcp_activation_evidence" ) assert data["telegram_dispatch_report_catalog_record_run_receipt_passed"] is False assert ( @@ -12648,7 +12752,7 @@ def test_candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_ "candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_commit_preview" ) assert data["phase"] == ( - "phase_116_market_intel_mcp_completion_audit" + "phase_117_market_intel_mcp_activation_evidence" ) assert data["telegram_dispatch_report_catalog_record_commit_passed"] is False assert ( @@ -12912,7 +13016,7 @@ def test_candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_ "candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_closeout_preview" ) assert data["phase"] == ( - "phase_116_market_intel_mcp_completion_audit" + "phase_117_market_intel_mcp_activation_evidence" ) assert data["telegram_dispatch_report_catalog_record_closeout_passed"] is False assert ( @@ -13182,7 +13286,7 @@ def test_candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_ "candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_archive_preview" ) assert data["phase"] == ( - "phase_116_market_intel_mcp_completion_audit" + "phase_117_market_intel_mcp_activation_evidence" ) assert data["telegram_dispatch_report_catalog_record_archive_passed"] is False assert ( @@ -13470,7 +13574,7 @@ def test_candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_ "candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_archive_summary_preview" ) assert data["phase"] == ( - "phase_116_market_intel_mcp_completion_audit" + "phase_117_market_intel_mcp_activation_evidence" ) assert data["telegram_dispatch_report_catalog_record_archive_summary_passed"] is False assert ( @@ -13757,7 +13861,7 @@ def test_candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_ "candidate_queue_review_ai_summary_persistence_telegram_dispatch_report_catalog_record_final_closeout_preview" ) assert data["phase"] == ( - "phase_116_market_intel_mcp_completion_audit" + "phase_117_market_intel_mcp_activation_evidence" ) assert data["telegram_dispatch_report_catalog_record_final_closeout_passed"] is False assert ( @@ -13843,7 +13947,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_116_market_intel_mcp_completion_audit" + assert data["phase"] == "phase_117_market_intel_mcp_activation_evidence" assert data["execute_requested"] is False assert data["read_only_query_executed"] is False assert data["database_connection_opened"] is False @@ -13900,7 +14004,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_116_market_intel_mcp_completion_audit" + assert data["phase"] == "phase_117_market_intel_mcp_activation_evidence" assert data["execute_requested"] is True assert data["apply_real_write_requested"] is True assert data["approval_token_present"] is False @@ -13989,7 +14093,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_116_market_intel_mcp_completion_audit" + assert data["phase"] == "phase_117_market_intel_mcp_activation_evidence" assert data["execute_requested"] is False assert data["read_only_query_executed"] is False assert data["database_connection_opened"] is False @@ -14043,7 +14147,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_116_market_intel_mcp_completion_audit" + assert data["phase"] == "phase_117_market_intel_mcp_activation_evidence" assert data["operator_drill_ready"] is True assert data["api_executes_cli"] is False assert data["api_reads_approval_token"] is False @@ -14099,7 +14203,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_116_market_intel_mcp_completion_audit" + assert data["phase"] == "phase_117_market_intel_mcp_activation_evidence" assert data["package_ready"] is True assert data["package_artifact_created"] is False assert data["api_writes_file"] is False @@ -14165,7 +14269,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_116_market_intel_mcp_completion_audit" + assert data["phase"] == "phase_117_market_intel_mcp_activation_evidence" 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 @@ -14467,7 +14571,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_116_market_intel_mcp_completion_audit" + assert data["phase"] == "phase_117_market_intel_mcp_activation_evidence" assert data["receipt_passed"] is True assert data["ready_for_api_database_write"] is False assert data["ready_for_scheduler_attach"] is False @@ -14515,7 +14619,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_116_market_intel_mcp_completion_audit" + assert data["phase"] == "phase_117_market_intel_mcp_activation_evidence" assert data["closeout_passed"] is True assert data["ready_for_next_manual_phase"] is True assert data["ready_for_api_database_write"] is False @@ -14564,7 +14668,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_116_market_intel_mcp_completion_audit" + assert data["phase"] == "phase_117_market_intel_mcp_activation_evidence" assert data["handoff_ready"] is True assert data["ready_for_manual_queue_review"] is True assert data["ready_for_api_database_write"] is False @@ -14622,7 +14726,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_116_market_intel_mcp_completion_audit" + assert data["phase"] == "phase_117_market_intel_mcp_activation_evidence" assert data["execute_requested"] is False assert data["review_inventory_ready"] is False assert data["ready_for_human_decision_review"] is False @@ -14688,7 +14792,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_116_market_intel_mcp_completion_audit" + assert data["phase"] == "phase_117_market_intel_mcp_activation_evidence" 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 @@ -14759,7 +14863,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_116_market_intel_mcp_completion_audit" + assert data["phase"] == "phase_117_market_intel_mcp_activation_evidence" 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 @@ -14835,7 +14939,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_116_market_intel_mcp_completion_audit" + assert data["phase"] == "phase_117_market_intel_mcp_activation_evidence" assert data["transaction_preview_created"] is False assert data["transaction_ready"] is False assert data["ready_for_manual_shell_update_window"] is False @@ -14917,7 +15021,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_116_market_intel_mcp_completion_audit" + assert data["phase"] == "phase_117_market_intel_mcp_activation_evidence" assert data["execute_requested"] is True assert data["apply_real_write_requested"] is True assert data["approval_token_present"] is False @@ -15003,7 +15107,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_116_market_intel_mcp_completion_audit" + assert data["phase"] == "phase_117_market_intel_mcp_activation_evidence" assert data["execute_requested"] is True assert data["apply_real_write_requested"] is True assert data["read_only_query_executed"] is False @@ -15086,7 +15190,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_116_market_intel_mcp_completion_audit" + assert data["phase"] == "phase_117_market_intel_mcp_activation_evidence" assert data["execute_requested"] is False assert data["read_only_query_executed"] is False assert data["database_connection_opened"] is False @@ -15169,7 +15273,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_116_market_intel_mcp_completion_audit" + assert data["phase"] == "phase_117_market_intel_mcp_activation_evidence" 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 @@ -15255,7 +15359,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_116_market_intel_mcp_completion_audit" + assert data["phase"] == "phase_117_market_intel_mcp_activation_evidence" assert data["package_ready"] is False assert data["package_artifact_created"] is False assert data["ready_for_api_review_state_update"] is False @@ -15346,7 +15450,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_116_market_intel_mcp_completion_audit" + "phase_117_market_intel_mcp_activation_evidence" ) assert data["ready_for_cli_operator_run"] is False assert data["ready_for_api_review_state_update"] is False @@ -15456,7 +15560,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_116_market_intel_mcp_completion_audit" + "phase_117_market_intel_mcp_activation_evidence" ) assert data["receipt_passed"] is False assert data["ready_for_api_review_state_update"] is False @@ -15542,7 +15646,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_116_market_intel_mcp_completion_audit" + "phase_117_market_intel_mcp_activation_evidence" ) assert data["closeout_passed"] is False assert data["ready_for_api_review_state_update"] is False @@ -15599,7 +15703,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_116_market_intel_mcp_completion_audit" + "phase_117_market_intel_mcp_activation_evidence" ) assert data["post_closeout_inventory_ready"] is False assert data["ready_for_api_review_state_update"] is False @@ -15652,7 +15756,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_116_market_intel_mcp_completion_audit" + assert data["phase"] == "phase_117_market_intel_mcp_activation_evidence" 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 @@ -15705,7 +15809,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_116_market_intel_mcp_completion_audit" + assert data["phase"] == "phase_117_market_intel_mcp_activation_evidence" assert data["archive_summary_ready"] is False assert data["summary_input_ready"] is False assert data["ready_for_ai_summary_generation"] is False @@ -15766,7 +15870,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_116_market_intel_mcp_completion_audit" + assert data["phase"] == "phase_117_market_intel_mcp_activation_evidence" 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 @@ -15835,7 +15939,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_116_market_intel_mcp_completion_audit" + assert data["phase"] == "phase_117_market_intel_mcp_activation_evidence" 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 @@ -15907,7 +16011,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_116_market_intel_mcp_completion_audit" + assert data["phase"] == "phase_117_market_intel_mcp_activation_evidence" 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 @@ -15980,7 +16084,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_116_market_intel_mcp_completion_audit" + "phase_117_market_intel_mcp_activation_evidence" ) assert data["summary_persistence_preflight_ready"] is False assert data["ready_for_summary_transaction_preview"] is False @@ -16051,7 +16155,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_116_market_intel_mcp_completion_audit" + "phase_117_market_intel_mcp_activation_evidence" ) assert data["summary_persistence_transaction_ready"] is False assert data["ready_for_summary_persistence_writer_gate"] is False @@ -16116,7 +16220,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_116_market_intel_mcp_completion_audit" + "phase_117_market_intel_mcp_activation_evidence" ) assert data["summary_persistence_writer_preflight_ready"] is False assert data["ready_for_summary_persistence_run_package"] is False @@ -16187,7 +16291,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_116_market_intel_mcp_completion_audit" + "phase_117_market_intel_mcp_activation_evidence" ) assert data["package_ready"] is False assert data["ready_for_summary_persistence_run_readiness"] is False @@ -16260,7 +16364,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_116_market_intel_mcp_completion_audit" + "phase_117_market_intel_mcp_activation_evidence" ) assert data["run_readiness_ready"] is False assert data["summary_persistence_run_readiness_ready"] is False @@ -16337,7 +16441,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_116_market_intel_mcp_completion_audit" + "phase_117_market_intel_mcp_activation_evidence" ) assert data["run_receipt_passed"] is False assert data["summary_persistence_run_receipt_passed"] is False @@ -16414,7 +16518,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_116_market_intel_mcp_completion_audit" + "phase_117_market_intel_mcp_activation_evidence" ) assert data["closeout_passed"] is False assert data["summary_persistence_closeout_passed"] is False @@ -16491,7 +16595,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_116_market_intel_mcp_completion_audit" + "phase_117_market_intel_mcp_activation_evidence" ) assert data["telegram_dispatch_gate_passed"] is False assert data["summary_persistence_telegram_dispatch_gate_passed"] is False @@ -16565,7 +16669,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_116_market_intel_mcp_completion_audit" + "phase_117_market_intel_mcp_activation_evidence" ) assert data["telegram_dispatch_run_package_ready"] is False assert data["summary_persistence_telegram_dispatch_run_package_ready"] is False @@ -16644,7 +16748,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_116_market_intel_mcp_completion_audit" + "phase_117_market_intel_mcp_activation_evidence" ) assert data["telegram_dispatch_run_readiness_ready"] is False assert ( @@ -16731,7 +16835,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_116_market_intel_mcp_completion_audit" + "phase_117_market_intel_mcp_activation_evidence" ) assert data["telegram_dispatch_run_receipt_passed"] is False assert data["summary_persistence_telegram_dispatch_run_receipt_passed"] is False @@ -16812,7 +16916,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_116_market_intel_mcp_completion_audit" + "phase_117_market_intel_mcp_activation_evidence" ) assert data["telegram_dispatch_closeout_passed"] is False assert data["summary_persistence_telegram_dispatch_closeout_passed"] is False @@ -16894,7 +16998,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_116_market_intel_mcp_completion_audit" + "phase_117_market_intel_mcp_activation_evidence" ) assert data["telegram_dispatch_archive_ready"] is False assert data["summary_persistence_telegram_dispatch_archive_ready"] is False @@ -16977,7 +17081,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_116_market_intel_mcp_completion_audit" + assert data["phase"] == "phase_117_market_intel_mcp_activation_evidence" assert data["receipt_passed"] is True assert data["ready_for_next_manual_review"] is True assert data["ready_for_api_database_write"] is False @@ -17002,7 +17106,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_116_market_intel_mcp_completion_audit" + assert plan["phase"] == "phase_117_market_intel_mcp_activation_evidence" assert plan["ready_to_attach_scheduler"] is False assert plan["scheduler_attached"] is False assert plan["scheduler_registration_executed"] is False @@ -17040,7 +17144,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_116_market_intel_mcp_completion_audit" + assert data["phase"] == "phase_117_market_intel_mcp_activation_evidence" assert data["scheduler_registration_executed"] is False assert data["crawler_job_started"] is False assert data["external_network_executed"] is False @@ -17051,7 +17155,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_116_market_intel_mcp_completion_audit" + assert plan["phase"] == "phase_117_market_intel_mcp_activation_evidence" assert plan["ready_for_review_queue"] is False assert plan["review_queue_created"] is False assert plan["auto_match_executed"] is False @@ -17087,7 +17191,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_116_market_intel_mcp_completion_audit" + assert data["phase"] == "phase_117_market_intel_mcp_activation_evidence" assert data["review_queue_created"] is False assert data["auto_confirm_executed"] is False assert data["external_network_executed"] is False @@ -17098,7 +17202,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_116_market_intel_mcp_completion_audit" + assert plan["phase"] == "phase_117_market_intel_mcp_activation_evidence" assert plan["ready_for_opportunity_queue"] is False assert plan["opportunity_queue_created"] is False assert plan["threat_alert_dispatched"] is False @@ -17139,7 +17243,7 @@ def test_opportunity_plan_route_is_preview_only(): assert response.status_code == 200 assert data["mode"] == "opportunity_plan_preview" - assert data["phase"] == "phase_116_market_intel_mcp_completion_audit" + assert data["phase"] == "phase_117_market_intel_mcp_activation_evidence" assert data["opportunity_queue_created"] is False assert data["threat_alert_dispatched"] is False assert data["ai_summary_generated"] is False @@ -17150,7 +17254,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_116_market_intel_mcp_completion_audit" + assert plan["phase"] == "phase_117_market_intel_mcp_activation_evidence" assert plan["ready_for_scoring_job"] is False assert plan["scoring_job_created"] is False assert plan["score_calculation_executed"] is False @@ -17198,7 +17302,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_116_market_intel_mcp_completion_audit" + assert data["phase"] == "phase_117_market_intel_mcp_activation_evidence" assert data["scoring_job_created"] is False assert data["score_calculation_executed"] is False assert data["sample_scores_generated"] is False @@ -17210,7 +17314,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_116_market_intel_mcp_completion_audit" + assert plan["phase"] == "phase_117_market_intel_mcp_activation_evidence" assert plan["ready_for_evidence_bundle"] is False assert plan["evidence_bundle_created"] is False assert plan["evidence_query_executed"] is False @@ -17256,7 +17360,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_116_market_intel_mcp_completion_audit" + assert data["phase"] == "phase_117_market_intel_mcp_activation_evidence" assert data["evidence_bundle_created"] is False assert data["evidence_query_executed"] is False assert data["sample_evidence_generated"] is False @@ -17269,7 +17373,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_116_market_intel_mcp_completion_audit" + assert plan["phase"] == "phase_117_market_intel_mcp_activation_evidence" assert plan["ready_for_alert_candidates"] is False assert plan["alert_candidate_created"] is False assert plan["alert_queue_created"] is False @@ -17354,7 +17458,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_116_market_intel_mcp_completion_audit" + assert data["phase"] == "phase_117_market_intel_mcp_activation_evidence" assert data["alert_candidate_created"] is False assert data["alert_queue_created"] is False assert data["review_queue_created"] is False @@ -17432,7 +17536,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_116_market_intel_mcp_completion_audit" + assert data["phase"] == "phase_117_market_intel_mcp_activation_evidence" assert data["deployment_actions_executed"] is False assert data["docker_command_executed"] is False assert data["ssh_command_executed"] is False @@ -17447,7 +17551,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_116_market_intel_mcp_completion_audit" + assert readiness["phase"] == "phase_117_market_intel_mcp_activation_evidence" assert readiness["execute_requested"] is False assert readiness["router_enabled"] is False assert readiness["external_mcp_complete"] is False @@ -17835,6 +17939,7 @@ def test_deployment_readiness_reports_app_only_release_gate(): assert readiness["checks"]["mcp_activation_runbook_preview_safe"] is True assert readiness["checks"]["mcp_fetch_gate_preview_safe"] is True assert readiness["checks"]["mcp_completion_audit_preview_safe"] is True + assert readiness["checks"]["mcp_activation_evidence_preview_safe"] is True assert readiness["checks"]["scheduler_plan_preview_safe"] is True assert readiness["checks"]["manual_sample_plan_preview_safe"] is True assert readiness["checks"]["manual_sample_acceptance_preview_safe"] is True @@ -18206,6 +18311,7 @@ def test_deployment_readiness_reports_app_only_release_gate(): assert "/api/market_intel/mcp_activation_runbook" in readiness["production_smoke_targets"] assert "/api/market_intel/mcp_fetch_gate" in readiness["production_smoke_targets"] assert "/api/market_intel/mcp_completion_audit" in readiness["production_smoke_targets"] + assert "/api/market_intel/mcp_activation_evidence" in readiness["production_smoke_targets"] assert "/api/market_intel/scheduler_plan" in readiness["production_smoke_targets"] assert "/api/market_intel/manual_sample_plan" in readiness["production_smoke_targets"] assert "/api/market_intel/manual_sample_acceptance" in readiness["production_smoke_targets"] @@ -18542,6 +18648,12 @@ def test_deployment_readiness_reports_app_only_release_gate(): assert readiness["mcp_completion_audit"]["api_writes_database"] is False assert readiness["mcp_completion_audit"]["api_uses_external_network"] is False assert readiness["mcp_completion_audit"]["ready_for_manual_fetch"] is False + assert readiness["mcp_activation_evidence"]["mode"] == "mcp_activation_evidence_preview" + assert readiness["mcp_activation_evidence"]["evidence_payload_received"] is False + assert readiness["mcp_activation_evidence"]["payload_persisted"] is False + assert readiness["mcp_activation_evidence"]["api_executes_health_check"] is False + assert readiness["mcp_activation_evidence"]["api_writes_database"] is False + assert readiness["mcp_activation_evidence"]["api_uses_external_network"] is False assert readiness["manual_sample_plan"]["mode"] == "manual_sample_fetch_plan_preview" assert readiness["manual_sample_plan"]["sample_fetch_executed"] is False assert readiness["manual_sample_plan"]["external_network_executed"] is False @@ -22161,7 +22273,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_116_market_intel_mcp_completion_audit" + assert drill["phase"] == "phase_117_market_intel_mcp_activation_evidence" assert drill["execute_requested"] is False assert drill["schema_state"] == "planned_no_db_probe" assert drill["drill_ready_for_operator_review"] is True @@ -22276,7 +22388,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_116_market_intel_mcp_completion_audit" + assert data["phase"] == "phase_117_market_intel_mcp_activation_evidence" assert data["execute_requested"] is False assert data["migration_executed"] is False assert data["rollback_executed"] is False @@ -22288,7 +22400,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_116_market_intel_mcp_completion_audit" + assert review["phase"] == "phase_117_market_intel_mcp_activation_evidence" assert review["execute_requested"] is False assert review["catalog_state"] == "planned_no_probe" assert review["seed_state"] == "planned_no_probe" @@ -22403,7 +22515,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_116_market_intel_mcp_completion_audit" + assert data["phase"] == "phase_117_market_intel_mcp_activation_evidence" assert data["execute_requested"] is False assert data["catalog_state"] == "planned_no_probe" assert data["migration_executed"] is False @@ -22416,7 +22528,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_116_market_intel_mcp_completion_audit" + assert smoke["phase"] == "phase_117_market_intel_mcp_activation_evidence" assert smoke["execute_requested"] is False assert smoke["smoke_result"] == "planned_no_execution" assert smoke["live_smoke_passed"] is False @@ -22478,7 +22590,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_116_market_intel_mcp_completion_audit" + assert data["phase"] == "phase_117_market_intel_mcp_activation_evidence" assert data["execute_requested"] is False assert data["smoke_result"] == "planned_no_execution" assert data["migration_executed"] is False @@ -22491,7 +22603,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_116_market_intel_mcp_completion_audit" + assert inventory["phase"] == "phase_117_market_intel_mcp_activation_evidence" assert inventory["execute_requested"] is False assert inventory["read_only_query_executed"] is False assert inventory["database_connection_opened"] is False @@ -22635,7 +22747,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_116_market_intel_mcp_completion_audit" + assert data["phase"] == "phase_117_market_intel_mcp_activation_evidence" assert data["execute_requested"] is False assert data["read_only_query_executed"] is False assert data["database_write_executed"] is False @@ -22871,7 +22983,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_116_market_intel_mcp_completion_audit" + assert data["phase"] == "phase_117_market_intel_mcp_activation_evidence" assert data["execute_requested"] is False assert data["apply_real_write_requested"] is False assert data["writes_executed"] is False @@ -22900,7 +23012,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_116_market_intel_mcp_completion_audit" + assert data["phase"] == "phase_117_market_intel_mcp_activation_evidence" assert data["execute_requested"] is False assert data["apply_real_write_requested"] is False assert data["approval_token_present"] is False