# -*- coding: utf-8 -*- from datetime import datetime from flask import Blueprint, current_app, jsonify, render_template, request from utils.decorator.async_job import async_job from .auth import login_required system_jobs_bp = Blueprint("system_jobs", __name__, url_prefix="/system_jobs") def _normalize_datetime_text(value): """统一时间文本格式为 `YYYY-MM-DD HH:MM:SS`。""" if value is None: return value if isinstance(value, datetime): return value.strftime("%Y-%m-%d %H:%M:%S") text = str(value) if "T" in text: return text.replace("T", " ")[:19] return text def _build_job_health_status(*, enabled: bool, running: bool, last_status: str, latest_success_at, latest_failure_summary: str) -> str: """根据任务启停、运行态和历史结果输出后台可读的健康状态。""" # 状态设计尽量贴近运维判断顺序: # 1. 停用态单独标记,避免和“从未执行”混淆; # 2. 执行中的任务优先展示 running,方便后台快速识别实时动作; # 3. 最近一次执行失败时直接标记 failed,让异常任务在列表里一眼可见; # 4. 有成功历史且最近不是失败时视为 healthy,否则落到 idle。 if not enabled: return "disabled" if running: return "running" if str(last_status or "").strip().lower() == "failed": return "failed" if latest_success_at or str(last_status or "").strip().lower() == "success": return "healthy" if str(latest_failure_summary or "").strip(): return "failed" return "idle" def _build_job_health_message(*, health_status: str, latest_success_at, latest_failure_summary: str) -> str: """为后台列表生成一句简短的任务健康提示。""" if health_status == "disabled": return "任务已停用" if health_status == "running": return "任务正在执行中" if health_status == "failed": return str(latest_failure_summary or "最近一次执行失败").strip() if health_status == "healthy": success_text = _normalize_datetime_text(latest_success_at) return f"最近成功于 {success_text}" if success_text else "任务近期执行正常" return "暂无执行记录" @system_jobs_bp.route("/") @login_required def page_system_jobs(): return render_template("system_jobs.html") @system_jobs_bp.route("/api/jobs", methods=["GET"]) @login_required def api_list_jobs(): server = current_app.dashboard_server db_rows = server.system_job_db.list_jobs() runtime_rows = async_job.get_jobs_snapshot() runtime_by_key = {row.get("job_key", ""): row for row in runtime_rows if row.get("job_key")} job_keys = [str(row.get("job_key") or "").strip() for row in db_rows if str(row.get("job_key") or "").strip()] latest_log_by_key = server.system_job_db.get_latest_logs_map(job_keys) history_summary_by_key = server.system_job_db.get_job_history_summary_map(job_keys) result = [] for row in db_rows: job_key = row.get("job_key") runtime = runtime_by_key.get(job_key, {}) latest_log = latest_log_by_key.get(job_key, {}) history_summary = history_summary_by_key.get(job_key, {}) last_status = runtime.get("last_status") or latest_log.get("status") or "never" last_run_at = runtime.get("last_run_at") or latest_log.get("triggered_at") last_error = runtime.get("last_error") or "" if not last_error and str(last_status or "").strip().lower() == "failed": last_error = ( str(latest_log.get("summary") or "").strip() or str(history_summary.get("latest_failure_summary") or "").strip() ) health_status = _build_job_health_status( enabled=bool(row.get("enabled", 0)), running=bool(runtime.get("running", False)), last_status=str(last_status or ""), latest_success_at=history_summary.get("latest_success_at"), latest_failure_summary=str(history_summary.get("latest_failure_summary") or ""), ) result.append( { "job_key": job_key, "name": row.get("name", ""), "description": row.get("description", ""), "trigger_type": row.get("trigger_type", ""), "trigger_config": row.get("trigger_config", {}), "enabled": bool(row.get("enabled", 0)), "runtime_job_id": runtime.get("id"), "runtime_enabled": runtime.get("enabled"), "running": runtime.get("running", False), "trigger_text": runtime.get("trigger_text", ""), "last_run_at": _normalize_datetime_text(last_run_at), "last_status": last_status, "last_error": last_error, "last_duration_ms": runtime.get("last_duration_ms") or latest_log.get("duration_ms"), "next_run_at": _normalize_datetime_text(runtime.get("next_run_at")), "run_count": runtime.get("run_count", 0), "success_count": runtime.get("success_count", 0), "fail_count": runtime.get("fail_count", 0), "latest_success_at": _normalize_datetime_text(history_summary.get("latest_success_at")), "latest_failed_at": _normalize_datetime_text(history_summary.get("latest_failed_at")), "latest_failure_summary": str(history_summary.get("latest_failure_summary") or "").strip(), "history_success_count": int(history_summary.get("history_success_count", 0) or 0), "history_fail_count": int(history_summary.get("history_fail_count", 0) or 0), "history_total_count": int(history_summary.get("history_total_count", 0) or 0), "health_status": health_status, "health_message": _build_job_health_message( health_status=health_status, latest_success_at=history_summary.get("latest_success_at"), latest_failure_summary=str(history_summary.get("latest_failure_summary") or "").strip(), ), } ) return jsonify({"success": True, "data": result}) @system_jobs_bp.route("/api/jobs/", methods=["PUT"]) @login_required def api_update_job(job_key: str): server = current_app.dashboard_server payload = request.get_json(silent=True) or {} updates = {} for key in ("name", "description", "trigger_type", "trigger_config", "enabled"): if key in payload: updates[key] = payload[key] if not updates: return jsonify({"success": False, "message": "没有可更新字段"}), 400 ok = server.system_job_db.update_job(job_key, updates) if not ok: return jsonify({"success": False, "message": "数据库更新失败"}), 500 # 配置变更后立即重载调度器,确保实时生效 server.system_job_loader.reload_from_db() return jsonify({"success": True, "message": "更新成功"}) @system_jobs_bp.route("/api/jobs//trigger", methods=["POST"]) @login_required def api_trigger_job(job_key: str): server = current_app.dashboard_server job_id = async_job.get_job_id_by_key(job_key) if not job_id: server.system_job_loader.reload_from_db() job_id = async_job.get_job_id_by_key(job_key) if not job_id: return jsonify({"success": False, "message": "任务未加载或已禁用"}), 404 ok, msg = async_job.trigger_job_now(job_id, operator="dashboard") code = 200 if ok else 400 return jsonify({"success": ok, "message": msg}), code @system_jobs_bp.route("/api/jobs//logs", methods=["GET"]) @login_required def api_job_logs(job_key: str): server = current_app.dashboard_server limit = int(request.args.get("limit", 100)) db_logs = server.system_job_db.get_job_logs(job_key, limit=limit) # 为了兼容前端既有表头(time/level/message),这里做一层字段映射。 logs = [] for row in db_logs: status = str(row.get("status") or "") level = "error" if status == "failed" else ("success" if status == "success" else "info") logs.append( { "time": _normalize_datetime_text(row.get("triggered_at")), "level": level, "message": row.get("summary") or "", "status": status, "duration_ms": row.get("duration_ms"), "detail_json": row.get("detail_json") or {}, } ) return jsonify({"success": True, "data": logs}) @system_jobs_bp.route("/api/reload", methods=["POST"]) @login_required def api_reload_jobs(): server = current_app.dashboard_server server.system_job_loader.reload_from_db() return jsonify({"success": True, "message": "已按数据库配置重载系统定时任务"})