Files
sanguo_moziplus_v2/scripts/status.sh
T
2026-05-23 13:30:57 +08:00

129 lines
4.4 KiB
Bash
Executable File

#!/usr/bin/env bash
# status.sh — 查看 moziplus v2 运行状态
set -euo pipefail
TARGET_DIR="${MOZIPLUS_V2_DIR:-$HOME/.sanguo_projects/sanguo_moziplus_v2}"
PM2_NAME="sanguo-moziplus-v2"
DETAILED=false
usage() {
echo "Usage: $0 [options]"
echo " --target=DIR 安装目录 (default: ~/.sanguo_projects/sanguo_moziplus_v2)"
echo " -v, --verbose 显示详细信息"
echo " -h, --help 显示帮助"
exit 0
}
for arg in "$@"; do
case "$arg" in
--target=*) TARGET_DIR="${arg#*=}" ;;
-v|--verbose) DETAILED=true ;;
-h|--help) usage ;;
esac
done
echo "━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━"
echo " moziplus v2 — Status"
echo "━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━"
# ── 版本 ──
VERSION=$(grep "^version" "$TARGET_DIR/pyproject.toml" 2>/dev/null | head -1 | cut -d= -f2 | tr -d " \"\047\t")
echo " Version: v${VERSION}"
# ── 安装目录 ──
if [ -d "$TARGET_DIR" ]; then
echo " Install: $TARGET_DIR"
else
echo " Install: $TARGET_DIR ❌ (not found)"
exit 1
fi
# ── PM2 状态 ──
echo ""
if pm2 describe "$PM2_NAME" >/dev/null 2>&1; then
PM2_INFO=$(pm2 jlist 2>/dev/null | python3 -c "
import sys, json
for p in json.load(sys.stdin):
if p['name'] == '$PM2_NAME':
s = p['pm2_env']['status']
pid = p.get('pid', '?')
r = p['pm2_env'].get('restart_time', 0)
m = round(p['monit']['memory'] / 1024 / 1024, 1)
c = p['monit']['cpu']
u = p['pm2_env'].get('pm_uptime', 0)
print(f'{s}|{pid}|{r}|{m}|{c}|{u}')
break
" 2>/dev/null || echo "offline|?|0|?|?|0")
IFS='|' read -r STATUS PID RESTARTS MEM CPU UPTIME_MS <<< "$PM2_INFO"
if [ "$STATUS" = "online" ]; then
echo " PM2: online ✅"
else
echo " PM2: $STATUS"
fi
echo " PID: ${PID:-?}"
echo " Restarts: ${RESTARTS:-0}"
echo " Memory: ${MEM}MB"
echo " CPU: ${CPU}%"
else
echo " PM2: not found ❌"
fi
# ── 健康检查 ──
echo ""
HEALTH=$(curl -sf http://localhost:8083/api/daemon/status 2>/dev/null || echo "")
if [ -n "$HEALTH" ]; then
H_STATUS=$(echo "$HEALTH" | python3 -c "import sys,json; d=json.load(sys.stdin); print(d.get('status','?'))" 2>/dev/null || echo "?")
H_TICKS=$(echo "$HEALTH" | python3 -c "import sys,json; d=json.load(sys.stdin); print(d.get('tick_count','?'))" 2>/dev/null || echo "?")
H_VERSION=$(echo "$HEALTH" | python3 -c "import sys,json; d=json.load(sys.stdin); print(d.get('version','?'))" 2>/dev/null || echo "?")
echo " Health: ${H_STATUS}"
echo " API ver: ${H_VERSION}"
echo " Ticks: ${H_TICKS}"
else
echo " Health: unreachable ❌"
fi
# ── 数据统计 ──
echo ""
DATA_DIR="$TARGET_DIR/data"
if [ -d "$DATA_DIR" ]; then
PROJ_COUNT=$(find "$DATA_DIR" -mindepth 1 -maxdepth 1 -type d | wc -l | tr -d ' ')
DATA_SIZE=$(du -sh "$DATA_DIR" 2>/dev/null | cut -f1)
echo " Projects: $PROJ_COUNT"
echo " Data size: $DATA_SIZE"
if [ "$DETAILED" = true ]; then
echo ""
echo " Project details:"
for proj_dir in "$DATA_DIR"/*/; do
proj_name=$(basename "$proj_dir")
db_file="$proj_dir/blackboard.db"
if [ -f "$db_file" ]; then
TASK_COUNT=$(sqlite3 "$db_file" "SELECT COUNT(*) FROM tasks" 2>/dev/null || echo "?")
ACTIVE_COUNT=$(sqlite3 "$db_file" "SELECT COUNT(*) FROM tasks WHERE status NOT IN ('done','cancelled','archived')" 2>/dev/null || echo "?")
PROJ_SIZE=$(du -sh "$proj_dir" 2>/dev/null | cut -f1)
printf " %-25s %s tasks (%s active) %s\n" "$proj_name" "$TASK_COUNT" "$ACTIVE_COUNT" "$PROJ_SIZE"
else
printf " %-25s (no db)\n" "$proj_name"
fi
done
fi
fi
# ── 配置 ──
if [ "$DETAILED" = true ]; then
echo ""
CONFIG_YAML="$TARGET_DIR/config/default.yaml"
if [ -f "$CONFIG_YAML" ]; then
echo " Config:"
PORT_VAL=$(grep "port:" "$CONFIG_YAML" 2>/dev/null | head -1 | awk '{print $2}')
INTERVAL_VAL=$(grep "interval:" "$CONFIG_YAML" 2>/dev/null | head -1 | awk '{print $2}')
AGENT_COUNT=$(grep " - " "$CONFIG_YAML" 2>/dev/null | wc -l | tr -d ' ')
echo " Port: ${PORT_VAL:-8083}"
echo " Interval: ${INTERVAL_VAL:-30}"
echo " Agents: ${AGENT_COUNT:-0} configured"
fi
fi
echo "━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━"