From 11942e2b9f3d43a6cab3765421e4c43b20decdac Mon Sep 17 00:00:00 2001 From: shanshanzhong Date: Sat, 3 Jan 2026 19:43:40 -0800 Subject: [PATCH] =?UTF-8?q?docs:=20=E6=B7=BB=E5=8A=A0=E9=A1=B9=E7=9B=AE?= =?UTF-8?q?=E8=AF=B4=E6=98=8E=E6=96=87=E6=A1=A3=E5=92=8C=E5=88=9D=E5=A7=8B?= =?UTF-8?q?=E5=8C=96=E9=85=8D=E7=BD=AE?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit ci: 添加docker工作流配置并更新项目名称 --- .gitea/workflows/docker.yml | 790 ++++++++++++++++++++++++++++++++++++ package.json | 8 +- 2 files changed, 794 insertions(+), 4 deletions(-) create mode 100644 .gitea/workflows/docker.yml diff --git a/.gitea/workflows/docker.yml b/.gitea/workflows/docker.yml new file mode 100644 index 0000000..2b5288f --- /dev/null +++ b/.gitea/workflows/docker.yml @@ -0,0 +1,790 @@ +name: CI + +on: + push: + branches: + - main + - dev + - cicd + pull_request: + branches: + - main + - dev + - cicd + +env: + DOMAIN_URL: git.kxsw.us + REPO: ${{ vars.REPO }} + TELEGRAM_BOT_TOKEN: 8114337882:AAHkEx03HSu7RxN4IHBJJEnsK9aPPzNLIk0 + TELEGRAM_CHAT_ID: "-4940243803" + DOCKER_REGISTRY: registry.kxsw.us + DOCKER_BUILDKIT: 1 + DOCKER_API_VERSION: "1.44" + # Host SSH - 根据分支动态选择 + SSH_HOST: ${{ github.ref_name == 'main' && vars.PRO_SSH_HOST || (github.ref_name == 'dev' && vars.DEV_SSH_HOST || vars.PRO_SSH_HOST) }} + SSH_PORT: ${{ github.ref_name == 'main' && vars.PRO_SSH_PORT || (github.ref_name == 'dev' && vars.DEV_SSH_PORT || vars.PRO_SSH_PORT) }} + SSH_USER: ${{ github.ref_name == 'main' && vars.PRO_SSH_USER || (github.ref_name == 'dev' && vars.DEV_SSH_USER || vars.PRO_SSH_USER) }} + SSH_PASSWORD: ${{ github.ref_name == 'main' && vars.PRO_SSH_PASSWORD || (github.ref_name == 'dev' && vars.DEV_SSH_PASSWORD || vars.PRO_SSH_PASSWORD) }} + +jobs: + build: + runs-on: fastvpn-admin01 + container: + image: node:20 + strategy: + matrix: + # 只有node支持版本号别名 + node: ['20.15.1'] + + steps: + - name: Checkout code + uses: actions/checkout@v4 + with: + fetch-depth: 0 + + - name: 缓存服务健康检查 + id: cache-health + continue-on-error: true + run: | + echo "检查缓存服务可用性..." + + # 设置缓存可用性标志 + CACHE_AVAILABLE=true + + # 测试GitHub Actions缓存API + if ! curl -s --connect-timeout 10 --max-time 30 "https://api.github.com/repos/${{ github.repository }}/actions/caches" > /dev/null 2>&1; then + echo "⚠️ GitHub Actions缓存服务不可用,将跳过缓存步骤" + CACHE_AVAILABLE=false + else + echo "✅ 缓存服务可用" + fi + + echo "CACHE_AVAILABLE=$CACHE_AVAILABLE" >> $GITHUB_ENV + echo "cache-available=$CACHE_AVAILABLE" >> $GITHUB_OUTPUT + + - name: 缓存降级提示 + if: env.CACHE_AVAILABLE == 'false' + run: | + echo "🔄 缓存服务不可用,构建将在无缓存模式下进行" + echo "⏱️ 这可能会增加构建时间,但不会影响构建结果" + echo "📦 所有依赖将重新下载和安装" + + - name: Install system tools (jq, docker, curl) + run: | + set -e + export DEBIAN_FRONTEND=noninteractive + echo "Waiting for apt/dpkg locks (unattended-upgrades) to release..." + # Wait up to 300s for unattended-upgrades/apt/dpkg locks + end=$((SECONDS+300)) + while true; do + LOCKS_BUSY=0 + # If unattended-upgrades is running, mark busy + if pgrep -x unattended-upgrades >/dev/null 2>&1; then LOCKS_BUSY=1; fi + # If fuser exists, check common lock files + if command -v fuser >/dev/null 2>&1; then + if fuser /var/lib/dpkg/lock >/dev/null 2>&1 \ + || fuser /var/lib/dpkg/lock-frontend >/dev/null 2>&1 \ + || fuser /var/lib/apt/lists/lock >/dev/null 2>&1; then + LOCKS_BUSY=1 + fi + fi + # Break if not busy + if [ "$LOCKS_BUSY" -eq 0 ]; then break; fi + # Timeout after ~5 minutes + if [ $SECONDS -ge $end ]; then + echo "Timeout waiting for apt/dpkg locks, proceeding with Dpkg::Lock::Timeout..." + break + fi + echo "Still waiting for locks..."; sleep 5 + done + apt-get update -y -o Dpkg::Lock::Timeout=600 + # 基础工具和GPG + apt-get install -y -o Dpkg::Lock::Timeout=600 jq curl ca-certificates gnupg + # 配置Docker官方源,安装新版CLI与Buildx插件(支持 API 1.44+) + install -m 0755 -d /etc/apt/keyrings + curl -fsSL https://download.docker.com/linux/debian/gpg | gpg --dearmor -o /etc/apt/keyrings/docker.gpg + chmod a+r /etc/apt/keyrings/docker.gpg + echo "deb [arch=$(dpkg --print-architecture) signed-by=/etc/apt/keyrings/docker.gpg] https://download.docker.com/linux/debian $(. /etc/os-release && echo $VERSION_CODENAME) stable" > /etc/apt/sources.list.d/docker.list + apt-get update -y -o Dpkg::Lock::Timeout=600 + apt-get install -y -o Dpkg::Lock::Timeout=600 docker-ce-cli docker-buildx-plugin + docker --version + jq --version + curl --version + + - name: Set up Docker Buildx + run: | + # Check if buildx is available + if docker buildx version >/dev/null 2>&1; then + echo "Docker Buildx is available" + # Create builder if it doesn't exist + if ! docker buildx ls | grep -q "builder"; then + docker buildx create --name builder --driver docker-container + fi + # Use the builder + docker buildx use builder + docker buildx inspect --bootstrap + else + echo "Docker Buildx not available, using regular docker build" + fi + + - name: Install Bun + run: | + echo "=== Installing Bun ===" + echo "Current working directory: $(pwd)" + echo "Current user: $(whoami)" + echo "Home directory: $HOME" + + # 设置Bun安装路径 + export BUN_INSTALL="$HOME/.bun" + echo "BUN_INSTALL=$BUN_INSTALL" >> $GITHUB_ENV + echo "PATH=$BUN_INSTALL/bin:${PATH}" >> $GITHUB_ENV + + # 检查缓存是否存在 + if [ -d "$BUN_INSTALL" ]; then + echo "✅ Bun cache found at $BUN_INSTALL" + ls -la "$BUN_INSTALL" || true + else + echo "❌ No Bun cache found, will install fresh" + fi + + # 安装Bun + curl -fsSL https://bun.sh/install | bash + + # 验证安装 + "$BUN_INSTALL/bin/bun" --version + echo "✅ Bun installed successfully" + + - name: Configure npm registry (npmmirror) and canvas mirror + run: | + echo "registry=https://registry.npmmirror.com" >> .npmrc + echo "canvas_binary_host_mirror=https://registry.npmmirror.com/-/binary/canvas" >> .npmrc + + - name: Install dependencies cache (Bun) + if: env.CACHE_AVAILABLE == 'true' + uses: actions/cache@v4 + continue-on-error: true + with: + path: ~/.bun + key: bun-${{ runner.os }}-${{ matrix.node }}-${{ hashFiles('bun.lock') }} + restore-keys: | + bun-${{ runner.os }}-${{ matrix.node }}- + bun-${{ runner.os }}- + + - name: Install dependencies cache (node_modules) + if: env.CACHE_AVAILABLE == 'true' + uses: actions/cache@v4 + continue-on-error: true + with: + path: | + node_modules + apps/*/node_modules + packages/*/node_modules + key: node-modules-${{ runner.os }}-${{ matrix.node }}-${{ hashFiles('bun.lock', 'package.json', 'apps/*/package.json', 'packages/*/package.json') }} + restore-keys: | + node-modules-${{ runner.os }}-${{ matrix.node }}- + node-modules-${{ runner.os }}- + + - name: 缓存状态检查和设置 + run: | + echo "=== 缓存状态检查 ===" + echo "检查缓存恢复状态..." + + # 检查各种缓存目录 + echo "Bun缓存: $([ -d ~/.bun ] && echo '✅ 已发现' || echo '❌ 缺失')" + echo "node_modules: $([ -d node_modules ] && echo '✅ 已发现' || echo '❌ 缺失')" + echo "Turbo缓存: $([ -d .turbo ] && echo '✅ 已发现' || echo '❌ 缺失')" + + # 显示缓存大小 + if [ -d ~/.bun ]; then + echo "Bun缓存大小: $(du -sh ~/.bun 2>/dev/null || echo '未知')" + fi + if [ -d node_modules ]; then + echo "node_modules大小: $(du -sh node_modules 2>/dev/null || echo '未知')" + fi + if [ -d .turbo ]; then + echo "Turbo缓存大小: $(du -sh .turbo 2>/dev/null || echo '未知')" + fi + + echo "=== 缓存设置 ===" + # 确保缓存目录存在且权限正确 + mkdir -p ~/.bun ~/.cache .turbo + chmod -R 755 ~/.bun ~/.cache .turbo 2>/dev/null || true + + # 设置Bun环境变量 + echo "BUN_INSTALL_CACHE_DIR=$HOME/.cache/bun" >> $GITHUB_ENV + echo "BUN_INSTALL_BIN_DIR=$HOME/.bun/bin" >> $GITHUB_ENV + + echo "✅ 缓存目录已准备完成" + + - name: Turborepo cache (.turbo) + if: env.CACHE_AVAILABLE == 'true' + uses: actions/cache@v4 + continue-on-error: true + with: + path: .turbo + key: turbo-${{ runner.os }}-${{ hashFiles('turbo.json') }}-${{ hashFiles('apps//package.json') }}-${{ hashFiles('packages//package.json') }}-${{ hashFiles('bun.lock') }} + restore-keys: | + turbo-${{ runner.os }}-${{ hashFiles('turbo.json') }}-${{ hashFiles('apps//package.json') }}-${{ hashFiles('packages//package.json') }}- + turbo-${{ runner.os }}-${{ hashFiles('turbo.json') }}- + turbo-${{ runner.os }}- + + - name: 安装依赖 (bun) + run: | + echo "=== 依赖安装调试信息 ===" + echo "当前目录: $(pwd)" + echo "Bun版本: $(bun --version)" + + # 检查node_modules缓存状态 + if [ -d "node_modules" ]; then + echo "✅ 发现node_modules缓存" + echo "node_modules大小: $(du -sh node_modules 2>/dev/null || echo '未知')" + else + echo "❌ 未发现node_modules缓存" + fi + + # 检查bun.lock文件 + if [ -f "bun.lock" ]; then + echo "✅ 发现bun.lock文件" + else + echo "❌ 未发现bun.lock文件" + fi + + echo "=== 开始安装依赖 ===" + echo "安装开始时间: $(date)" + bun install --frozen-lockfile + echo "安装完成时间: $(date)" + + echo "=== 依赖安装完成 ===" + echo "最终node_modules大小: $(du -sh node_modules 2>/dev/null || echo '未知')" + + # 验证缓存效果 + echo "=== 缓存效果验证 ===" + if [ -d "node_modules" ]; then + echo "✅ 依赖安装成功" + echo "包数量: $(ls node_modules | wc -l 2>/dev/null || echo '未知')" + else + echo "⚠️ 依赖可能未完全安装" + fi + + + - name: Decide build target (admin/user/both) + run: | + set -e + COMMIT_MSG="${{ github.event.head_commit.message }}" + BUILD_TARGET="both" + if echo "$COMMIT_MSG" | grep -qi "\[admin-only\]"; then + BUILD_TARGET="admin" + elif echo "$COMMIT_MSG" | grep -qi "\[user-only\]"; then + BUILD_TARGET="user" + else + if git rev-parse HEAD^ >/dev/null 2>&1; then + RANGE="HEAD^..HEAD" + else + RANGE="$(git rev-list --max-parents=0 HEAD)..HEAD" + fi + CHANGED=$(git diff --name-only $RANGE || true) + ADMIN_MATCH=$(echo "$CHANGED" | grep -E '^(apps/admin/|docker/ppanel-admin-web/)' || true) + USER_MATCH=$(echo "$CHANGED" | grep -E '^(apps/user/|docker/ppanel-user-web/)' || true) + PACKAGE_MATCH=$(echo "$CHANGED" | grep -E '^(packages/|turbo.json|package.json|bun.lock)' || true) + if [ -n "$PACKAGE_MATCH" ]; then + BUILD_TARGET="both" + else + if [ -n "$ADMIN_MATCH" ] && [ -z "$USER_MATCH" ]; then BUILD_TARGET="admin"; fi + if [ -n "$USER_MATCH" ] && [ -z "$ADMIN_MATCH" ]; then BUILD_TARGET="user"; fi + if [ -n "$ADMIN_MATCH" ] && [ -n "$USER_MATCH" ]; then BUILD_TARGET="both"; fi + fi + fi + echo "BUILD_TARGET=$BUILD_TARGET" >> $GITHUB_ENV + echo "Decided BUILD_TARGET=$BUILD_TARGET" + + - name: Read version from package.json + run: | + if [ "$BUILD_TARGET" = "admin" ]; then + VERSION=$(jq -r .version apps/admin/package.json) + echo "使用 admin 应用版本: $VERSION" + elif [ "$BUILD_TARGET" = "user" ]; then + VERSION=$(jq -r .version apps/user/package.json) + echo "使用 user 应用版本: $VERSION" + else + # both 或其他情况使用根目录版本 + VERSION=$(jq -r .version package.json) + echo "使用根目录版本: $VERSION" + fi + if [ "$VERSION" = "null" ] || [ -z "$VERSION" ] || [ "$VERSION" = "undefined" ]; then + echo "检测到版本为空,回退到根目录版本" + VERSION=$(jq -r .version package.json) + fi + echo "VERSION=$VERSION" >> $GITHUB_ENV + + - name: 根据分支动态设置API地址 + run: | + if [ "${{ github.ref_name }}" = "main" ]; then + echo "NEXT_PUBLIC_API_URL=https://api.airoport.co" >> $GITHUB_ENV + echo "为main分支设置生产环境API地址" + elif [ "${{ github.ref_name }}" = "develop" ]; then + echo "NEXT_PUBLIC_API_URL=https://api.kxsw.us" >> $GITHUB_ENV + echo "为 develop 分支设置开发环境API地址" + else + echo "NEXT_PUBLIC_API_URL=https://api.hifast.biz" >> $GITHUB_ENV + echo "为其他分支设置默认API地址" + fi + echo "BRANCH=${{ github.ref_name }}" >> $GITHUB_ENV + + - name: Cache Next.js build artifacts (.next/cache) + if: env.CACHE_AVAILABLE == 'true' + uses: actions/cache@v4 + continue-on-error: true + with: + path: | + apps/admin/.next/cache + apps/user/.next/cache + key: nextcache-${{ runner.os }}-${{ hashFiles('apps//package.json') }}-${{ hashFiles('packages//package.json') }}-${{ hashFiles('turbo.json') }}-${{ hashFiles('bun.lock') }} + restore-keys: | + nextcache-${{ runner.os }}-${{ hashFiles('apps//package.json') }}-${{ hashFiles('packages//package.json') }}- + nextcache-${{ runner.os }}- + + - name: Cache build outputs + if: env.CACHE_AVAILABLE == 'true' + uses: actions/cache@v4 + continue-on-error: true + with: + path: | + apps/admin/.next + apps/user/.next + apps/admin/dist + apps/user/dist + key: build-${{ runner.os }}-${{ hashFiles('apps//*.ts', 'apps//*.tsx', 'apps//*.js', 'apps//*.jsx') }}-${{ hashFiles('packages//*.ts', 'packages//*.tsx') }}-${{ hashFiles('bun.lock') }} + restore-keys: | + build-${{ runner.os }}-${{ hashFiles('apps//*.ts', 'apps//*.tsx', 'apps//*.js', 'apps//*.jsx') }}- + build-${{ runner.os }}- + + - name: Cache ESLint + if: env.CACHE_AVAILABLE == 'true' + uses: actions/cache@v4 + continue-on-error: true + with: + path: | + .eslintcache + apps/admin/.eslintcache + apps/user/.eslintcache + key: eslint-${{ runner.os }}-${{ hashFiles('.eslintrc*', 'apps//.eslintrc*', 'packages//.eslintrc*') }}-${{ hashFiles('bun.lock') }} + restore-keys: | + eslint-${{ runner.os }}- + + - name: Cache TypeScript + if: env.CACHE_AVAILABLE == 'true' + uses: actions/cache@v4 + continue-on-error: true + with: + path: | + .tsbuildinfo + apps/admin/.tsbuildinfo + apps/user/.tsbuildinfo + packages//.tsbuildinfo + key: typescript-${{ runner.os }}-${{ hashFiles('tsconfig*.json', 'apps//tsconfig*.json', 'packages//tsconfig*.json') }}-${{ hashFiles('bun.lock') }} + restore-keys: | + typescript-${{ runner.os }}- + + - name: 构建管理面板 + if: env.BUILD_TARGET == 'admin' || env.BUILD_TARGET == 'both' + run: bun run build --filter=ppanel-admin-web + + - name: 构建用户面板 + if: env.BUILD_TARGET == 'user' || env.BUILD_TARGET == 'both' + run: bun run build --filter=ppanel-user-web + + - name: 构建并推送管理面板Docker镜像 + if: env.BUILD_TARGET == 'admin' || env.BUILD_TARGET == 'both' + run: | + if docker buildx version >/dev/null 2>&1; then + echo "使用docker buildx进行优化构建" + docker buildx build \ + --platform linux/amd64 \ + --cache-from type=registry,ref=${{ env.DOCKER_REGISTRY }}/ppanel/fastvpn-admin-web:cache \ + --cache-to type=registry,ref=${{ env.DOCKER_REGISTRY }}/ppanel/fastvpn-admin-web:cache,mode=max \ + -f ./docker/ppanel-admin-web/Dockerfile \ + -t ${{ env.DOCKER_REGISTRY }}/ppanel/fastvpn-admin-web:${{ env.VERSION }} \ + --push . + else + echo "使用常规docker构建" + docker build -f ./docker/ppanel-admin-web/Dockerfile -t ${{ env.DOCKER_REGISTRY }}/ppanel/fastvpn-admin-web:${{ env.VERSION }} . + docker push ${{ env.DOCKER_REGISTRY }}/ppanel/fastvpn-admin-web:${{ env.VERSION }} + fi + + - name: 构建并推送用户面板Docker镜像 + if: env.BUILD_TARGET == 'user' || env.BUILD_TARGET == 'both' + run: | + if docker buildx version >/dev/null 2>&1; then + echo "使用docker buildx进行优化构建" + docker buildx build \ + --platform linux/amd64 \ + --cache-from type=registry,ref=${{ env.DOCKER_REGISTRY }}/ppanel/ppanel-user-web:cache \ + --cache-to type=registry,ref=${{ env.DOCKER_REGISTRY }}/ppanel/ppanel-user-web:cache,mode=max \ + -f ./docker/ppanel-user-web/Dockerfile \ + -t ${{ env.DOCKER_REGISTRY }}/ppanel/ppanel-user-web:${{ env.VERSION }} \ + --push . + else + echo "使用常规docker构建" + docker build -f ./docker/ppanel-user-web/Dockerfile -t ${{ env.DOCKER_REGISTRY }}/ppanel/ppanel-user-web:${{ env.VERSION }} . + docker push ${{ env.DOCKER_REGISTRY }}/ppanel/ppanel-user-web:${{ env.VERSION }} + fi + + - name: SSH连接预检查 + if: env.BUILD_TARGET == 'admin' || env.BUILD_TARGET == 'user' || env.BUILD_TARGET == 'both' + uses: appleboy/ssh-action@v1.0.3 + with: + host: ${{ env.SSH_HOST }} + username: ${{ env.SSH_USER }} + password: ${{ env.SSH_PASSWORD }} + port: ${{ env.SSH_PORT }} + timeout: 300s + command_timeout: 600s + debug: true + script: | + echo "=== SSH连接测试 ===" + echo "连接时间: $(date)" + echo "服务器主机名: $(hostname)" + echo "当前用户: $(whoami)" + echo "系统信息: $(uname -a)" + echo "Docker版本: $(docker --version 2>/dev/null || echo 'Docker未安装')" + echo "✅ SSH连接成功" + + - name: 部署管理面板到服务器 + if: env.BUILD_TARGET == 'admin' || env.BUILD_TARGET == 'both' + uses: appleboy/ssh-action@v1.0.3 + with: + host: ${{ env.SSH_HOST }} + username: ${{ env.SSH_USER }} + password: ${{ env.SSH_PASSWORD }} + port: ${{ env.SSH_PORT }} + timeout: 300s + command_timeout: 600s + script: | + echo "=== SSH变量调试信息 ===" + echo "DOCKER_REGISTRY: ${{ env.DOCKER_REGISTRY }}" + echo "VERSION: ${{ env.VERSION }}" + echo "NEXT_PUBLIC_API_URL: ${{ env.NEXT_PUBLIC_API_URL }}" + echo "BRANCH: ${{ env.BRANCH }}" + + echo "=== 部署管理面板 ===" + + # 网络连通性检查 + echo "检查镜像服务器连通性..." + REGISTRY_HOST=$(echo "${{ env.DOCKER_REGISTRY }}" | sed 's|https\?://||' | cut -d'/' -f1) + echo "镜像仓库地址: $REGISTRY_HOST" + + if ping -c 3 "$REGISTRY_HOST" > /dev/null 2>&1; then + echo "✅ 镜像服务器连通性正常" + else + echo "⚠️ 镜像服务器ping失败,但继续尝试拉取镜像" + fi + + # 检查Docker登录状态 + echo "检查Docker登录状态..." + if docker info > /dev/null 2>&1; then + echo "✅ Docker服务正常" + else + echo "❌ Docker服务异常" + exit 1 + fi + + # 拉取镜像(带重试) + echo "拉取Docker镜像..." + for i in {1..3}; do + echo "尝试拉取镜像 ($i/3): ${{ env.DOCKER_REGISTRY }}/ppanel/ppanel-admin-web:${{ env.VERSION }}" + if docker pull ${{ env.DOCKER_REGISTRY }}/ppanel/ppanel-admin-web:${{ env.VERSION }}; then + echo "✅ 镜像拉取成功" + break + else + echo "❌ 镜像拉取失败,重试 $i/3" + echo "检查网络和镜像仓库状态..." + + # 显示详细错误信息 + echo "--- 网络诊断信息 ---" + echo "DNS解析测试:" + nslookup "$REGISTRY_HOST" || echo "DNS解析失败" + echo "网络连通性测试:" + ping -c 2 "$REGISTRY_HOST" || echo "ping失败" + echo "Docker镜像仓库连接测试:" + curl -I "https://$REGISTRY_HOST/v2/" 2>/dev/null || echo "仓库API访问失败" + + sleep 5 + if [ $i -eq 3 ]; then + echo "❌ 镜像拉取失败,部署终止" + echo "请检查:" + echo "1. 网络连接是否正常" + echo "2. 镜像仓库是否可访问" + echo "3. 镜像标签是否存在" + echo "4. Docker登录凭据是否正确" + exit 1 + fi + fi + done + + # 安全停止和移除容器 + echo "检查现有容器状态..." + CONTAINER_NAME="ppanel-admin-web" + + # 检查容器是否存在 + if docker ps -aq -f name=$CONTAINER_NAME | grep -q .; then + echo "发现现有容器,开始清理..." + + # 检查容器是否正在运行 + if docker ps -q -f name=$CONTAINER_NAME | grep -q .; then + echo "停止运行中的容器..." + docker stop $CONTAINER_NAME --time=15 || true + sleep 5 + fi + + # 检查容器是否正在被移除 + echo "检查容器移除状态..." + for i in {1..15}; do + # 尝试获取容器状态 + CONTAINER_STATUS=$(docker inspect $CONTAINER_NAME --format='{{.State.Status}}' 2>/dev/null || echo "not_found") + + if [ "$CONTAINER_STATUS" = "not_found" ]; then + echo "✅ 容器已不存在" + break + elif [ "$CONTAINER_STATUS" = "removing" ]; then + echo "⏳ 容器正在移除中,等待完成... $i/15" + sleep 3 + else + echo "尝试移除容器... $i/15" + if docker rm -f $CONTAINER_NAME 2>/dev/null; then + echo "✅ 容器移除成功" + break + else + echo "⚠️ 容器移除失败,重试..." + sleep 2 + fi + fi + + # 最后一次尝试强制清理 + if [ $i -eq 15 ]; then + echo "🔧 执行强制清理..." + docker kill $CONTAINER_NAME 2>/dev/null || true + sleep 2 + docker rm -f $CONTAINER_NAME 2>/dev/null || true + sleep 2 + fi + done + else + echo "✅ 未发现现有容器" + fi + + echo "启动新容器..." + docker run -d \ + --add-host api.airoport.co:103.150.215.40 \ + --name ppanel-admin-web \ + --restart unless-stopped \ + -p 3001:3000 \ + -e NEXT_PUBLIC_API_URL=${{ env.NEXT_PUBLIC_API_URL }} \ + ${{ env.DOCKER_REGISTRY }}/ppanel/ppanel-admin-web:${{ env.VERSION }} + + # 验证容器启动 + echo "验证容器启动状态..." + for i in {1..10}; do + if docker ps -q -f name=ppanel-admin-web | grep -q .; then + echo "✅ 管理面板部署成功" + docker ps -f name=ppanel-admin-web --format "table {{.Names}}\t{{.Status}}\t{{.Ports}}" + exit 0 + else + echo "等待容器启动... $i/10" + sleep 3 + fi + done + + echo "❌ 管理面板部署失败 - 容器未能正常启动" + docker logs ppanel-admin-web || true + exit 1 + + - name: 部署用户面板到服务器 + if: env.BUILD_TARGET == 'user' || env.BUILD_TARGET == 'both' + uses: appleboy/ssh-action@v1.0.3 + with: + host: ${{ env.SSH_HOST }} + username: ${{ env.SSH_USER }} + password: ${{ env.SSH_PASSWORD }} + port: ${{ env.SSH_PORT }} + timeout: 300s + command_timeout: 600s + debug: true + script: | + echo "=== SSH变量调试信息 ===" + echo "DOCKER_REGISTRY: ${{ env.DOCKER_REGISTRY }}" + echo "VERSION: ${{ env.VERSION }}" + echo "NEXT_PUBLIC_API_URL: ${{ env.NEXT_PUBLIC_API_URL }}" + echo "BRANCH: ${{ env.BRANCH }}" + + echo "=== 部署用户面板 ===" + + # 网络连通性检查 + echo "检查镜像服务器连通性..." + REGISTRY_HOST=$(echo "${{ env.DOCKER_REGISTRY }}" | sed 's|https\?://||' | cut -d'/' -f1) + echo "镜像仓库地址: $REGISTRY_HOST" + + if ping -c 3 "$REGISTRY_HOST" > /dev/null 2>&1; then + echo "✅ 镜像服务器连通性正常" + else + echo "⚠️ 镜像服务器ping失败,但继续尝试拉取镜像" + fi + + # 检查Docker登录状态 + echo "检查Docker登录状态..." + if docker info > /dev/null 2>&1; then + echo "✅ Docker服务正常" + else + echo "❌ Docker服务异常" + exit 1 + fi + + # 拉取镜像(带重试) + echo "拉取Docker镜像..." + for i in {1..3}; do + echo "尝试拉取镜像 ($i/3): ${{ env.DOCKER_REGISTRY }}/ppanel/ppanel-user-web:${{ env.VERSION }}" + if docker pull ${{ env.DOCKER_REGISTRY }}/ppanel/ppanel-user-web:${{ env.VERSION }}; then + echo "✅ 镜像拉取成功" + break + else + echo "❌ 镜像拉取失败,重试 $i/3" + echo "检查网络和镜像仓库状态..." + + # 显示详细错误信息 + echo "--- 网络诊断信息 ---" + echo "DNS解析测试:" + nslookup "$REGISTRY_HOST" || echo "DNS解析失败" + echo "网络连通性测试:" + ping -c 2 "$REGISTRY_HOST" || echo "ping失败" + echo "Docker镜像仓库连接测试:" + curl -I "https://$REGISTRY_HOST/v2/" 2>/dev/null || echo "仓库API访问失败" + + sleep 5 + if [ $i -eq 3 ]; then + echo "❌ 镜像拉取失败,部署终止" + echo "请检查:" + echo "1. 网络连接是否正常" + echo "2. 镜像仓库是否可访问" + echo "3. 镜像标签是否存在" + echo "4. Docker登录凭据是否正确" + exit 1 + fi + fi + done + + # 安全停止和移除容器 + echo "检查现有容器状态..." + CONTAINER_NAME="ppanel-user-web" + + # 检查容器是否存在 + if docker ps -aq -f name=$CONTAINER_NAME | grep -q .; then + echo "发现现有容器,开始清理..." + + # 检查容器是否正在运行 + if docker ps -q -f name=$CONTAINER_NAME | grep -q .; then + echo "停止运行中的容器..." + docker stop $CONTAINER_NAME --time=15 || true + sleep 5 + fi + + # 检查容器是否正在被移除 + echo "检查容器移除状态..." + for i in {1..15}; do + # 尝试获取容器状态 + CONTAINER_STATUS=$(docker inspect $CONTAINER_NAME --format='{{.State.Status}}' 2>/dev/null || echo "not_found") + + if [ "$CONTAINER_STATUS" = "not_found" ]; then + echo "✅ 容器已不存在" + break + elif [ "$CONTAINER_STATUS" = "removing" ]; then + echo "⏳ 容器正在移除中,等待完成... $i/15" + sleep 3 + else + echo "尝试移除容器... $i/15" + if docker rm -f $CONTAINER_NAME 2>/dev/null; then + echo "✅ 容器移除成功" + break + else + echo "⚠️ 容器移除失败,重试..." + sleep 2 + fi + fi + + # 最后一次尝试强制清理 + if [ $i -eq 15 ]; then + echo "🔧 执行强制清理..." + docker kill $CONTAINER_NAME 2>/dev/null || true + sleep 2 + docker rm -f $CONTAINER_NAME 2>/dev/null || true + sleep 2 + fi + done + else + echo "✅ 未发现现有容器" + fi + + echo "启动新容器..." + docker run -d \ + --add-host api.airoport.co:103.150.215.40 \ + --name ppanel-user-web \ + --restart unless-stopped \ + -p 3002:3000 \ + -e NEXT_PUBLIC_API_URL=${{ env.NEXT_PUBLIC_API_URL }} \ + ${{ env.DOCKER_REGISTRY }}/ppanel/ppanel-user-web:${{ env.VERSION }} + + # 验证容器启动 + echo "验证容器启动状态..." + for i in {1..10}; do + if docker ps -q -f name=ppanel-user-web | grep -q .; then + echo "✅ 用户面板部署成功" + docker ps -f name=ppanel-user-web --format "table {{.Names}}\t{{.Status}}\t{{.Ports}}" + exit 0 + else + echo "等待容器启动... $i/10" + sleep 3 + fi + done + + echo "❌ 用户面板部署失败 - 容器未能正常启动" + docker logs ppanel-user-web || true + exit 1 + + # 步骤5: TG通知 (成功) + - name: 📱 发送成功通知到Telegram + if: success() + uses: appleboy/telegram-action@master + with: + token: ${{ env.TELEGRAM_BOT_TOKEN }} + to: ${{ env.TELEGRAM_CHAT_ID }} + message: | + ✅ 部署成功! + + 📦 项目: ${{ github.repository }} + 🌿 分支: ${{ github.ref_name }} + 🔖 版本: ${{ env.VERSION }} + 🎯 构建目标: ${{ env.BUILD_TARGET }} + 🔗 API地址: ${{ env.NEXT_PUBLIC_API_URL }} + 📝 提交: ${{ github.sha }} + 👤 提交者: ${{ github.actor }} + 🕐 时间: ${{ github.event.head_commit.timestamp }} + + 🚀 服务已成功部署到生产环境 + + # 步骤5: TG通知 (失败) + - name: 📱 发送失败通知到Telegram + if: failure() + uses: appleboy/telegram-action@master + with: + token: ${{ env.TELEGRAM_BOT_TOKEN }} + to: ${{ env.TELEGRAM_CHAT_ID }} + message: | + ❌ 部署失败! + + 📦 项目: ${{ github.repository }} + 🌿 分支: ${{ github.ref_name }} + 🔖 版本: ${{ env.VERSION }} + 🎯 构建目标: ${{ env.BUILD_TARGET }} + 📝 提交: ${{ github.sha }} + 👤 提交者: ${{ github.actor }} + 🕐 时间: ${{ github.event.head_commit.timestamp }} + + ⚠️ 请检查构建日志获取详细信息 diff --git a/package.json b/package.json index 4c7da53..031dc8e 100644 --- a/package.json +++ b/package.json @@ -1,14 +1,14 @@ { - "name": "ppanel-web", + "name": "fastvpn-web", "version": "1.6.0", "private": true, - "homepage": "https://github.com/perfect-panel/ppanel-web", + "homepage": "https://github.com/perfect-panel/fastvpn-web", "bugs": { - "url": "https://github.com/perfect-panel/ppanel-web/issues/new" + "url": "https://github.com/perfect-panel/fastvpn-web/issues/new" }, "repository": { "type": "git", - "url": "https://github.com/perfect-panel/ppanel-web.git" + "url": "https://github.com/perfect-panel/fastvpn-web.git" }, "license": "GUN", "workspaces": [