[#24] web和alert的集成测试
This commit is contained in:
parent
59a38513a4
commit
471f3e13f9
@ -1,5 +1,5 @@
|
|||||||
DATA_ROOT=/home/argus/tmp/private/argus
|
DATA_ROOT=/home/argus/tmp/private/argus
|
||||||
ARGUS_UID=1048
|
ARGUS_BUILD_UID=1048
|
||||||
ARGUS_GID=1048
|
ARGUS_BUILD_GID=1048
|
||||||
|
|
||||||
USE_INTRANET=false
|
USE_INTRANET=false
|
||||||
@ -1,19 +0,0 @@
|
|||||||
global:
|
|
||||||
resolve_timeout: 5m
|
|
||||||
|
|
||||||
route:
|
|
||||||
group_by: ['alertname', 'instance'] # 分组:相同 alertname + instance 的告警合并
|
|
||||||
group_wait: 30s # 第一个告警后,等 30s 看是否有同组告警一起发
|
|
||||||
group_interval: 5m # 同组告警变化后,至少 5 分钟再发一次
|
|
||||||
repeat_interval: 3h # 相同告警,3 小时重复提醒一次
|
|
||||||
receiver: 'null'
|
|
||||||
|
|
||||||
receivers:
|
|
||||||
- name: 'null'
|
|
||||||
|
|
||||||
inhibit_rules:
|
|
||||||
- source_match:
|
|
||||||
severity: 'critical' # critical 告警存在时
|
|
||||||
target_match:
|
|
||||||
severity: 'warning' # 抑制相同 instance 的 warning 告警
|
|
||||||
equal: ['instance']
|
|
||||||
@ -1 +0,0 @@
|
|||||||
172.18.0.2
|
|
||||||
@ -1,19 +0,0 @@
|
|||||||
#!/usr/bin/env bash
|
|
||||||
set -euo pipefail
|
|
||||||
root="$(cd "$(dirname "${BASH_SOURCE[0]}")/../" && pwd)"
|
|
||||||
project_root="$(cd "$root/../../.." && pwd)"
|
|
||||||
|
|
||||||
source "$project_root/scripts/common/build_user.sh"
|
|
||||||
load_build_user
|
|
||||||
|
|
||||||
# 创建新的private目录结构 (基于argus目录结构)
|
|
||||||
echo "[INFO] Creating private directory structure for supervisor-based containers..."
|
|
||||||
mkdir -p "$root/private/argus/alert/alertmanager"
|
|
||||||
mkdir -p "$root/private/argus/etc/"
|
|
||||||
|
|
||||||
# 设置数据目录权限
|
|
||||||
echo "[INFO] Setting permissions for data directories..."
|
|
||||||
chown -R "${ARGUS_BUILD_UID}:${ARGUS_BUILD_GID}" "$root/private/argus/alert/alertmanager" 2>/dev/null || true
|
|
||||||
chown -R "${ARGUS_BUILD_UID}:${ARGUS_BUILD_GID}" "$root/private/argus/etc" 2>/dev/null || true
|
|
||||||
|
|
||||||
echo "[INFO] Supervisor-based containers will manage their own scripts and configurations"
|
|
||||||
@ -1,10 +0,0 @@
|
|||||||
#!/usr/bin/env bash
|
|
||||||
set -euo pipefail
|
|
||||||
cd "$(dirname "$0")/.."
|
|
||||||
compose_cmd="docker compose"
|
|
||||||
if ! $compose_cmd version >/dev/null 2>&1; then
|
|
||||||
if command -v docker-compose >/dev/null 2>&1; then compose_cmd="docker-compose"; else
|
|
||||||
echo "需要 Docker Compose,请安装后重试" >&2; exit 1; fi
|
|
||||||
fi
|
|
||||||
$compose_cmd -p alert-mvp up -d --remove-orphans
|
|
||||||
echo "[OK] 服务已启动:Alertmanager http://localhost:9093"
|
|
||||||
@ -1,106 +0,0 @@
|
|||||||
#!/bin/bash
|
|
||||||
set -euo pipefail
|
|
||||||
|
|
||||||
# ==========================================================
|
|
||||||
# Alertmanager 测试脚本
|
|
||||||
# ==========================================================
|
|
||||||
|
|
||||||
ALERTMANAGER_URL="http://localhost:9093"
|
|
||||||
TEST_ALERT_NAME_CRITICAL="NodeDown"
|
|
||||||
TEST_ALERT_NAME_WARNING="HighCPU"
|
|
||||||
TMP_LOG="/tmp/test-alertmanager.log"
|
|
||||||
|
|
||||||
# 等待参数
|
|
||||||
am_wait_attempts=30
|
|
||||||
am_wait_interval=2
|
|
||||||
|
|
||||||
GREEN="\033[1;32m"
|
|
||||||
RED="\033[1;31m"
|
|
||||||
YELLOW="\033[1;33m"
|
|
||||||
RESET="\033[0m"
|
|
||||||
|
|
||||||
# ==========================================================
|
|
||||||
# 函数定义
|
|
||||||
# ==========================================================
|
|
||||||
|
|
||||||
wait_for_alertmanager() {
|
|
||||||
local attempt=1
|
|
||||||
echo "[INFO] 等待 Alertmanager 启动中..."
|
|
||||||
while (( attempt <= am_wait_attempts )); do
|
|
||||||
if curl -fsS "${ALERTMANAGER_URL}/api/v2/status" >/dev/null 2>&1; then
|
|
||||||
echo -e "${GREEN}[OK] Alertmanager 已就绪 (attempt=${attempt}/${am_wait_attempts})${RESET}"
|
|
||||||
return 0
|
|
||||||
fi
|
|
||||||
echo "[..] Alertmanager 尚未就绪 (${attempt}/${am_wait_attempts})"
|
|
||||||
sleep "${am_wait_interval}"
|
|
||||||
(( attempt++ ))
|
|
||||||
done
|
|
||||||
echo -e "${RED}[ERROR] Alertmanager 在 ${am_wait_attempts} 次尝试后仍未就绪${RESET}"
|
|
||||||
return 1
|
|
||||||
}
|
|
||||||
|
|
||||||
log_step() {
|
|
||||||
echo -e "${YELLOW}==== $1 ====${RESET}"
|
|
||||||
}
|
|
||||||
|
|
||||||
# ==========================================================
|
|
||||||
# 主流程
|
|
||||||
# ==========================================================
|
|
||||||
|
|
||||||
log_step "测试 Alertmanager 开始"
|
|
||||||
echo "[INFO] Alertmanager 地址: $ALERTMANAGER_URL"
|
|
||||||
|
|
||||||
# Step 1: 等待 Alertmanager 启动
|
|
||||||
wait_for_alertmanager
|
|
||||||
|
|
||||||
# Step 2: 触发一个critical测试告警
|
|
||||||
echo "[INFO] 发送critical测试告警..."
|
|
||||||
curl -fsS -X POST "${ALERTMANAGER_URL}/api/v2/alerts" \
|
|
||||||
-H "Content-Type: application/json" \
|
|
||||||
-d '[
|
|
||||||
{
|
|
||||||
"labels": {
|
|
||||||
"alertname": "'"${TEST_ALERT_NAME_CRITICAL}"'",
|
|
||||||
"instance": "node-1",
|
|
||||||
"severity": "critical"
|
|
||||||
},
|
|
||||||
"annotations": {
|
|
||||||
"summary": "节点 node-1 宕机"
|
|
||||||
}
|
|
||||||
}
|
|
||||||
]' \
|
|
||||||
-o "$TMP_LOG"
|
|
||||||
|
|
||||||
if [ $? -eq 0 ]; then
|
|
||||||
echo -e "${GREEN}[OK] 已成功发送critical测试告警${RESET}"
|
|
||||||
else
|
|
||||||
echo -e "${RED}[ERROR] critical告警发送失败!${RESET}"
|
|
||||||
cat "$TMP_LOG"
|
|
||||||
exit 1
|
|
||||||
fi
|
|
||||||
|
|
||||||
# Step 3: 触发一个warning测试告警
|
|
||||||
echo "[INFO] 发送warning测试告警..."
|
|
||||||
curl -fsS -X POST "${ALERTMANAGER_URL}/api/v2/alerts" \
|
|
||||||
-H "Content-Type: application/json" \
|
|
||||||
-d '[
|
|
||||||
{
|
|
||||||
"labels": {
|
|
||||||
"alertname": "'"${TEST_ALERT_NAME_WARNING}"'",
|
|
||||||
"instance": "node-1",
|
|
||||||
"severity": "warning"
|
|
||||||
},
|
|
||||||
"annotations": {
|
|
||||||
"summary": "节点 node-1 CPU 使用率过高"
|
|
||||||
}
|
|
||||||
}
|
|
||||||
]' \
|
|
||||||
-o "$TMP_LOG"
|
|
||||||
|
|
||||||
if [ $? -eq 0 ]; then
|
|
||||||
echo -e "${GREEN}[OK] 已成功发送warning测试告警${RESET}"
|
|
||||||
else
|
|
||||||
echo -e "${RED}[ERROR] warning告警发送失败!${RESET}"
|
|
||||||
cat "$TMP_LOG"
|
|
||||||
exit 1
|
|
||||||
fi
|
|
||||||
@ -1,71 +0,0 @@
|
|||||||
#!/bin/bash
|
|
||||||
set -euo pipefail
|
|
||||||
|
|
||||||
# ==========================================================
|
|
||||||
# Alertmanager 测试脚本(含启动等待)
|
|
||||||
# ==========================================================
|
|
||||||
|
|
||||||
ALERTMANAGER_URL="http://localhost:9093"
|
|
||||||
TEST_ALERT_NAME_CRITICAL="NodeDown"
|
|
||||||
TEST_ALERT_NAME_WARNING="HighCPU"
|
|
||||||
TMP_LOG="/tmp/test-alertmanager.log"
|
|
||||||
|
|
||||||
# 等待参数
|
|
||||||
am_wait_attempts=30
|
|
||||||
am_wait_interval=2
|
|
||||||
|
|
||||||
GREEN="\033[1;32m"
|
|
||||||
RED="\033[1;31m"
|
|
||||||
YELLOW="\033[1;33m"
|
|
||||||
RESET="\033[0m"
|
|
||||||
|
|
||||||
# ==========================================================
|
|
||||||
# 函数定义
|
|
||||||
# ==========================================================
|
|
||||||
|
|
||||||
wait_for_alertmanager() {
|
|
||||||
local attempt=1
|
|
||||||
echo "[INFO] 等待 Alertmanager 启动中..."
|
|
||||||
while (( attempt <= am_wait_attempts )); do
|
|
||||||
if curl -fsS "${ALERTMANAGER_URL}/api/v2/status" >/dev/null 2>&1; then
|
|
||||||
echo -e "${GREEN}[OK] Alertmanager 已就绪 (attempt=${attempt}/${am_wait_attempts})${RESET}"
|
|
||||||
return 0
|
|
||||||
fi
|
|
||||||
echo "[..] Alertmanager 尚未就绪 (${attempt}/${am_wait_attempts})"
|
|
||||||
sleep "${am_wait_interval}"
|
|
||||||
(( attempt++ ))
|
|
||||||
done
|
|
||||||
echo -e "${RED}[ERROR] Alertmanager 在 ${am_wait_attempts} 次尝试后仍未就绪${RESET}"
|
|
||||||
return 1
|
|
||||||
}
|
|
||||||
|
|
||||||
log_step() {
|
|
||||||
echo -e "${YELLOW}==== $1 ====${RESET}"
|
|
||||||
}
|
|
||||||
|
|
||||||
# ==========================================================
|
|
||||||
# 主流程
|
|
||||||
# ==========================================================
|
|
||||||
|
|
||||||
log_step "查询 Alertmanager 当前告警列表开始"
|
|
||||||
echo "[INFO] Alertmanager 地址: $ALERTMANAGER_URL"
|
|
||||||
|
|
||||||
# Step 1: 等待 Alertmanager 启动
|
|
||||||
wait_for_alertmanager
|
|
||||||
|
|
||||||
# Step 2: 查询当前告警列表
|
|
||||||
echo "[INFO] 查询当前告警..."
|
|
||||||
sleep 1
|
|
||||||
curl -fsS "${ALERTMANAGER_URL}/api/v2/alerts" | jq '.' || {
|
|
||||||
echo -e "${RED}[WARN] 无法解析返回 JSON,请检查 jq 是否安装${RESET}"
|
|
||||||
curl -s "${ALERTMANAGER_URL}/api/v2/alerts"
|
|
||||||
}
|
|
||||||
|
|
||||||
# Step 3: 检查告警是否包含 NodeDown
|
|
||||||
if curl -fsS "${ALERTMANAGER_URL}/api/v2/alerts" | grep -q "${TEST_ALERT_NAME_CRITICAL}"; then
|
|
||||||
echo -e "${GREEN}✅ 测试通过:Alertmanager 已成功接收告警 ${TEST_ALERT_NAME_CRITICAL}${RESET}"
|
|
||||||
else
|
|
||||||
echo -e "${RED}❌ 测试失败:未检测到告警 ${TEST_ALERT_NAME_CRITICAL}${RESET}"
|
|
||||||
fi
|
|
||||||
|
|
||||||
log_step "测试结束"
|
|
||||||
@ -1,21 +0,0 @@
|
|||||||
#!/usr/bin/env bash
|
|
||||||
set -euo pipefail
|
|
||||||
cd "$(dirname "$0")/.."
|
|
||||||
compose_cmd="docker compose"
|
|
||||||
if ! $compose_cmd version >/dev/null 2>&1; then
|
|
||||||
if command -v docker-compose >/dev/null 2>&1; then compose_cmd="docker-compose"; else
|
|
||||||
echo "需要 Docker Compose,请安装后重试" >&2; exit 1; fi
|
|
||||||
fi
|
|
||||||
$compose_cmd -p alert-mvp down
|
|
||||||
echo "[OK] 已停止所有容器"
|
|
||||||
|
|
||||||
# 清理private目录内容
|
|
||||||
echo "[INFO] 清理private目录内容..."
|
|
||||||
cd "$(dirname "$0")/.."
|
|
||||||
if [ -d "private" ]; then
|
|
||||||
# 删除private目录及其所有内容
|
|
||||||
rm -rf private
|
|
||||||
echo "[OK] 已清理private目录"
|
|
||||||
else
|
|
||||||
echo "[INFO] private目录不存在,无需清理"
|
|
||||||
fi
|
|
||||||
@ -1,105 +0,0 @@
|
|||||||
#!/usr/bin/env bash
|
|
||||||
set -euo pipefail
|
|
||||||
|
|
||||||
echo "======================================="
|
|
||||||
echo "ARGUS Alert System End-to-End Test"
|
|
||||||
echo "======================================="
|
|
||||||
echo ""
|
|
||||||
|
|
||||||
# 记录测试开始时间
|
|
||||||
test_start_time=$(date +%s)
|
|
||||||
|
|
||||||
# 函数:等待服务就绪
|
|
||||||
wait_for_services() {
|
|
||||||
echo "[INFO] Waiting for all services to be ready..."
|
|
||||||
local max_attempts=${SERVICE_WAIT_ATTEMPTS:-120}
|
|
||||||
local attempt=1
|
|
||||||
|
|
||||||
while [ $attempt -le $max_attempts ]; do
|
|
||||||
if curl -fs http://localhost:9093/api/v2/status >/dev/null 2>&1; then
|
|
||||||
echo "[OK] All services are ready!"
|
|
||||||
return 0
|
|
||||||
fi
|
|
||||||
echo " Waiting for services... ($attempt/$max_attempts)"
|
|
||||||
sleep 5
|
|
||||||
((attempt++))
|
|
||||||
done
|
|
||||||
|
|
||||||
echo "[ERROR] Services not ready after $max_attempts attempts"
|
|
||||||
return 1
|
|
||||||
}
|
|
||||||
|
|
||||||
# 函数:显示测试步骤
|
|
||||||
show_step() {
|
|
||||||
echo ""
|
|
||||||
echo "🔄 Step $1: $2"
|
|
||||||
echo "----------------------------------------"
|
|
||||||
}
|
|
||||||
|
|
||||||
# 函数:验证步骤结果
|
|
||||||
verify_step() {
|
|
||||||
if [ $? -eq 0 ]; then
|
|
||||||
echo "✅ $1 - SUCCESS"
|
|
||||||
else
|
|
||||||
echo "❌ $1 - FAILED"
|
|
||||||
exit 1
|
|
||||||
fi
|
|
||||||
}
|
|
||||||
|
|
||||||
# 开始端到端测试
|
|
||||||
show_step "1" "Bootstrap - Initialize environment"
|
|
||||||
./scripts/01_bootstrap.sh
|
|
||||||
verify_step "Bootstrap"
|
|
||||||
|
|
||||||
show_step "2" "Startup - Start all services"
|
|
||||||
./scripts/02_up.sh
|
|
||||||
verify_step "Service startup"
|
|
||||||
|
|
||||||
# 等待服务完全就绪
|
|
||||||
wait_for_services || exit 1
|
|
||||||
|
|
||||||
# 发送告警数据
|
|
||||||
show_step "3" "Add alerts - Send test alerts to Alertmanager"
|
|
||||||
./scripts/03_alertmanager_add_alert.sh
|
|
||||||
verify_step "Send test alerts"
|
|
||||||
|
|
||||||
# 查询告警数据
|
|
||||||
show_step "4" "Verify data - Query Alertmanager"
|
|
||||||
./scripts/04_query_alerts.sh
|
|
||||||
verify_step "Data verification"
|
|
||||||
|
|
||||||
|
|
||||||
# 检查服务健康状态
|
|
||||||
show_step "Health" "Check service health"
|
|
||||||
echo "[INFO] Checking service health..."
|
|
||||||
|
|
||||||
# 检查 Alertmanager 状态
|
|
||||||
if curl -fs "http://localhost:9093/api/v2/status" >/dev/null 2>&1; then
|
|
||||||
am_status="available"
|
|
||||||
echo "✅ Alertmanager status: $am_status"
|
|
||||||
else
|
|
||||||
am_status="unavailable"
|
|
||||||
echo "⚠️ Alertmanager status: $am_status"
|
|
||||||
fi
|
|
||||||
verify_step "Service health check"
|
|
||||||
|
|
||||||
# 清理环境
|
|
||||||
show_step "5" "Cleanup - Stop all services"
|
|
||||||
./scripts/05_down.sh
|
|
||||||
verify_step "Service cleanup"
|
|
||||||
|
|
||||||
# 计算总测试时间
|
|
||||||
test_end_time=$(date +%s)
|
|
||||||
total_time=$((test_end_time - test_start_time))
|
|
||||||
|
|
||||||
echo ""
|
|
||||||
echo "======================================="
|
|
||||||
echo "🎉 END-TO-END TEST COMPLETED SUCCESSFULLY!"
|
|
||||||
echo "======================================="
|
|
||||||
echo "📊 Test Summary:"
|
|
||||||
echo " • Total time: ${total_time}s"
|
|
||||||
echo " • Alertmanager status: $am_status"
|
|
||||||
echo " • All services started and stopped successfully"
|
|
||||||
echo ""
|
|
||||||
echo "✅ The ARGUS Alert system is working correctly!"
|
|
||||||
echo ""
|
|
||||||
113
src/alert/tests/scripts/verify_alertmanager.sh
Normal file
113
src/alert/tests/scripts/verify_alertmanager.sh
Normal file
@ -0,0 +1,113 @@
|
|||||||
|
#!/bin/bash
|
||||||
|
# verify_alertmanager.sh
|
||||||
|
# 用于部署后验证 Prometheus 与 Alertmanager 通信链路是否正常
|
||||||
|
|
||||||
|
set -euo pipefail
|
||||||
|
|
||||||
|
#=============================
|
||||||
|
# 基础配置
|
||||||
|
#=============================
|
||||||
|
PROM_URL="${PROM_URL:-http://prom.metric.argus.com:9090}"
|
||||||
|
ALERT_URL="${ALERT_URL:-http://alertmanager.alert.argus.com:9093}"
|
||||||
|
# TODO: 根据实际部署环境调整规则目录
|
||||||
|
DATA_ROOT="${DATA_ROOT:-/private/argus}"
|
||||||
|
RULE_DIR = "$DATA_ROOT/metric/prometheus/rules"
|
||||||
|
TMP_RULE="/tmp/test_rule.yml"
|
||||||
|
|
||||||
|
#=============================
|
||||||
|
# 辅助函数
|
||||||
|
#=============================
|
||||||
|
GREEN="\033[32m"; RED="\033[31m"; YELLOW="\033[33m"; RESET="\033[0m"
|
||||||
|
|
||||||
|
log_info() { echo -e "${YELLOW}[INFO]${RESET} $1"; }
|
||||||
|
log_success() { echo -e "${GREEN}[OK]${RESET} $1"; }
|
||||||
|
log_error() { echo -e "${RED}[ERROR]${RESET} $1"; }
|
||||||
|
|
||||||
|
fail_exit() { log_error "$1"; exit 1; }
|
||||||
|
|
||||||
|
#=============================
|
||||||
|
# Step 1: 检查 Alertmanager 是否可访问
|
||||||
|
#=============================
|
||||||
|
log_info "检查 Alertmanager 状态..."
|
||||||
|
if curl -sSf "${ALERT_URL}/api/v2/status" >/dev/null 2>&1; then
|
||||||
|
log_success "Alertmanager 服务正常 (${ALERT_URL})"
|
||||||
|
else
|
||||||
|
fail_exit "无法访问 Alertmanager,请检查端口映射与容器状态。"
|
||||||
|
fi
|
||||||
|
|
||||||
|
#=============================
|
||||||
|
# Step 2: 手动发送测试告警
|
||||||
|
#=============================
|
||||||
|
log_info "发送手动测试告警..."
|
||||||
|
curl -s -XPOST "${ALERT_URL}/api/v2/alerts" -H "Content-Type: application/json" -d '[
|
||||||
|
{
|
||||||
|
"labels": {
|
||||||
|
"alertname": "ManualTestAlert",
|
||||||
|
"severity": "info"
|
||||||
|
},
|
||||||
|
"annotations": {
|
||||||
|
"summary": "This is a test alert from deploy verification"
|
||||||
|
},
|
||||||
|
"startsAt": "'$(date -Iseconds)'"
|
||||||
|
}
|
||||||
|
]' >/dev/null && log_success "测试告警已成功发送到 Alertmanager"
|
||||||
|
|
||||||
|
#=============================
|
||||||
|
# Step 3: 检查 Prometheus 配置中是否包含 Alertmanager
|
||||||
|
#=============================
|
||||||
|
log_info "检查 Prometheus 是否配置了 Alertmanager..."
|
||||||
|
if curl -s "${PROM_URL}/api/v1/status/config" | grep -q "alertmanagers"; then
|
||||||
|
log_success "Prometheus 已配置 Alertmanager 目标"
|
||||||
|
else
|
||||||
|
fail_exit "Prometheus 未配置 Alertmanager,请检查 prometheus.yml"
|
||||||
|
fi
|
||||||
|
|
||||||
|
#=============================
|
||||||
|
# Step 4: 创建并加载测试告警规则
|
||||||
|
#=============================
|
||||||
|
log_info "创建临时测试规则 ${TMP_RULE} ..."
|
||||||
|
cat <<EOF > "${TMP_RULE}"
|
||||||
|
groups:
|
||||||
|
- name: deploy-verify-group
|
||||||
|
rules:
|
||||||
|
- alert: DeployVerifyAlert
|
||||||
|
expr: vector(1)
|
||||||
|
labels:
|
||||||
|
severity: warning
|
||||||
|
annotations:
|
||||||
|
summary: "Deployment verification alert"
|
||||||
|
EOF
|
||||||
|
|
||||||
|
mkdir -p "${RULE_DIR}"
|
||||||
|
cp "${TMP_RULE}" "${RULE_DIR}/test_rule.yml"
|
||||||
|
|
||||||
|
log_info "重载 Prometheus 以加载新规则..."
|
||||||
|
if curl -s -X POST "${PROM_URL}/-/reload" >/dev/null; then
|
||||||
|
log_success "Prometheus 已重载规则"
|
||||||
|
else
|
||||||
|
fail_exit "Prometheus reload 失败,请检查 API 可访问性。"
|
||||||
|
fi
|
||||||
|
|
||||||
|
#=============================
|
||||||
|
# Step 5: 等待并验证 Alertmanager 是否收到告警
|
||||||
|
#=============================
|
||||||
|
log_info "等待告警触发 (约5秒)..."
|
||||||
|
sleep 5
|
||||||
|
|
||||||
|
if curl -s "${ALERT_URL}/api/v2/alerts" | grep -q "DeployVerifyAlert"; then
|
||||||
|
log_success "Prometheus → Alertmanager 告警链路验证成功"
|
||||||
|
else
|
||||||
|
fail_exit "未在 Alertmanager 中检测到 DeployVerifyAlert,请检查网络或配置。"
|
||||||
|
fi
|
||||||
|
|
||||||
|
#=============================
|
||||||
|
# Step 6: 清理测试规则
|
||||||
|
#=============================
|
||||||
|
log_info "清理临时测试规则..."
|
||||||
|
rm -f "${RULE_DIR}/test_rule.yml" "${TMP_RULE}"
|
||||||
|
|
||||||
|
curl -s -X POST "${PROM_URL}/-/reload" >/dev/null \
|
||||||
|
&& log_success "Prometheus 已清理验证规则" \
|
||||||
|
|| log_error "Prometheus reload 清理失败,请手动确认。"
|
||||||
|
|
||||||
|
log_success "部署验证全部通过!Prometheus ↔ Alertmanager 通信正常。"
|
||||||
@ -47,6 +47,8 @@
|
|||||||
- `./scripts/11_metric_node_install.sh` 在 CPU 节点安装并验证端点
|
- `./scripts/11_metric_node_install.sh` 在 CPU 节点安装并验证端点
|
||||||
- `./scripts/12_metric_gpu_install.sh` 在 GPU 节点安装并等待 9100/9400 就绪(仅启用 GPU 时)
|
- `./scripts/12_metric_gpu_install.sh` 在 GPU 节点安装并等待 9100/9400 就绪(仅启用 GPU 时)
|
||||||
- `./scripts/13_metric_verify.sh` 对 master/Prometheus/数据面/Grafana 做综合校验(含 GPU 时校验 dcgm 指标)
|
- `./scripts/13_metric_verify.sh` 对 master/Prometheus/数据面/Grafana 做综合校验(含 GPU 时校验 dcgm 指标)
|
||||||
|
- `./scripts/15_alert_verify.sh` 对alertmanager进行校验
|
||||||
|
- `./scripts/16_web_verify.sh` 对web页面进行校验综合校验。
|
||||||
- `./scripts/14_metric_cleanup.sh` 清理 FTP 产物
|
- `./scripts/14_metric_cleanup.sh` 清理 FTP 产物
|
||||||
- `./scripts/09_down.sh` 回收容器、网络并清理 `private*/`、`tmp/`
|
- `./scripts/09_down.sh` 回收容器、网络并清理 `private*/`、`tmp/`
|
||||||
|
|
||||||
@ -133,6 +135,13 @@
|
|||||||
- `09_down.sh`
|
- `09_down.sh`
|
||||||
- 目的:栈销毁与环境清理;必要时使用临时容器修正属主再删除 `private*` 目录
|
- 目的:栈销毁与环境清理;必要时使用临时容器修正属主再删除 `private*` 目录
|
||||||
|
|
||||||
|
- `15_alert_verify.sh`
|
||||||
|
- 目的:验证alertmanager的可用性、Prometheus到alertmanager的连通性。
|
||||||
|
- 操作:在Prometheus中增加一个恒为真的告警规则,查看alertmanager是否收到该告警
|
||||||
|
- `16_web_verify.sh`
|
||||||
|
- 目的:验证web页面是否可用。
|
||||||
|
- 使用playwright分别验证各个模块的页面是否可用,以及符合预期。
|
||||||
|
|
||||||
---
|
---
|
||||||
|
|
||||||
### 常见问题与排查
|
### 常见问题与排查
|
||||||
|
|||||||
@ -55,6 +55,8 @@ SCRIPTS=(
|
|||||||
"11_metric_node_install.sh"
|
"11_metric_node_install.sh"
|
||||||
"12_metric_gpu_install.sh"
|
"12_metric_gpu_install.sh"
|
||||||
"13_metric_verify.sh"
|
"13_metric_verify.sh"
|
||||||
|
"15_alert_verify.sh"
|
||||||
|
"16_web_verify.sh"
|
||||||
)
|
)
|
||||||
|
|
||||||
# 如未禁用清理,则追加清理与下线步骤(保持原有顺序)
|
# 如未禁用清理,则追加清理与下线步骤(保持原有顺序)
|
||||||
|
|||||||
103
src/sys/tests/scripts/15_alert_verify.sh
Normal file
103
src/sys/tests/scripts/15_alert_verify.sh
Normal file
@ -0,0 +1,103 @@
|
|||||||
|
#!/bin/bash
|
||||||
|
# verify_alertmanager.sh
|
||||||
|
# Verify the communication between Prometheus and Alertmanager after deployment
|
||||||
|
|
||||||
|
set -euo pipefail
|
||||||
|
|
||||||
|
echo "[INFO] Verifying Prometheus ↔ Alertmanager communication..."
|
||||||
|
|
||||||
|
SCRIPT_DIR="$(cd "$(dirname "${BASH_SOURCE[0]}")" && pwd)"
|
||||||
|
TEST_ROOT="$(cd "$SCRIPT_DIR/.." && pwd)"
|
||||||
|
TMP_DIR="$TEST_ROOT/tmp"
|
||||||
|
mkdir -p "$TMP_DIR"
|
||||||
|
|
||||||
|
PRIVATE_CORE="$TEST_ROOT/private"
|
||||||
|
|
||||||
|
#=============================
|
||||||
|
# Load environment variables
|
||||||
|
#=============================
|
||||||
|
if [[ -f "$TEST_ROOT/.env" ]]; then
|
||||||
|
set -a; source "$TEST_ROOT/.env"; set +a
|
||||||
|
fi
|
||||||
|
|
||||||
|
#=============================
|
||||||
|
# Basic configuration
|
||||||
|
#=============================
|
||||||
|
PROM_URL="http://localhost:${PROMETHEUS_PORT:-9090}"
|
||||||
|
ALERT_URL="http://localhost:${ALERTMANAGER_PORT:-9093}"
|
||||||
|
RULE_DIR="$PRIVATE_CORE/argus/metric/prometheus/rules"
|
||||||
|
TMP_RULE="$TMP_DIR/test_rule.yml"
|
||||||
|
|
||||||
|
#=============================
|
||||||
|
# Helper functions
|
||||||
|
#=============================
|
||||||
|
GREEN="\033[32m"; RED="\033[31m"; YELLOW="\033[33m"; RESET="\033[0m"
|
||||||
|
|
||||||
|
log_info() { echo -e "${YELLOW}[INFO]${RESET} $1"; }
|
||||||
|
log_success() { echo -e "${GREEN}[OK]${RESET} $1"; }
|
||||||
|
log_warn() { echo -e "${YELLOW}[WARN]${RESET} $1"; }
|
||||||
|
log_error() { echo -e "${RED}[ERROR]${RESET} $1"; }
|
||||||
|
|
||||||
|
fail_exit() { log_error "$1"; exit 1; }
|
||||||
|
|
||||||
|
#=============================
|
||||||
|
# Step 1: Check Alertmanager accessibility
|
||||||
|
#=============================
|
||||||
|
log_info "Checking Alertmanager status..."
|
||||||
|
if curl -sSf "${ALERT_URL}/api/v2/status" >/dev/null 2>&1; then
|
||||||
|
log_success "Alertmanager is reachable at ${ALERT_URL}"
|
||||||
|
else
|
||||||
|
fail_exit "Alertmanager is not reachable. Please check container or port mapping."
|
||||||
|
fi
|
||||||
|
|
||||||
|
#=============================
|
||||||
|
# Step 2: Create and load a temporary test alert rule
|
||||||
|
#=============================
|
||||||
|
log_info "Creating temporary alert rule at ${TMP_RULE}..."
|
||||||
|
cat <<EOF > "${TMP_RULE}"
|
||||||
|
groups:
|
||||||
|
- name: deploy-verify-group
|
||||||
|
rules:
|
||||||
|
- alert: DeployVerifyAlert
|
||||||
|
expr: vector(1)
|
||||||
|
labels:
|
||||||
|
severity: warning
|
||||||
|
annotations:
|
||||||
|
summary: "Deployment verification alert"
|
||||||
|
EOF
|
||||||
|
|
||||||
|
mkdir -p "${RULE_DIR}"
|
||||||
|
cp "${TMP_RULE}" "${RULE_DIR}/test_rule.yml"
|
||||||
|
|
||||||
|
log_info "Reloading Prometheus to apply the test rule..."
|
||||||
|
if curl -s -X POST "${PROM_URL}/-/reload" >/dev/null; then
|
||||||
|
log_success "Prometheus successfully reloaded rules"
|
||||||
|
else
|
||||||
|
fail_exit "Failed to reload Prometheus. Check API accessibility."
|
||||||
|
fi
|
||||||
|
|
||||||
|
#=============================
|
||||||
|
# Step 3: Verify alert received by Alertmanager
|
||||||
|
#=============================
|
||||||
|
log_info "Waiting for alert propagation (~5 seconds)..."
|
||||||
|
sleep 5
|
||||||
|
|
||||||
|
if curl -s "${ALERT_URL}/api/v2/alerts" | grep -q "DeployVerifyAlert"; then
|
||||||
|
log_success "Prometheus → Alertmanager alert path verified successfully"
|
||||||
|
else
|
||||||
|
fail_exit "DeployVerifyAlert not found in Alertmanager. Check configuration or network."
|
||||||
|
fi
|
||||||
|
|
||||||
|
#=============================
|
||||||
|
# Step 4: Cleanup test rule
|
||||||
|
#=============================
|
||||||
|
log_info "Cleaning up temporary alert rule..."
|
||||||
|
rm -f "${RULE_DIR}/test_rule.yml" "${TMP_RULE}"
|
||||||
|
|
||||||
|
if curl -s -X POST "${PROM_URL}/-/reload" >/dev/null; then
|
||||||
|
log_success "Prometheus successfully reloaded after cleanup"
|
||||||
|
else
|
||||||
|
log_warn "Prometheus reload after cleanup failed. Please check manually."
|
||||||
|
fi
|
||||||
|
|
||||||
|
log_success "Alertmanager verification completed successfully. Communication with Prometheus is healthy."
|
||||||
100
src/sys/tests/scripts/16_web_verify.sh
Normal file
100
src/sys/tests/scripts/16_web_verify.sh
Normal file
@ -0,0 +1,100 @@
|
|||||||
|
#!/usr/bin/env bash
|
||||||
|
# verify-web-test.sh
|
||||||
|
# Verify frontend service availability and run Playwright end-to-end tests
|
||||||
|
|
||||||
|
set -euo pipefail
|
||||||
|
|
||||||
|
echo '[INFO] Verifying Web frontend...'
|
||||||
|
|
||||||
|
SCRIPT_DIR="$(cd "$(dirname "${BASH_SOURCE[0]}")" && pwd)"
|
||||||
|
TEST_ROOT="$(cd "$SCRIPT_DIR/.." && pwd)"
|
||||||
|
REPO_ROOT="$(cd "$TEST_ROOT/../../.." && pwd)"
|
||||||
|
WEB_DIR="$REPO_ROOT/src/web"
|
||||||
|
|
||||||
|
#=============================
|
||||||
|
# Load environment variables
|
||||||
|
#=============================
|
||||||
|
if [[ -f "$TEST_ROOT/.env" ]]; then
|
||||||
|
set -a; source "$TEST_ROOT/.env"; set +a
|
||||||
|
fi
|
||||||
|
|
||||||
|
REPORT_DIR="$WEB_DIR/playwright-report"
|
||||||
|
FRONTEND_URL="http://localhost:${WEB_PROXY_PORT_8080:-8080}"
|
||||||
|
TIMEOUT=120 # max wait time (seconds) for frontend to be ready
|
||||||
|
|
||||||
|
#=============================
|
||||||
|
# Helper functions
|
||||||
|
#=============================
|
||||||
|
GREEN="\033[32m"; RED="\033[31m"; YELLOW="\033[33m"; RESET="\033[0m"
|
||||||
|
|
||||||
|
log_info() { echo -e "${YELLOW}[INFO]${RESET} $1"; }
|
||||||
|
log_success() { echo -e "${GREEN}[OK]${RESET} $1"; }
|
||||||
|
log_warn() { echo -e "${YELLOW}[WARN]${RESET} $1"; }
|
||||||
|
log_error() { echo -e "${RED}[ERROR]${RESET} $1"; }
|
||||||
|
|
||||||
|
fail_exit() { log_error "$1"; exit 1; }
|
||||||
|
|
||||||
|
#=============================
|
||||||
|
# Step 1: Wait for frontend service
|
||||||
|
#=============================
|
||||||
|
log_info "[1/4] Checking if frontend service is up (${FRONTEND_URL})..."
|
||||||
|
|
||||||
|
for ((i=1; i<=TIMEOUT; i++)); do
|
||||||
|
STATUS_CODE=$(curl -s -o /dev/null -w "%{http_code}" "$FRONTEND_URL" || true)
|
||||||
|
if [[ "$STATUS_CODE" == "200" ]]; then
|
||||||
|
log_success "Frontend service is accessible at ${FRONTEND_URL}"
|
||||||
|
break
|
||||||
|
fi
|
||||||
|
sleep 2
|
||||||
|
if [[ $i -eq $TIMEOUT ]]; then
|
||||||
|
fail_exit "Timeout waiting for frontend service to become ready (${TIMEOUT}s)."
|
||||||
|
fi
|
||||||
|
done
|
||||||
|
|
||||||
|
#=============================
|
||||||
|
# Step 2: Run Playwright tests
|
||||||
|
#=============================
|
||||||
|
log_info "[2/4] Running Playwright automated tests..."
|
||||||
|
|
||||||
|
cd "$WEB_DIR"
|
||||||
|
|
||||||
|
# Ensure dependencies installed
|
||||||
|
if [ ! -d "node_modules" ]; then
|
||||||
|
log_warn "Dependencies not found. Installing via npm ci..."
|
||||||
|
npm ci
|
||||||
|
fi
|
||||||
|
|
||||||
|
# Clean previous reports
|
||||||
|
rm -rf "$REPORT_DIR"
|
||||||
|
|
||||||
|
# Run Playwright tests with reporters
|
||||||
|
set +e # temporarily disable exit-on-error to capture test result
|
||||||
|
npx playwright test tests/playwright --reporter=list,html
|
||||||
|
TEST_RESULT=$?
|
||||||
|
set -e # re-enable strict mode
|
||||||
|
|
||||||
|
#=============================
|
||||||
|
# Step 3: Check test results
|
||||||
|
#=============================
|
||||||
|
log_info "[3/4] Checking test results..."
|
||||||
|
|
||||||
|
if [[ $TEST_RESULT -eq 0 ]]; then
|
||||||
|
log_success "All Playwright tests passed successfully."
|
||||||
|
else
|
||||||
|
log_error "Some Playwright tests failed. Please review the test report."
|
||||||
|
fi
|
||||||
|
|
||||||
|
#=============================
|
||||||
|
# Step 4: Report generation
|
||||||
|
#=============================
|
||||||
|
log_info "[4/4] Checking Playwright report..."
|
||||||
|
|
||||||
|
if [[ -d "$REPORT_DIR" ]]; then
|
||||||
|
log_success "Test report generated at: $REPORT_DIR"
|
||||||
|
echo "You can view it using:"
|
||||||
|
echo " npx playwright show-report"
|
||||||
|
else
|
||||||
|
log_warn "Report directory not found. Check Playwright execution logs."
|
||||||
|
fi
|
||||||
|
|
||||||
|
log_success "Web frontend verify success. Playwright automated tests passed."
|
||||||
3
src/web/.gitignore
vendored
3
src/web/.gitignore
vendored
@ -1,6 +1,9 @@
|
|||||||
# Node modules
|
# Node modules
|
||||||
node_modules/
|
node_modules/
|
||||||
|
|
||||||
|
# playwright report
|
||||||
|
playwright-report/
|
||||||
|
|
||||||
# Build output
|
# Build output
|
||||||
/dist
|
/dist
|
||||||
/build
|
/build
|
||||||
|
|||||||
64
src/web/package-lock.json
generated
64
src/web/package-lock.json
generated
@ -20,6 +20,7 @@
|
|||||||
},
|
},
|
||||||
"devDependencies": {
|
"devDependencies": {
|
||||||
"@eslint/js": "^9.33.0",
|
"@eslint/js": "^9.33.0",
|
||||||
|
"@playwright/test": "^1.56.1",
|
||||||
"@types/react": "^19.1.10",
|
"@types/react": "^19.1.10",
|
||||||
"@types/react-dom": "^19.1.7",
|
"@types/react-dom": "^19.1.7",
|
||||||
"@vitejs/plugin-react": "^5.0.0",
|
"@vitejs/plugin-react": "^5.0.0",
|
||||||
@ -1231,6 +1232,22 @@
|
|||||||
"react": "^18.x || ^19.x"
|
"react": "^18.x || ^19.x"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
"node_modules/@playwright/test": {
|
||||||
|
"version": "1.56.1",
|
||||||
|
"resolved": "https://registry.npmjs.org/@playwright/test/-/test-1.56.1.tgz",
|
||||||
|
"integrity": "sha512-vSMYtL/zOcFpvJCW71Q/OEGQb7KYBPAdKh35WNSkaZA75JlAO8ED8UN6GUNTm3drWomcbcqRPFqQbLae8yBTdg==",
|
||||||
|
"dev": true,
|
||||||
|
"license": "Apache-2.0",
|
||||||
|
"dependencies": {
|
||||||
|
"playwright": "1.56.1"
|
||||||
|
},
|
||||||
|
"bin": {
|
||||||
|
"playwright": "cli.js"
|
||||||
|
},
|
||||||
|
"engines": {
|
||||||
|
"node": ">=18"
|
||||||
|
}
|
||||||
|
},
|
||||||
"node_modules/@rolldown/pluginutils": {
|
"node_modules/@rolldown/pluginutils": {
|
||||||
"version": "1.0.0-beta.34",
|
"version": "1.0.0-beta.34",
|
||||||
"resolved": "https://registry.npmjs.org/@rolldown/pluginutils/-/pluginutils-1.0.0-beta.34.tgz",
|
"resolved": "https://registry.npmjs.org/@rolldown/pluginutils/-/pluginutils-1.0.0-beta.34.tgz",
|
||||||
@ -2860,6 +2877,53 @@
|
|||||||
"url": "https://github.com/sponsors/jonschlinkert"
|
"url": "https://github.com/sponsors/jonschlinkert"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
"node_modules/playwright": {
|
||||||
|
"version": "1.56.1",
|
||||||
|
"resolved": "https://registry.npmjs.org/playwright/-/playwright-1.56.1.tgz",
|
||||||
|
"integrity": "sha512-aFi5B0WovBHTEvpM3DzXTUaeN6eN0qWnTkKx4NQaH4Wvcmc153PdaY2UBdSYKaGYw+UyWXSVyxDUg5DoPEttjw==",
|
||||||
|
"dev": true,
|
||||||
|
"license": "Apache-2.0",
|
||||||
|
"dependencies": {
|
||||||
|
"playwright-core": "1.56.1"
|
||||||
|
},
|
||||||
|
"bin": {
|
||||||
|
"playwright": "cli.js"
|
||||||
|
},
|
||||||
|
"engines": {
|
||||||
|
"node": ">=18"
|
||||||
|
},
|
||||||
|
"optionalDependencies": {
|
||||||
|
"fsevents": "2.3.2"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"node_modules/playwright-core": {
|
||||||
|
"version": "1.56.1",
|
||||||
|
"resolved": "https://registry.npmjs.org/playwright-core/-/playwright-core-1.56.1.tgz",
|
||||||
|
"integrity": "sha512-hutraynyn31F+Bifme+Ps9Vq59hKuUCz7H1kDOcBs+2oGguKkWTU50bBWrtz34OUWmIwpBTWDxaRPXrIXkgvmQ==",
|
||||||
|
"dev": true,
|
||||||
|
"license": "Apache-2.0",
|
||||||
|
"bin": {
|
||||||
|
"playwright-core": "cli.js"
|
||||||
|
},
|
||||||
|
"engines": {
|
||||||
|
"node": ">=18"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"node_modules/playwright/node_modules/fsevents": {
|
||||||
|
"version": "2.3.2",
|
||||||
|
"resolved": "https://registry.npmjs.org/fsevents/-/fsevents-2.3.2.tgz",
|
||||||
|
"integrity": "sha512-xiqMQR4xAeHTuB9uWm+fFRcIOgKBMiOBP+eXiyT7jsgVCq1bkVygt00oASowB7EdtpOHaaPgKt812P9ab+DDKA==",
|
||||||
|
"dev": true,
|
||||||
|
"hasInstallScript": true,
|
||||||
|
"license": "MIT",
|
||||||
|
"optional": true,
|
||||||
|
"os": [
|
||||||
|
"darwin"
|
||||||
|
],
|
||||||
|
"engines": {
|
||||||
|
"node": "^8.16.0 || ^10.6.0 || >=11.0.0"
|
||||||
|
}
|
||||||
|
},
|
||||||
"node_modules/postcss": {
|
"node_modules/postcss": {
|
||||||
"version": "8.5.6",
|
"version": "8.5.6",
|
||||||
"resolved": "https://registry.npmjs.org/postcss/-/postcss-8.5.6.tgz",
|
"resolved": "https://registry.npmjs.org/postcss/-/postcss-8.5.6.tgz",
|
||||||
|
|||||||
@ -7,7 +7,9 @@
|
|||||||
"dev": "vite",
|
"dev": "vite",
|
||||||
"build": "vite build",
|
"build": "vite build",
|
||||||
"lint": "eslint .",
|
"lint": "eslint .",
|
||||||
"preview": "vite preview"
|
"preview": "vite preview",
|
||||||
|
"test:web": "playwright test",
|
||||||
|
"test:web:report": "playwright show-report"
|
||||||
},
|
},
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
"@emotion/react": "^11.14.0",
|
"@emotion/react": "^11.14.0",
|
||||||
@ -22,6 +24,7 @@
|
|||||||
},
|
},
|
||||||
"devDependencies": {
|
"devDependencies": {
|
||||||
"@eslint/js": "^9.33.0",
|
"@eslint/js": "^9.33.0",
|
||||||
|
"@playwright/test": "^1.56.1",
|
||||||
"@types/react": "^19.1.10",
|
"@types/react": "^19.1.10",
|
||||||
"@types/react-dom": "^19.1.7",
|
"@types/react-dom": "^19.1.7",
|
||||||
"@vitejs/plugin-react": "^5.0.0",
|
"@vitejs/plugin-react": "^5.0.0",
|
||||||
|
|||||||
19
src/web/playwright.config.ts
Normal file
19
src/web/playwright.config.ts
Normal file
@ -0,0 +1,19 @@
|
|||||||
|
import { defineConfig } from '@playwright/test';
|
||||||
|
|
||||||
|
export default defineConfig({
|
||||||
|
testDir: './tests',
|
||||||
|
testIgnore: ['**/src/assets/**', '**/*.png', '**/*.jpg', '**/*.svg'],
|
||||||
|
timeout: 30 * 1000,
|
||||||
|
retries: 1,
|
||||||
|
use: {
|
||||||
|
headless: true,
|
||||||
|
viewport: { width: 1280, height: 720 },
|
||||||
|
ignoreHTTPSErrors: true,
|
||||||
|
screenshot: 'only-on-failure',
|
||||||
|
video: 'retain-on-failure'
|
||||||
|
},
|
||||||
|
reporter: [
|
||||||
|
['list'],
|
||||||
|
['html', { open: 'never', outputFolder: 'playwright-report' }]
|
||||||
|
]
|
||||||
|
});
|
||||||
@ -1,6 +1,5 @@
|
|||||||
import grafanaLogo from "../assets/grafana.png";
|
import grafanaLogo from "../assets/grafana.png";
|
||||||
import prometheusLogo from "../assets/prometheus.png";
|
import prometheusLogo from "../assets/prometheus.png";
|
||||||
import esLogo from "../assets/es.png";
|
|
||||||
import kibanaLogo from "../assets/kibana.png";
|
import kibanaLogo from "../assets/kibana.png";
|
||||||
import { EXTERNAL_HOST } from "./api";
|
import { EXTERNAL_HOST } from "./api";
|
||||||
|
|
||||||
|
|||||||
@ -1 +0,0 @@
|
|||||||
172.18.0.3
|
|
||||||
77
src/web/tests/playwright/alerts.spec.ts
Normal file
77
src/web/tests/playwright/alerts.spec.ts
Normal file
@ -0,0 +1,77 @@
|
|||||||
|
import { test, expect } from "@playwright/test";
|
||||||
|
|
||||||
|
test.describe("Alerts 页面功能测试", () => {
|
||||||
|
test.beforeEach(async ({ page }) => {
|
||||||
|
await page.goto("http://web.argus.com/alerts"); // 根据你实际路由调整
|
||||||
|
});
|
||||||
|
|
||||||
|
test("页面加载并显示告警统计", async ({ page }) => {
|
||||||
|
await expect(page.locator("text=告警详情")).toBeVisible();
|
||||||
|
await expect(page.locator("text=总数")).toBeVisible();
|
||||||
|
await expect(page.locator("text=严重")).toBeVisible();
|
||||||
|
await expect(page.locator("text=警告")).toBeVisible();
|
||||||
|
await expect(page.locator("text=信息")).toBeVisible();
|
||||||
|
});
|
||||||
|
|
||||||
|
test("筛选功能验证", async ({ page }) => {
|
||||||
|
const severitySelect = page.getByLabel("严重性");
|
||||||
|
const stateSelect = page.getByLabel("状态");
|
||||||
|
const nodeSelect = page.getByLabel("节点");
|
||||||
|
|
||||||
|
await severitySelect.selectOption("critical");
|
||||||
|
await expect(severitySelect).toHaveValue("critical");
|
||||||
|
|
||||||
|
await stateSelect.selectOption("active");
|
||||||
|
await expect(stateSelect).toHaveValue("active");
|
||||||
|
|
||||||
|
await nodeSelect.selectOption("all");
|
||||||
|
await expect(nodeSelect).toHaveValue("all");
|
||||||
|
});
|
||||||
|
|
||||||
|
test("排序功能", async ({ page }) => {
|
||||||
|
const severityHeader = page.locator("th:has-text('严重性') button");
|
||||||
|
await severityHeader.click(); // 切换升序
|
||||||
|
await severityHeader.click(); // 切换降序
|
||||||
|
|
||||||
|
const instanceHeader = page.locator("th:has-text('节点') button");
|
||||||
|
await instanceHeader.click();
|
||||||
|
await instanceHeader.click();
|
||||||
|
});
|
||||||
|
|
||||||
|
test("分页功能", async ({ page }) => {
|
||||||
|
const nextButton = page.locator("button:has-text('下一页')");
|
||||||
|
const prevButton = page.locator("button:has-text('上一页')");
|
||||||
|
|
||||||
|
if (await nextButton.isEnabled()) {
|
||||||
|
await nextButton.click();
|
||||||
|
await expect(prevButton).toBeEnabled();
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
test("展开更多信息行", async ({ page }) => {
|
||||||
|
const infoIcons = page.locator("table tbody tr td [title='显示/隐藏更多信息']");
|
||||||
|
if (await infoIcons.count() > 0) {
|
||||||
|
await infoIcons.first().click();
|
||||||
|
// 展开的详情行应出现
|
||||||
|
const details = page.locator("table tbody tr >> text=alertname");
|
||||||
|
const detailCount = await details.count();
|
||||||
|
expect(detailCount).toBeGreaterThan(0);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
test("自动刷新开关与刷新按钮", async ({ page }) => {
|
||||||
|
const switchBtn = page.getByRole("switch", { name: "自动刷新" });
|
||||||
|
const refreshBtn = page.getByTitle("刷新");
|
||||||
|
|
||||||
|
await expect(switchBtn).toBeVisible();
|
||||||
|
await expect(refreshBtn).toBeVisible();
|
||||||
|
|
||||||
|
// 手动点击刷新按钮
|
||||||
|
await refreshBtn.click();
|
||||||
|
|
||||||
|
// 自动刷新开关切换
|
||||||
|
const isChecked = await switchBtn.isChecked();
|
||||||
|
await switchBtn.click();
|
||||||
|
await expect(switchBtn).toHaveJSProperty("checked", !isChecked);
|
||||||
|
});
|
||||||
|
});
|
||||||
58
src/web/tests/playwright/dashboard.spec.ts
Normal file
58
src/web/tests/playwright/dashboard.spec.ts
Normal file
@ -0,0 +1,58 @@
|
|||||||
|
import { test, expect } from '@playwright/test';
|
||||||
|
|
||||||
|
test.describe('Dashboard 页面测试', () => {
|
||||||
|
|
||||||
|
test.beforeEach(async ({ page }) => {
|
||||||
|
// 打开仪表盘页面
|
||||||
|
await page.goto('http://web.argus.com/dashboard', { waitUntil: 'networkidle' });
|
||||||
|
});
|
||||||
|
|
||||||
|
test('应能成功加载页面并显示标题', async ({ page }) => {
|
||||||
|
await expect(page.locator('text=仪表盘')).toBeVisible();
|
||||||
|
});
|
||||||
|
|
||||||
|
test('应显示节点健康状态卡片', async ({ page }) => {
|
||||||
|
const healthCard = page.locator('text=节点健康状态');
|
||||||
|
await expect(healthCard).toBeVisible();
|
||||||
|
|
||||||
|
// 检查环形图是否渲染
|
||||||
|
const ring = page.locator('svg'); // RingProgress 是 SVG 渲染的
|
||||||
|
const ringCount = await ring.count();
|
||||||
|
expect(ringCount).toBeGreaterThan(0);
|
||||||
|
});
|
||||||
|
|
||||||
|
test('应显示告警统计信息', async ({ page }) => {
|
||||||
|
const alertCard = page.locator('text=告警统计');
|
||||||
|
await expect(alertCard).toBeVisible();
|
||||||
|
|
||||||
|
// 检查告警类别
|
||||||
|
const labels = ['总数', '严重', '警告', '信息'];
|
||||||
|
for (const label of labels) {
|
||||||
|
await expect(page.locator(`text=${label}`)).toBeVisible();
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
test('应正确渲染集群节点表格', async ({ page }) => {
|
||||||
|
const tableHeaders = ['ID', '名称', '状态', '类型', '版本'];
|
||||||
|
for (const header of tableHeaders) {
|
||||||
|
await expect(page.locator(`th:has-text("${header}")`)).toBeVisible();
|
||||||
|
}
|
||||||
|
|
||||||
|
// 至少有一行节点数据
|
||||||
|
const rows = await page.locator('tbody tr').count();
|
||||||
|
expect(rows).toBeGreaterThan(0);
|
||||||
|
});
|
||||||
|
|
||||||
|
test('“查看更多”链接应存在并指向 /nodeInfo', async ({ page }) => {
|
||||||
|
const link = page.locator('a:has-text("查看更多")');
|
||||||
|
await expect(link).toBeVisible();
|
||||||
|
const href = await link.getAttribute('href');
|
||||||
|
expect(href).toContain('/nodeInfo');
|
||||||
|
});
|
||||||
|
|
||||||
|
test('页面应无加载错误提示', async ({ page }) => {
|
||||||
|
await expect(page.locator('text=加载中...')).toHaveCount(0);
|
||||||
|
await expect(page.locator('text=数据加载失败')).toHaveCount(0);
|
||||||
|
});
|
||||||
|
|
||||||
|
});
|
||||||
28
src/web/tests/playwright/helpers/entrycards-helpers.ts
Normal file
28
src/web/tests/playwright/helpers/entrycards-helpers.ts
Normal file
@ -0,0 +1,28 @@
|
|||||||
|
import { Page, expect } from '@playwright/test';
|
||||||
|
import type { metricsEntries } from '../../../src/config/entries';
|
||||||
|
|
||||||
|
export async function testEntryCards(page: Page, entries: typeof metricsEntries, checkLinkNavigation = false) {
|
||||||
|
for (const entry of entries) {
|
||||||
|
// 卡片文本可见
|
||||||
|
const card = page.locator(`text=${entry.label}`);
|
||||||
|
await expect(card).toBeVisible();
|
||||||
|
|
||||||
|
// 卡片链接正确
|
||||||
|
const link = card.locator('..').locator('a');
|
||||||
|
await expect(link).toHaveAttribute('href', entry.href);
|
||||||
|
|
||||||
|
// 图标存在
|
||||||
|
const img = card.locator('..').locator('img');
|
||||||
|
await expect(img).toBeVisible();
|
||||||
|
await expect(img).toHaveAttribute('src', /\/assets\/.+/);
|
||||||
|
|
||||||
|
if (checkLinkNavigation) {
|
||||||
|
const [newPage] = await Promise.all([
|
||||||
|
page.context().waitForEvent('page'),
|
||||||
|
link.click(),
|
||||||
|
]);
|
||||||
|
await expect(newPage).toHaveURL(entry.href);
|
||||||
|
await newPage.close();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
25
src/web/tests/playwright/helpers/testUtils.ts
Normal file
25
src/web/tests/playwright/helpers/testUtils.ts
Normal file
@ -0,0 +1,25 @@
|
|||||||
|
import { Page, expect } from '@playwright/test';
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 通用函数:验证页面导航是否正确
|
||||||
|
*/
|
||||||
|
export async function checkPage(page: Page, path: string, title: string) {
|
||||||
|
await page.goto('http://web.argus.com/');
|
||||||
|
const menu = page.getByRole('link', { name: title });
|
||||||
|
await expect(menu).toBeVisible();
|
||||||
|
await menu.click();
|
||||||
|
await expect(page).toHaveURL(new RegExp(`${path}`));
|
||||||
|
await expect(page.locator('body')).toContainText(title);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 检查页面是否存在 JS 错误
|
||||||
|
*/
|
||||||
|
export async function noConsoleError(page: Page) {
|
||||||
|
const errors: string[] = [];
|
||||||
|
page.on('console', msg => {
|
||||||
|
if (msg.type() === 'error') errors.push(msg.text());
|
||||||
|
});
|
||||||
|
await page.waitForLoadState('networkidle');
|
||||||
|
expect(errors, `发现 JS 错误: ${errors.join(', ')}`).toHaveLength(0);
|
||||||
|
}
|
||||||
11
src/web/tests/playwright/logs.spec.ts
Normal file
11
src/web/tests/playwright/logs.spec.ts
Normal file
@ -0,0 +1,11 @@
|
|||||||
|
import { test, expect } from '@playwright/test';
|
||||||
|
import { logsEntries } from './test-entries';
|
||||||
|
import { testEntryCards } from './helpers/entrycards-helpers';
|
||||||
|
|
||||||
|
test.describe('Logs Page', () => {
|
||||||
|
test('should render all log cards', async ({ page }) => {
|
||||||
|
await page.goto('http://web.argus.com/logs');
|
||||||
|
await expect(page.locator('h2', { hasText: '日志详情' })).toBeVisible();
|
||||||
|
await testEntryCards(page, logsEntries);
|
||||||
|
});
|
||||||
|
});
|
||||||
12
src/web/tests/playwright/metric.spec.ts
Normal file
12
src/web/tests/playwright/metric.spec.ts
Normal file
@ -0,0 +1,12 @@
|
|||||||
|
import { test, expect } from '@playwright/test';
|
||||||
|
import { metricsEntries } from './test-entries';
|
||||||
|
import { testEntryCards } from './helpers/entrycards-helpers';
|
||||||
|
|
||||||
|
|
||||||
|
test.describe('Metrics Page', () => {
|
||||||
|
test('should render all metric cards', async ({ page }) => {
|
||||||
|
await page.goto('http://web.argus.com/metrics');
|
||||||
|
await expect(page.locator('h2', { hasText: '指标详情' })).toBeVisible();
|
||||||
|
await testEntryCards(page, metricsEntries);
|
||||||
|
});
|
||||||
|
});
|
||||||
96
src/web/tests/playwright/node-info.spec.ts
Normal file
96
src/web/tests/playwright/node-info.spec.ts
Normal file
@ -0,0 +1,96 @@
|
|||||||
|
import { test, expect } from "@playwright/test";
|
||||||
|
|
||||||
|
test.describe("节点信息页面 NodeInfo", () => {
|
||||||
|
// 每次测试前打开目标页面
|
||||||
|
test.beforeEach(async ({ page }) => {
|
||||||
|
await page.goto("http://web.argus.com/node");
|
||||||
|
});
|
||||||
|
|
||||||
|
test("页面标题应该正确显示", async ({ page }) => {
|
||||||
|
const title = page.getByRole("heading", { name: "节点信息" });
|
||||||
|
await expect(title).toBeVisible();
|
||||||
|
});
|
||||||
|
|
||||||
|
test("节点表格应该加载数据", async ({ page }) => {
|
||||||
|
const rows = page.locator("table tbody tr");
|
||||||
|
const count = await rows.count();
|
||||||
|
expect(count).toBeGreaterThan(0);
|
||||||
|
});
|
||||||
|
|
||||||
|
test('节点详情测试', async ({ page }) => {
|
||||||
|
// 点击第一个节点的“查看详情”
|
||||||
|
const firstDetailBtn = page.locator('text=查看详情').first();
|
||||||
|
await firstDetailBtn.click();
|
||||||
|
|
||||||
|
const drawer = page.locator('role=dialog[name="节点详情"]');
|
||||||
|
await expect(drawer).toBeVisible();
|
||||||
|
|
||||||
|
// ========================
|
||||||
|
// 1️⃣ 验证基础信息
|
||||||
|
// ========================
|
||||||
|
await expect(drawer.locator('text=注册时间')).toBeVisible();
|
||||||
|
await expect(drawer.locator('text=最近上报时间')).toBeVisible();
|
||||||
|
await expect(drawer.locator('text=最后更新时间')).toBeVisible();
|
||||||
|
|
||||||
|
// ========================
|
||||||
|
// 2️⃣ NodeConfigCard 编辑/保存
|
||||||
|
// ========================
|
||||||
|
const configEditBtn = drawer.locator('div:has-text("配置信息") >> role=button', { hasText: '' });
|
||||||
|
await configEditBtn.click(); // 开启编辑
|
||||||
|
|
||||||
|
const keyInput = drawer.locator('input[placeholder="Key"]').first();
|
||||||
|
const valueInput = drawer.locator('input[placeholder="Value"]').first();
|
||||||
|
|
||||||
|
await keyInput.fill('testKey');
|
||||||
|
await valueInput.fill('testValue');
|
||||||
|
|
||||||
|
const saveBtn = drawer.locator('div:has-text("配置信息") >> role=button', { hasText: '' }).filter({ hasText: '' }).first();
|
||||||
|
await saveBtn.click();
|
||||||
|
|
||||||
|
// 保存后,新配置应该显示在列表中
|
||||||
|
await expect(drawer.locator('text=testKey')).toBeVisible();
|
||||||
|
await expect(drawer.locator('text=testValue')).toBeVisible();
|
||||||
|
|
||||||
|
// ========================
|
||||||
|
// 3️⃣ NodeLabelCard 标签管理
|
||||||
|
// ========================
|
||||||
|
const labelEditBtn = drawer.locator('div:has-text("标签信息") >> role=button', { hasText: '' });
|
||||||
|
await labelEditBtn.click(); // 开启编辑
|
||||||
|
|
||||||
|
const newTagInput = drawer.locator('input[placeholder="新增标签"]');
|
||||||
|
await newTagInput.fill('newTag1');
|
||||||
|
|
||||||
|
const addTagBtn = drawer.locator('div:has-text("标签信息") >> role=button', { hasText: '' }).nth(1);
|
||||||
|
await addTagBtn.click();
|
||||||
|
|
||||||
|
// 保存后标签应该显示
|
||||||
|
const saveLabelBtn = drawer.locator('div:has-text("标签信息") >> role=button', { hasText: '' }).first();
|
||||||
|
await saveLabelBtn.click();
|
||||||
|
await expect(drawer.locator('text=newTag1')).toBeVisible();
|
||||||
|
|
||||||
|
// ========================
|
||||||
|
// 4️⃣ NodeHealthCard 健康信息展示
|
||||||
|
// ========================
|
||||||
|
const healthModule = drawer.locator('div:has-text("健康信息") >> text=healthy').first();
|
||||||
|
await expect(healthModule).toBeVisible();
|
||||||
|
|
||||||
|
// 可选择打开 Info Popover
|
||||||
|
const infoBtn = drawer.locator('div:has-text("健康信息") >> role=button').first();
|
||||||
|
await infoBtn.click();
|
||||||
|
const errorCount = await drawer.locator('text=Error').count();
|
||||||
|
expect(errorCount).toBeGreaterThan(0);
|
||||||
|
|
||||||
|
// ========================
|
||||||
|
// 5️⃣ Drawer 关闭
|
||||||
|
// ========================
|
||||||
|
const closeBtn = drawer.locator('button[aria-label="Close"]');
|
||||||
|
await closeBtn.click();
|
||||||
|
await expect(drawer).toHaveCount(0);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
|
||||||
|
test("Grafana按钮链接应正确", async ({ page }) => {
|
||||||
|
const grafanaLink = page.getByRole("link", { name: "Grafana" }).first();
|
||||||
|
await expect(grafanaLink).toHaveAttribute("href", /\/d\/node_gpu_metrics/);
|
||||||
|
});
|
||||||
14
src/web/tests/playwright/test-entries.ts
Normal file
14
src/web/tests/playwright/test-entries.ts
Normal file
@ -0,0 +1,14 @@
|
|||||||
|
import { EXTERNAL_HOST } from "../../src/config/api";
|
||||||
|
|
||||||
|
export const metricsEntries = [
|
||||||
|
{ label: "Grafana", href: EXTERNAL_HOST.GRAFANA_DASHBOARD, icon: '' },
|
||||||
|
{ label: "Prometheus", href: EXTERNAL_HOST.PROMETHEUS, icon: '' },
|
||||||
|
];
|
||||||
|
|
||||||
|
export const logsEntries = [
|
||||||
|
{ label: "Kibana", href: EXTERNAL_HOST.KIBANA, icon: '' },
|
||||||
|
];
|
||||||
|
|
||||||
|
export const alertsEntries = [
|
||||||
|
{ label: "Alertmanager", href: EXTERNAL_HOST.ALERTS, icon: '' },
|
||||||
|
];
|
||||||
21
src/web/tests/playwright/web-pages.spec.ts
Normal file
21
src/web/tests/playwright/web-pages.spec.ts
Normal file
@ -0,0 +1,21 @@
|
|||||||
|
import { test } from '@playwright/test';
|
||||||
|
import { checkPage, noConsoleError } from './helpers/testUtils';
|
||||||
|
|
||||||
|
const BASE_URL = 'http://web.argus.com';
|
||||||
|
const pages = [
|
||||||
|
{ path: '/dashboard', title: '仪表盘' },
|
||||||
|
{ path: '/nodeInfo', title: '节点信息' },
|
||||||
|
{ path: '/metrics', title: '指标详情' },
|
||||||
|
{ path: '/logs', title: '日志详情' },
|
||||||
|
{ path: '/alerts', title: '告警详情' }
|
||||||
|
];
|
||||||
|
|
||||||
|
test.describe('Argus Web 页面可用性巡检', () => {
|
||||||
|
for (const { path, title } of pages) {
|
||||||
|
test(`${title} 页面加载验证`, async ({ page }) => {
|
||||||
|
await page.goto(`${BASE_URL}${path}`);
|
||||||
|
await checkPage(page, path, title);
|
||||||
|
await noConsoleError(page);
|
||||||
|
});
|
||||||
|
}
|
||||||
|
});
|
||||||
@ -1,19 +0,0 @@
|
|||||||
#!/usr/bin/env bash
|
|
||||||
set -euo pipefail
|
|
||||||
root="$(cd "$(dirname "${BASH_SOURCE[0]}")/../" && pwd)"
|
|
||||||
project_root="$(cd "$root/../../.." && pwd)"
|
|
||||||
|
|
||||||
source "$project_root/scripts/common/build_user.sh"
|
|
||||||
load_build_user
|
|
||||||
|
|
||||||
# 创建新的private目录结构 (基于argus目录结构)
|
|
||||||
echo "[INFO] Creating private directory structure for supervisor-based containers..."
|
|
||||||
mkdir -p "$root/private/argus/web"
|
|
||||||
mkdir -p "$root/private/argus/etc/"
|
|
||||||
|
|
||||||
# 设置数据目录权限
|
|
||||||
echo "[INFO] Setting permissions for data directories..."
|
|
||||||
chown -R "${ARGUS_BUILD_UID}:${ARGUS_BUILD_GID}" "$root/private/argus/web" 2>/dev/null || true
|
|
||||||
chown -R "${ARGUS_BUILD_UID}:${ARGUS_BUILD_GID}" "$root/private/argus/etc" 2>/dev/null || true
|
|
||||||
|
|
||||||
echo "[INFO] Supervisor-based containers will manage their own scripts and configurations"
|
|
||||||
@ -1,10 +0,0 @@
|
|||||||
#!/usr/bin/env bash
|
|
||||||
set -euo pipefail
|
|
||||||
cd "$(dirname "$0")/.."
|
|
||||||
compose_cmd="docker compose"
|
|
||||||
if ! $compose_cmd version >/dev/null 2>&1; then
|
|
||||||
if command -v docker-compose >/dev/null 2>&1; then compose_cmd="docker-compose"; else
|
|
||||||
echo "需要 Docker Compose,请安装后重试" >&2; exit 1; fi
|
|
||||||
fi
|
|
||||||
$compose_cmd -p alert-mvp up -d --remove-orphans
|
|
||||||
echo "[OK] 服务已启动:Web Frontend http://localhost:8080"
|
|
||||||
@ -1,93 +0,0 @@
|
|||||||
#!/usr/bin/env bash
|
|
||||||
set -euo pipefail
|
|
||||||
|
|
||||||
WEB_URL=${WEB_URL:-"http://localhost:8080"}
|
|
||||||
API_URL=${API_URL:-"http://master.argus.com/api/v1/master/nodes"}
|
|
||||||
TIMEOUT=10
|
|
||||||
|
|
||||||
GREEN="\033[1;32m"
|
|
||||||
RED="\033[1;31m"
|
|
||||||
YELLOW="\033[1;33m"
|
|
||||||
RESET="\033[0m"
|
|
||||||
|
|
||||||
echo "[info] 测试 Argus Web 前端启动状态..."
|
|
||||||
echo "--------------------------------------------------"
|
|
||||||
|
|
||||||
# 等待 web 前端可用
|
|
||||||
attempt=1
|
|
||||||
while (( attempt <= 10 )); do
|
|
||||||
if curl -fsS -m "$TIMEOUT" -o /dev/null "$WEB_URL"; then
|
|
||||||
echo "[ok] Web 前端已启动 (${attempt}/10)"
|
|
||||||
break
|
|
||||||
fi
|
|
||||||
echo "[..] 等待 Web 前端启动中 (${attempt}/10)"
|
|
||||||
sleep 3
|
|
||||||
(( attempt++ ))
|
|
||||||
done
|
|
||||||
|
|
||||||
if (( attempt > 10 )); then
|
|
||||||
echo "[err] Web 前端在 30 秒内未启动"
|
|
||||||
exit 1
|
|
||||||
fi
|
|
||||||
|
|
||||||
# 1. 检查首页可访问性
|
|
||||||
echo "[test] 检查首页访问..."
|
|
||||||
if curl -fsS "$WEB_URL" -m "$TIMEOUT" | grep -q "<title>"; then
|
|
||||||
echo -e "[${GREEN}ok${RESET}] 首页可访问"
|
|
||||||
else
|
|
||||||
echo -e "[${RED}err${RESET}] 首页访问失败"
|
|
||||||
exit 1
|
|
||||||
fi
|
|
||||||
|
|
||||||
# 2. 检查静态资源加载
|
|
||||||
echo "[test] 检查静态资源..."
|
|
||||||
if curl -fsS "$WEB_URL/static/js" -m "$TIMEOUT" | grep -q "Cannot GET"; then
|
|
||||||
echo -e "[${YELLOW}warn${RESET}] 静态资源路径可能未正确配置"
|
|
||||||
else
|
|
||||||
echo -e "[${GREEN}ok${RESET}] 静态资源服务正常"
|
|
||||||
fi
|
|
||||||
|
|
||||||
# 3. 检查前端路由兼容
|
|
||||||
echo "[test] 检查 React Router 路由兼容..."
|
|
||||||
if curl -fsS "$WEB_URL/dashboard" -m "$TIMEOUT" | grep -q "<title>"; then
|
|
||||||
echo -e "[${GREEN}ok${RESET}] React Router 路由兼容正常"
|
|
||||||
else
|
|
||||||
echo -e "[${YELLOW}warn${RESET}] /dashboard 路由未正确返回 index.html"
|
|
||||||
fi
|
|
||||||
|
|
||||||
# 4. 测试 API 代理访问
|
|
||||||
echo "[test] 检查 API 代理..."
|
|
||||||
if curl -fsS "$API_URL" -m "$TIMEOUT" | grep -q "nodes"; then
|
|
||||||
echo -e "[${GREEN}ok${RESET}] API 代理成功"
|
|
||||||
else
|
|
||||||
echo -e "[${YELLOW}warn${RESET}] API 代理请求失败,请检查 Nginx proxy_pass"
|
|
||||||
fi
|
|
||||||
|
|
||||||
# 5. 页面关键字验证
|
|
||||||
echo "[test] 检查关键内容..."
|
|
||||||
if curl -fsS "$WEB_URL" | grep -q "Argus"; then
|
|
||||||
echo -e "[${GREEN}ok${RESET}] 页面包含关键字 'Argus'"
|
|
||||||
else
|
|
||||||
echo -e "[${YELLOW}warn${RESET}] 页面内容中未找到 'Argus'"
|
|
||||||
fi
|
|
||||||
|
|
||||||
# 6. DNS 检查
|
|
||||||
echo "[test] 检查 DNS 解析..."
|
|
||||||
if dig +short web.argus.com >/dev/null; then
|
|
||||||
echo -e "[${GREEN}ok${RESET}] 域名 web.argus.com 解析正常"
|
|
||||||
else
|
|
||||||
echo -e "[${YELLOW}warn${RESET}] 域名 web.argus.com 解析失败"
|
|
||||||
fi
|
|
||||||
|
|
||||||
# 7. 响应时间测试
|
|
||||||
echo "[test] 检查响应时间..."
|
|
||||||
response_time=$(curl -o /dev/null -s -w "%{time_total}\n" "$WEB_URL")
|
|
||||||
echo "[info] 响应时间: ${response_time}s"
|
|
||||||
if (( $(echo "$response_time > 2.0" | bc -l) )); then
|
|
||||||
echo -e "[${YELLOW}warn${RESET}] 响应时间较慢(>2s)"
|
|
||||||
else
|
|
||||||
echo -e "[${GREEN}ok${RESET}] 响应时间正常"
|
|
||||||
fi
|
|
||||||
|
|
||||||
echo "--------------------------------------------------"
|
|
||||||
echo "[done] Web 前端测试完成 ✅"
|
|
||||||
@ -1,21 +0,0 @@
|
|||||||
#!/usr/bin/env bash
|
|
||||||
set -euo pipefail
|
|
||||||
cd "$(dirname "$0")/.."
|
|
||||||
compose_cmd="docker compose"
|
|
||||||
if ! $compose_cmd version >/dev/null 2>&1; then
|
|
||||||
if command -v docker-compose >/dev/null 2>&1; then compose_cmd="docker-compose"; else
|
|
||||||
echo "需要 Docker Compose,请安装后重试" >&2; exit 1; fi
|
|
||||||
fi
|
|
||||||
$compose_cmd -p alert-mvp down
|
|
||||||
echo "[OK] 已停止所有容器"
|
|
||||||
|
|
||||||
# 清理private目录内容
|
|
||||||
echo "[INFO] 清理private目录内容..."
|
|
||||||
cd "$(dirname "$0")/.."
|
|
||||||
if [ -d "private" ]; then
|
|
||||||
# 删除private目录及其所有内容
|
|
||||||
rm -rf private
|
|
||||||
echo "[OK] 已清理private目录"
|
|
||||||
else
|
|
||||||
echo "[INFO] private目录不存在,无需清理"
|
|
||||||
fi
|
|
||||||
@ -1,85 +0,0 @@
|
|||||||
#!/usr/bin/env bash
|
|
||||||
set -euo pipefail
|
|
||||||
|
|
||||||
echo "======================================="
|
|
||||||
echo "ARGUS Web System End-to-End Test"
|
|
||||||
echo "======================================="
|
|
||||||
echo ""
|
|
||||||
|
|
||||||
# 记录测试开始时间
|
|
||||||
test_start_time=$(date +%s)
|
|
||||||
|
|
||||||
# 函数:等待服务就绪
|
|
||||||
wait_for_services() {
|
|
||||||
echo "[INFO] Waiting for all services to be ready..."
|
|
||||||
local max_attempts=${SERVICE_WAIT_ATTEMPTS:-120}
|
|
||||||
local attempt=1
|
|
||||||
|
|
||||||
while [ $attempt -le $max_attempts ]; do
|
|
||||||
if curl -fs http://localhost:8080 >/dev/null 2>&1; then
|
|
||||||
echo "[OK] All services are ready!"
|
|
||||||
return 0
|
|
||||||
fi
|
|
||||||
echo " Waiting for services... ($attempt/$max_attempts)"
|
|
||||||
sleep 5
|
|
||||||
((attempt++))
|
|
||||||
done
|
|
||||||
|
|
||||||
echo "[ERROR] Services not ready after $max_attempts attempts"
|
|
||||||
return 1
|
|
||||||
}
|
|
||||||
|
|
||||||
# 函数:显示测试步骤
|
|
||||||
show_step() {
|
|
||||||
echo ""
|
|
||||||
echo "🔄 Step $1: $2"
|
|
||||||
echo "----------------------------------------"
|
|
||||||
}
|
|
||||||
|
|
||||||
# 函数:验证步骤结果
|
|
||||||
verify_step() {
|
|
||||||
if [ $? -eq 0 ]; then
|
|
||||||
echo "✅ $1 - SUCCESS"
|
|
||||||
else
|
|
||||||
echo "❌ $1 - FAILED"
|
|
||||||
exit 1
|
|
||||||
fi
|
|
||||||
}
|
|
||||||
|
|
||||||
# 开始端到端测试
|
|
||||||
show_step "1" "Bootstrap - Initialize environment"
|
|
||||||
./scripts/01_bootstrap.sh
|
|
||||||
verify_step "Bootstrap"
|
|
||||||
|
|
||||||
show_step "2" "Startup - Start all services"
|
|
||||||
./scripts/02_up.sh
|
|
||||||
verify_step "Service startup"
|
|
||||||
|
|
||||||
# 等待服务完全就绪
|
|
||||||
wait_for_services || exit 1
|
|
||||||
|
|
||||||
# 测试前端页面
|
|
||||||
show_step "3" "Web - Check frontend availability"
|
|
||||||
./scripts/03_web_health_check.sh
|
|
||||||
verify_step "Web frontend availability"
|
|
||||||
|
|
||||||
# 清理环境
|
|
||||||
show_step "4" "Cleanup - Stop all services"
|
|
||||||
./scripts/04_down.sh
|
|
||||||
verify_step "Service cleanup"
|
|
||||||
|
|
||||||
# 计算总测试时间
|
|
||||||
test_end_time=$(date +%s)
|
|
||||||
total_time=$((test_end_time - test_start_time))
|
|
||||||
|
|
||||||
echo ""
|
|
||||||
echo "======================================="
|
|
||||||
echo "🎉 END-TO-END TEST COMPLETED SUCCESSFULLY!"
|
|
||||||
echo "======================================="
|
|
||||||
echo "📊 Test Summary:"
|
|
||||||
echo " • Total time: ${total_time}s"
|
|
||||||
echo " • Alertmanager status: $am_status"
|
|
||||||
echo " • All services started and stopped successfully"
|
|
||||||
echo ""
|
|
||||||
echo "✅ The ARGUS Web system is working correctly!"
|
|
||||||
echo ""
|
|
||||||
77
src/web/tests/scripts/verify-web-frontend.sh
Normal file
77
src/web/tests/scripts/verify-web-frontend.sh
Normal file
@ -0,0 +1,77 @@
|
|||||||
|
#!/usr/bin/env bash
|
||||||
|
set -euo pipefail
|
||||||
|
|
||||||
|
# -----------------------------------------
|
||||||
|
# Web 前端自动化验证脚本(部署后执行)
|
||||||
|
# -----------------------------------------
|
||||||
|
|
||||||
|
PROJECT_ROOT="$(dirname "$0")"
|
||||||
|
WEB_DIR="$PROJECT_ROOT"
|
||||||
|
REPORT_DIR="$WEB_DIR/playwright-report"
|
||||||
|
FRONTEND_URL="http://web.argus.com:8080"
|
||||||
|
TIMEOUT=120 # 最长等待前端启动时间(秒)
|
||||||
|
|
||||||
|
echo "🔍 [1/4] 检查前端服务是否已启动 (${FRONTEND_URL}) ..."
|
||||||
|
|
||||||
|
# 等待前端服务可访问
|
||||||
|
for ((i=1; i<=$TIMEOUT; i++)); do
|
||||||
|
STATUS_CODE=$(curl -s -o /dev/null -w "%{http_code}" "$FRONTEND_URL" || true)
|
||||||
|
if [[ "$STATUS_CODE" == "200" ]]; then
|
||||||
|
echo "✅ 前端服务已启动并可访问"
|
||||||
|
break
|
||||||
|
fi
|
||||||
|
sleep 2
|
||||||
|
if [ $i -eq $TIMEOUT ]; then
|
||||||
|
echo "❌ 等待前端启动超时 (${TIMEOUT}s)"
|
||||||
|
exit 1
|
||||||
|
fi
|
||||||
|
done
|
||||||
|
|
||||||
|
# -----------------------------------------
|
||||||
|
# 2. 执行 Playwright 测试
|
||||||
|
# -----------------------------------------
|
||||||
|
echo "[2/4] 执行 Playwright 自动化测试..."
|
||||||
|
|
||||||
|
cd "$WEB_DIR"
|
||||||
|
|
||||||
|
# 确保依赖已安装
|
||||||
|
if [ ! -d "node_modules" ]; then
|
||||||
|
echo "未检测到依赖,开始安装..."
|
||||||
|
npm ci
|
||||||
|
fi
|
||||||
|
|
||||||
|
# 清理旧报告
|
||||||
|
rm -rf "$REPORT_DIR"
|
||||||
|
|
||||||
|
# 运行测试(带失败检测)
|
||||||
|
set +e # 暂时关闭自动退出,便于捕获测试结果
|
||||||
|
npx playwright test tests/playwright --reporter=list,html
|
||||||
|
TEST_RESULT=$?
|
||||||
|
set -e # 恢复严格模式
|
||||||
|
|
||||||
|
# -----------------------------------------
|
||||||
|
# 3. 检查测试结果
|
||||||
|
# -----------------------------------------
|
||||||
|
echo "[3/4] 检查测试结果..."
|
||||||
|
|
||||||
|
if [ $TEST_RESULT -eq 0 ]; then
|
||||||
|
echo "[✓] 所有测试通过!"
|
||||||
|
else
|
||||||
|
echo "[X] 存在测试未通过,请查看报告。"
|
||||||
|
fi
|
||||||
|
|
||||||
|
# -----------------------------------------
|
||||||
|
# 4. 输出报告信息
|
||||||
|
# -----------------------------------------
|
||||||
|
echo "[4/4] 生成测试报告..."
|
||||||
|
|
||||||
|
if [ -d "$REPORT_DIR" ]; then
|
||||||
|
echo "测试报告已生成:$REPORT_DIR"
|
||||||
|
echo "可执行以下命令查看详细报告:"
|
||||||
|
echo " npx playwright show-report"
|
||||||
|
else
|
||||||
|
echo "未生成报告目录,请检查执行日志。"
|
||||||
|
fi
|
||||||
|
|
||||||
|
# 将测试结果作为退出码返回
|
||||||
|
exit $TEST_RESULT
|
||||||
Loading…
x
Reference in New Issue
Block a user