active-exception-id-trigger1-regression.sh 8.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122
  1. #!/usr/bin/env bash
  2. # trigger=1 active_exception_id 回归 — BUG-S8-DETECTION-STATE-ACTIVE-EXC-ID-TRIGGER1-001(dev/test only, aidopdev)。
  3. #
  4. # 验证 trigger_count_required=1 首 tick 建单后:
  5. # - exception 创建成功
  6. # - rule_detection_state.active_exception_id IS NOT NULL
  7. # - rule_detection_state.active_exception_id == 新建 exception.id
  8. # - DetectionLog CREATED.exception_id == 新建 exception.id
  9. #
  10. # 修复前:fresh.Id 在 InsertAsync 后未回填,外层 UPDATE WHERE id=state.Id 命中 0 行 → active_exception_id 永远 NULL。
  11. # 修复后:fresh = await _detectionStateRep.AsInsertable(fresh).ExecuteReturnEntityAsync(),沿用
  12. # S8ManualReportService 既定模式回填 Id,UPDATE WHERE id=state.Id 命中 1 行。
  13. #
  14. # 严守:
  15. # - 不影响 demo rule 10/11/12
  16. # - TEMP rule 测后 enabled=0;TEMP exception soft delete
  17. # - 不清 detection_log / rule_detection_state / 演示数据
  18. set -uo pipefail
  19. SCRIPT_DIR="$(cd "$(dirname "${BASH_SOURCE[0]}")" && pwd)"
  20. # shellcheck source=./s8-regression-common.sh
  21. source "${SCRIPT_DIR}/s8-regression-common.sh"
  22. auth_load
  23. RULE='TEMP_SCHED_TIMEOUT_TRIGGER1_ACTIVE_ID'
  24. DEDUP_OBJ='TEMP-TRIGGER1-ACTIVE-ID-001'
  25. PARAMS_JSON='{"dueAtField":"due_at","statusField":"status","completedStates":["CLOSED","DONE","COMPLETED"],"objectCodeField":"related_object_code","objectIdField":"source_object_id","graceMinutes":0,"exceptionTypeCode":"DELIVERY_DELAY"}'
  26. HIT_EXPR="SELECT '${DEDUP_OBJ}' AS related_object_code, '${DEDUP_OBJ}' AS source_object_id, DATE_SUB(NOW(), INTERVAL 1 HOUR) AS due_at, 'PENDING' AS status"
  27. baseline_before=$(read_baseline)
  28. DEMO_SNAPSHOT=$(snapshot_demo_rule_state)
  29. echo "==== active-exception-id-trigger1-regression baseline_before=${baseline_before} demo_snapshot=${DEMO_SNAPSHOT} ===="
  30. # ---------------------------------------------------------------------------
  31. # Setup TEMP rule (trigger=1, recover=1, hit expression)
  32. # ---------------------------------------------------------------------------
  33. echo "---- Setup TEMP rule ${RULE} (trigger=1, recover=1) ----"
  34. exists=$(mysql_run "SELECT COUNT(*) FROM ado_s8_watch_rule WHERE rule_code='${RULE}' AND tenant_id=${TENANT_ID} AND factory_id=${FACTORY_ID};")
  35. if [[ "${exists}" == "0" ]]; then
  36. mysql_run_strict "INSERT INTO ado_s8_watch_rule
  37. (tenant_id, factory_id, rule_code, scene_code, data_source_id, watch_object_type, expression, severity,
  38. poll_interval_seconds, enabled, created_at, rule_type, source_object_type, params_json,
  39. consecutive_failure_count, trigger_count_required, recover_count_required, next_run_at)
  40. VALUES (${TENANT_ID}, ${FACTORY_ID}, '${RULE}', 'S7', 1, 'ORDER',
  41. \"${HIT_EXPR}\", 'HIGH', 60, 1, NOW(), 'TIMEOUT', 'ORDER', '${PARAMS_JSON}',
  42. 0, 1, 1, NULL);" >/dev/null
  43. echo " inserted ${RULE}"
  44. else
  45. mysql_run_strict "UPDATE ado_s8_watch_rule SET enabled=1, expression=\"${HIT_EXPR}\", params_json='${PARAMS_JSON}',
  46. trigger_count_required=1, recover_count_required=1,
  47. paused_until=NULL, pause_reason=NULL,
  48. lock_token=NULL, locked_by=NULL, lock_until=NULL, running_started_at=NULL,
  49. consecutive_failure_count=0, last_status=NULL, last_error=NULL,
  50. next_run_at=NULL, updated_at=NOW()
  51. WHERE rule_code='${RULE}' AND tenant_id=${TENANT_ID} AND factory_id=${FACTORY_ID};" >/dev/null
  52. echo " re-armed ${RULE}"
  53. fi
  54. RULE_ID=$(get_rule_id_by_code "${RULE}")
  55. DEDUP_KEY="T${TENANT_ID}:F${FACTORY_ID}:R${RULE}:ORDER:${DEDUP_OBJ}"
  56. echo " RULE_ID=${RULE_ID} DEDUP_KEY=${DEDUP_KEY}"
  57. # 清理上次残留 detection_state / TEMP exception,确保从全新状态起跑
  58. mysql_run_strict "DELETE FROM ado_s8_rule_detection_state WHERE rule_code='${RULE}' AND tenant_id=${TENANT_ID} AND factory_id=${FACTORY_ID};" >/dev/null
  59. mysql_run_strict "UPDATE ado_s8_exception SET is_deleted=1, updated_at=NOW() WHERE source_rule_code='${RULE}' AND is_deleted=0;" >/dev/null
  60. marker=$(mysql_run "SELECT NOW();")
  61. sleep 1
  62. # ---------------------------------------------------------------------------
  63. # Trigger first tick via /run-once(trigger=1:本 tick 即应建单 + 回填 active_exception_id)
  64. # ---------------------------------------------------------------------------
  65. echo "---- Trigger first tick via run-once ----"
  66. resp=$(run_once_endpoint || true)
  67. [[ -z "${resp}" ]] && record_fail "run-once returned empty"
  68. sleep 2
  69. # ---------------------------------------------------------------------------
  70. # 验证
  71. # ---------------------------------------------------------------------------
  72. echo "---- Verify ----"
  73. exc_count=$(mysql_run "SELECT COUNT(*) FROM ado_s8_exception WHERE source_rule_code='${RULE}' AND is_deleted=0 AND status<>'CLOSED';")
  74. exc_id=$(mysql_run "SELECT IFNULL(MAX(id),0) FROM ado_s8_exception WHERE source_rule_code='${RULE}' AND is_deleted=0 AND status<>'CLOSED';")
  75. state_active=$(mysql_run "SELECT IFNULL(active_exception_id,0) FROM ado_s8_rule_detection_state WHERE rule_code='${RULE}' AND dedup_key='${DEDUP_KEY}';")
  76. state_hit=$(mysql_run "SELECT IFNULL(consecutive_hit_count,-1) FROM ado_s8_rule_detection_state WHERE rule_code='${RULE}' AND dedup_key='${DEDUP_KEY}';")
  77. created_logs=$(mysql_run "SELECT COUNT(*) FROM ado_s8_detection_log WHERE rule_code='${RULE}' AND detect_result='CREATED' AND detected_at >= '${marker}';")
  78. created_log_excid=$(mysql_run "SELECT IFNULL(exception_id,0) FROM ado_s8_detection_log WHERE rule_code='${RULE}' AND detect_result='CREATED' AND detected_at >= '${marker}' ORDER BY id DESC LIMIT 1;")
  79. echo " exc_count=${exc_count} exc_id=${exc_id} state_active=${state_active} state_hit=${state_hit} CREATED_logs=${created_logs} log.exception_id=${created_log_excid}"
  80. [[ "${exc_count}" == "1" ]] && record_pass "1 active exception created" || record_fail "expected 1 active exception, got ${exc_count}"
  81. [[ "${exc_id}" -gt 0 ]] && record_pass "exception.id=${exc_id} non-zero" || record_fail "exception id missing"
  82. [[ "${state_hit}" == "1" ]] && record_pass "detection_state.consecutive_hit_count=1" || record_fail "expected state.hit_count=1, got ${state_hit}"
  83. [[ "${state_active}" -gt 0 ]] && record_pass "state.active_exception_id=${state_active} (NOT NULL)" || record_fail "state.active_exception_id NULL — BUG NOT FIXED"
  84. [[ "${state_active}" == "${exc_id}" ]] && record_pass "state.active_exception_id == exception.id (${exc_id})" || record_fail "state.active_exception_id=${state_active} != exception.id=${exc_id}"
  85. [[ "${created_logs}" -ge 1 ]] && record_pass "CREATED detection_log present (${created_logs})" || record_fail "CREATED log missing"
  86. [[ "${created_log_excid}" == "${exc_id}" ]] && record_pass "CREATED log.exception_id == exception.id (${exc_id})" || record_fail "CREATED log.exception_id=${created_log_excid} != exception.id=${exc_id}"
  87. # ---------------------------------------------------------------------------
  88. # Cleanup
  89. # ---------------------------------------------------------------------------
  90. echo "---- Cleanup ----"
  91. mysql_run_strict "UPDATE ado_s8_watch_rule SET enabled=0, paused_until=NULL, pause_reason=NULL, lock_token=NULL, locked_by=NULL, lock_until=NULL, running_started_at=NULL, next_run_at=NULL, updated_at=NOW() WHERE id=${RULE_ID};" >/dev/null
  92. TEMP_EXC_IDS=$(mysql_run "SELECT IFNULL(GROUP_CONCAT(id),'none') FROM ado_s8_exception WHERE source_rule_code='${RULE}' AND is_deleted=0;")
  93. mysql_run_strict "UPDATE ado_s8_exception SET is_deleted=1, updated_at=NOW() WHERE source_rule_code='${RULE}' AND is_deleted=0;" >/dev/null
  94. cleanup_temp_sched_approval_ghost_tasks
  95. echo " disabled rule_id=${RULE_ID}; soft-deleted exception ids=[${TEMP_EXC_IDS}]; ghost approval cancelled"
  96. final_enabled=$(get_rule_field "${RULE_ID}" enabled)
  97. [[ "${final_enabled}" == "0" ]] && record_pass "cleanup: rule enabled=0" || record_fail "cleanup: rule still enabled=${final_enabled}"
  98. temp_visible=$(mysql_run "SELECT COUNT(*) FROM ado_s8_exception WHERE source_rule_code='${RULE}' AND is_deleted=0;")
  99. [[ "${temp_visible}" == "0" ]] && record_pass "cleanup: TEMP exception not in default list" || record_fail "cleanup: ${temp_visible} TEMP exception(s) still visible"
  100. # demo rule 漂移检测 + baseline 守恒
  101. assert_demo_rule_state_unchanged "${DEMO_SNAPSHOT}"
  102. assert_baseline_unchanged "${baseline_before}"
  103. print_summary
  104. echo "TEMP rule: ${RULE} (id=${RULE_ID}); TEMP exception ids=[${TEMP_EXC_IDS}]; final state.active_exception_id=${state_active}, exc_id=${exc_id}"
  105. exit_by_summary