Files
claude-relay-service/scripts/monitor-enhanced.sh
mouyong 8bdf495ffa feat: enhance monitoring and Gemini service functionality
- Add enhanced monitoring scripts (monitor-enhanced.sh, status-unified.sh)
- Improve Gemini relay service with better error handling and token management
- Update authentication middleware for better compatibility
- Add new package dependencies for enhanced functionality
- Update .gitignore and app configuration

🤖 Generated with [Claude Code](https://claude.ai/code)

Co-Authored-By: Claude <noreply@anthropic.com>
2025-08-09 14:06:56 +08:00

273 lines
10 KiB
Bash
Executable File
Raw Permalink Blame History

This file contains ambiguous Unicode characters

This file contains Unicode characters that might be confused with other characters. If you think that this is intentional, you can safely ignore this warning. Use the Escape button to reveal them.

#!/bin/bash
# Claude Relay Service - 增强版实时监控脚本
# 结合并发监控和系统状态的完整监控方案
# 加载环境变量
if [ -f .env ]; then
export $(grep -v '^#' .env | xargs)
fi
echo "🔍 Claude Relay Service - 增强版实时监控"
echo "按 Ctrl+C 退出 | 按 's' 切换详细/简单模式"
echo "========================================"
# 获取服务配置
SERVICE_HOST=${HOST:-127.0.0.1}
SERVICE_PORT=${PORT:-3000}
# 如果HOST是0.0.0.0客户端应该连接localhost
if [ "$SERVICE_HOST" = "0.0.0.0" ]; then
SERVICE_HOST="127.0.0.1"
fi
SERVICE_URL="http://${SERVICE_HOST}:${SERVICE_PORT}"
# 获取Redis配置
REDIS_HOST=${REDIS_HOST:-127.0.0.1}
REDIS_PORT=${REDIS_PORT:-6379}
REDIS_CMD="redis-cli -h $REDIS_HOST -p $REDIS_PORT"
if [ ! -z "$REDIS_PASSWORD" ]; then
REDIS_CMD="redis-cli -h $REDIS_HOST -p $REDIS_PORT -a $REDIS_PASSWORD"
fi
# 检查Redis连接
if ! $REDIS_CMD ping > /dev/null 2>&1; then
echo "❌ Redis连接失败请检查Redis服务是否运行"
echo " 配置: $REDIS_HOST:$REDIS_PORT"
exit 1
fi
# 显示模式: simple(简单) / detailed(详细)
DISPLAY_MODE="simple"
# 获取API Key详细信息
get_api_key_info() {
local api_key_id=$1
local api_key_name=$($REDIS_CMD hget "apikey:$api_key_id" name 2>/dev/null)
local concurrency_limit=$($REDIS_CMD hget "apikey:$api_key_id" concurrencyLimit 2>/dev/null)
local token_limit=$($REDIS_CMD hget "apikey:$api_key_id" tokenLimit 2>/dev/null)
local created_at=$($REDIS_CMD hget "apikey:$api_key_id" createdAt 2>/dev/null)
if [ -z "$api_key_name" ]; then
api_key_name="Unknown"
fi
if [ -z "$concurrency_limit" ] || [ "$concurrency_limit" = "0" ]; then
concurrency_limit="无限制"
fi
if [ -z "$token_limit" ] || [ "$token_limit" = "0" ]; then
token_limit="无限制"
else
token_limit=$(printf "%'d" $token_limit)
fi
echo "$api_key_name|$concurrency_limit|$token_limit|$created_at"
}
# 获取使用统计信息
get_usage_stats() {
local api_key_id=$1
local today=$(date '+%Y-%m-%d')
local current_month=$(date '+%Y-%m')
# 获取总体使用量
local total_requests=$($REDIS_CMD hget "usage:$api_key_id" totalRequests 2>/dev/null)
local total_tokens=$($REDIS_CMD hget "usage:$api_key_id" totalTokens 2>/dev/null)
# 获取今日使用量
local daily_requests=$($REDIS_CMD hget "usage:daily:$api_key_id:$today" requests 2>/dev/null)
local daily_tokens=$($REDIS_CMD hget "usage:daily:$api_key_id:$today" tokens 2>/dev/null)
total_requests=${total_requests:-0}
total_tokens=${total_tokens:-0}
daily_requests=${daily_requests:-0}
daily_tokens=${daily_tokens:-0}
echo "$total_requests|$total_tokens|$daily_requests|$daily_tokens"
}
# 格式化数字
format_number() {
local num=$1
if [ "$num" -ge 1000000 ]; then
echo "$(echo "scale=1; $num / 1000000" | bc 2>/dev/null)M"
elif [ "$num" -ge 1000 ]; then
echo "$(echo "scale=1; $num / 1000" | bc 2>/dev/null)K"
else
echo "$num"
fi
}
# 获取系统信息
get_system_info() {
# Redis信息
local redis_info=$($REDIS_CMD info server 2>/dev/null)
local redis_memory_info=$($REDIS_CMD info memory 2>/dev/null)
local redis_version=$(echo "$redis_info" | grep redis_version | cut -d: -f2 | tr -d '\r' 2>/dev/null)
local redis_uptime=$(echo "$redis_info" | grep uptime_in_seconds | cut -d: -f2 | tr -d '\r' 2>/dev/null)
local used_memory=$(echo "$redis_memory_info" | grep used_memory_human | cut -d: -f2 | tr -d '\r' 2>/dev/null)
local redis_uptime_hours=0
if [ ! -z "$redis_uptime" ]; then
redis_uptime_hours=$((redis_uptime / 3600))
fi
# 服务状态
local service_status="unknown"
local service_uptime="0"
if command -v curl > /dev/null 2>&1; then
local health_response=$(curl -s ${SERVICE_URL}/health 2>/dev/null)
if [ $? -eq 0 ]; then
service_status=$(echo "$health_response" | grep -o '"status":"[^"]*"' | cut -d'"' -f4 | head -1)
service_uptime=$(echo "$health_response" | grep -o '"uptime":[^,}]*' | cut -d: -f2 | head -1)
fi
fi
local service_uptime_hours="0"
if [ ! -z "$service_uptime" ] && [ "$service_uptime" != "null" ]; then
service_uptime_hours=$(echo "scale=1; $service_uptime / 3600" | bc 2>/dev/null)
fi
echo "$redis_version|$redis_uptime_hours|$used_memory|$service_status|$service_uptime_hours"
}
# 主监控函数
monitor_enhanced() {
while true; do
clear
echo "🔍 Claude Relay Service - 增强版实时监控 | $(date '+%Y-%m-%d %H:%M:%S')"
echo "模式: $DISPLAY_MODE | 服务: $SERVICE_URL | Redis: $REDIS_HOST:$REDIS_PORT"
echo "========================================"
# 获取系统信息
local system_info=$(get_system_info)
local redis_version=$(echo "$system_info" | cut -d'|' -f1)
local redis_uptime=$(echo "$system_info" | cut -d'|' -f2)
local redis_memory=$(echo "$system_info" | cut -d'|' -f3)
local service_status=$(echo "$system_info" | cut -d'|' -f4)
local service_uptime=$(echo "$system_info" | cut -d'|' -f5)
# 系统状态概览
echo "🏥 系统状态概览:"
if [ "$service_status" = "healthy" ]; then
echo " ✅ 服务: 健康 (运行 ${service_uptime}h)"
else
echo " ⚠️ 服务: 异常 ($service_status)"
fi
echo " 📊 Redis: v${redis_version} (运行 ${redis_uptime}h, 内存 ${redis_memory})"
echo ""
# 获取并发信息
local concurrency_keys=$($REDIS_CMD --scan --pattern "concurrency:*" 2>/dev/null)
local total_concurrent=0
local active_keys=0
local concurrent_details=""
if [ ! -z "$concurrency_keys" ]; then
for key in $concurrency_keys; do
local count=$($REDIS_CMD get "$key" 2>/dev/null)
if [ ! -z "$count" ] && [ "$count" -gt 0 ]; then
local api_key_id=${key#concurrency:}
local key_info=$(get_api_key_info "$api_key_id")
local key_name=$(echo "$key_info" | cut -d'|' -f1)
local concurrency_limit=$(echo "$key_info" | cut -d'|' -f2)
concurrent_details="${concurrent_details}${key_name}:${count}/${concurrency_limit} "
total_concurrent=$((total_concurrent + count))
active_keys=$((active_keys + 1))
fi
done
fi
# 并发状态显示
echo "📊 当前并发状态:"
if [ $total_concurrent -eq 0 ]; then
echo " 💤 无活跃并发连接"
else
echo " 🔥 总并发: $total_concurrent 个连接 ($active_keys 个API Key)"
if [ "$DISPLAY_MODE" = "detailed" ]; then
echo " 📋 详情: $concurrent_details"
fi
fi
echo ""
# API Key统计
local total_keys=$($REDIS_CMD keys "apikey:*" 2>/dev/null | grep -v "apikey:hash_map" | wc -l)
local total_accounts=$($REDIS_CMD keys "claude:account:*" 2>/dev/null | wc -l)
echo "📋 资源统计:"
echo " 🔑 API Keys: $total_keys"
echo " 🏢 Claude账户: $total_accounts"
# 详细模式显示更多信息
if [ "$DISPLAY_MODE" = "detailed" ]; then
echo ""
echo "📈 使用统计 (今日/总计)"
# 获取所有API Key
local api_keys=$($REDIS_CMD keys "apikey:*" 2>/dev/null | grep -v "apikey:hash_map")
local total_daily_requests=0
local total_daily_tokens=0
local total_requests=0
local total_tokens=0
if [ ! -z "$api_keys" ]; then
for key in $api_keys; do
local api_key_id=${key#apikey:}
local key_info=$(get_api_key_info "$api_key_id")
local key_name=$(echo "$key_info" | cut -d'|' -f1)
local usage_info=$(get_usage_stats "$api_key_id")
local key_total_requests=$(echo "$usage_info" | cut -d'|' -f1)
local key_total_tokens=$(echo "$usage_info" | cut -d'|' -f2)
local key_daily_requests=$(echo "$usage_info" | cut -d'|' -f3)
local key_daily_tokens=$(echo "$usage_info" | cut -d'|' -f4)
total_daily_requests=$((total_daily_requests + key_daily_requests))
total_daily_tokens=$((total_daily_tokens + key_daily_tokens))
total_requests=$((total_requests + key_total_requests))
total_tokens=$((total_tokens + key_total_tokens))
if [ $((key_daily_requests + key_total_requests)) -gt 0 ]; then
echo " 📱 $key_name: ${key_daily_requests}req/$(format_number $key_daily_tokens) | ${key_total_requests}req/$(format_number $key_total_tokens)"
fi
done
fi
echo " 🌍 系统总计: ${total_daily_requests}req/$(format_number $total_daily_tokens) | ${total_requests}req/$(format_number $total_tokens)"
fi
echo ""
echo "🔄 刷新间隔: 5秒 | 按 Ctrl+C 退出 | 按 Enter 切换详细/简单模式"
# 非阻塞读取用户输入
read -t 5 user_input
if [ $? -eq 0 ]; then
case "$user_input" in
"s"|"S"|"")
if [ "$DISPLAY_MODE" = "simple" ]; then
DISPLAY_MODE="detailed"
else
DISPLAY_MODE="simple"
fi
;;
esac
fi
done
}
# 信号处理
cleanup() {
echo ""
echo "👋 监控已停止"
exit 0
}
trap cleanup SIGINT SIGTERM
# 开始监控
monitor_enhanced