name: Epic UI Build & Deploy run-name: ${{ gitea.actor }} 正在构建 Epic UI 前端项目 🚀 on: push: branches: [ main, master, develop ] pull_request: branches: [ main, master ] jobs: build: runs-on: gitea_labels container: image: gitea-ci-bash:latest env: # 方法一,指定容器将工具缓存路径存放到 /toolcache ,该目录actRunner会默认持久化它 RUNNER_TOOL_CACHE: /toolcache steps: - name: 检出代码 shell: bash run: | echo "📥 检出代码到工作目录..." echo "当前目录: $(pwd)" echo "仓库: ${{ gitea.repository }}" echo "分支: ${{ gitea.ref }}" echo "提交: ${{ gitea.sha }}" # 解析分支名 BRANCH_NAME=$(echo "${{ gitea.ref }}" | sed 's#refs/heads/##') # 拉取代码(使用token鉴权) git clone --depth=1 -b "$BRANCH_NAME" "http://1c18ee1ab9a9cb291506d0c5c016a33be7d59e8c:x-oauth-basic@gitea.htoop.cn/${{ gitea.repository }}.git" . if [ $? -ne 0 ]; then echo "❌ 代码检出失败" exit 1 fi # 校验分支 CURRENT_BRANCH=$(git rev-parse --abbrev-ref HEAD) if [ "$CURRENT_BRANCH" != "$BRANCH_NAME" ]; then echo "❌ 检出分支与预期不符: $CURRENT_BRANCH != $BRANCH_NAME" exit 1 fi # 校验提交 CURRENT_COMMIT=$(git rev-parse HEAD) if [ -n "${{ gitea.sha }}" ] && [ "$CURRENT_COMMIT" != "${{ gitea.sha }}" ]; then echo "❌ 检出提交与预期不符: $CURRENT_COMMIT != ${{ gitea.sha }}" exit 1 fi echo "✅ 代码检出成功,分支和提交校验通过" ls -la - name: 安装Node.js环境 shell: bash run: | echo "🔧 安装Node.js环境..." if command -v node &> /dev/null; then echo "✅ Node.js已安装: $(node --version)" echo "✅ npm已安装: $(npm --version)" else echo "📥 下载并安装Node.js..." NODE_VERSION="18.19.0" NODE_ARCH="linux-x64" MIRRORS=( "https://mirrors.aliyun.com/nodejs-release/v${NODE_VERSION}/node-v${NODE_VERSION}-${NODE_ARCH}.tar.xz" "https://nodejs.org/dist/v${NODE_VERSION}/node-v${NODE_VERSION}-${NODE_ARCH}.tar.xz" ) DOWNLOAD_SUCCESS=false for mirror in "${MIRRORS[@]}"; do echo "尝试从镜像下载: $mirror" if wget -q --timeout=30 --tries=3 "$mirror" -O node.tar.xz; then echo "✅ 下载成功: $mirror" DOWNLOAD_SUCCESS=true break else echo "❌ 下载失败: $mirror" continue fi done if [ "$DOWNLOAD_SUCCESS" = false ]; then echo "❌ 所有镜像源下载失败" exit 1 fi echo "解压Node.js到/usr/local..." tar -C /usr/local -xJf node.tar.xz --strip-components=1 export PATH=$PATH:/usr/local/bin echo "验证Node.js安装..." node --version npm --version rm node.tar.xz echo "✅ Node.js安装完成" fi - name: 检查Node.js环境 shell: bash run: | echo "🔍 检查Node.js环境..." export PATH=$PATH:/usr/local/bin node --version npm --version echo "Node.js路径: $(which node)" echo "npm路径: $(which npm)" - name: 安装pnpm shell: bash run: | echo "📦 安装pnpm包管理器..." export PATH=$PATH:/usr/local/bin if command -v pnpm &> /dev/null; then echo "✅ pnpm已安装: $(pnpm --version)" else echo "📥 安装pnpm..." npm install -g pnpm echo "✅ pnpm安装完成: $(pnpm --version)" fi - name: 检查缓存目录权限 shell: bash run: | echo "🔍 检查缓存目录权限和状态..." echo "=== /toolcache 目录检查 ===" ls -la /toolcache/ 2>/dev/null || echo "/toolcache 目录不存在或无法访问" echo "=== /shared/cache 目录检查 ===" ls -la /shared/cache/ 2>/dev/null || echo "/shared/cache 目录不存在或无法访问" echo "=== /tmp 目录检查 ===" ls -la /tmp/ | head -10 echo "=== 当前工作目录 ===" pwd ls -la - name: 恢复依赖缓存 shell: bash run: | echo "📦 检查并恢复依赖缓存..." # 生成缓存键 CACHE_KEY=$(md5sum pnpm-lock.yaml | cut -d' ' -f1) echo "缓存键: $CACHE_KEY" # 定义多个可能的缓存位置 CACHE_LOCATIONS=( "/toolcache/node_modules_${CACHE_KEY}.tar.gz" "/shared/cache/node_modules_${CACHE_KEY}.tar.gz" "/tmp/node_modules_${CACHE_KEY}.tar.gz" "/var/cache/node_modules_${CACHE_KEY}.tar.gz" ) CACHE_RESTORED=false for cache_file in "${CACHE_LOCATIONS[@]}"; do echo "🔍 检查缓存文件: $cache_file" if [ -f "$cache_file" ]; then echo "✅ 找到缓存文件: $cache_file" echo "📦 缓存文件大小: $(du -sh "$cache_file" | cut -f1)" echo "正在恢复缓存..." if tar -xzf "$cache_file"; then echo "✅ 缓存恢复成功" CACHE_RESTORED=true break else echo "❌ 缓存恢复失败: $cache_file" continue fi fi done if [ "$CACHE_RESTORED" = false ]; then echo "📥 未找到有效缓存,将重新安装依赖" else echo "📦 恢复的 node_modules 大小:" du -sh node_modules 2>/dev/null || echo "node_modules 目录不存在" fi - name: 安装依赖 shell: bash run: | echo "📦 开始安装依赖..." export PATH=$PATH:/usr/local/bin echo "开始时间: $(date)" # 检查是否已有node_modules if [ -d "node_modules" ] && [ -f "node_modules/.pnpm-debug.log" ]; then echo "✅ 检测到已存在的node_modules,跳过安装" else echo "📥 执行pnpm install..." pnpm install --frozen-lockfile echo "✅ 依赖安装完成" fi echo "📦 node_modules 目录大小:" du -sh node_modules 2>/dev/null || echo "node_modules 目录不存在" - name: 保存依赖缓存 shell: bash run: | echo "💾 保存依赖缓存..." # 生成缓存键 CACHE_KEY=$(md5sum pnpm-lock.yaml | cut -d' ' -f1) echo "缓存键: $CACHE_KEY" # 检查node_modules是否存在 if [ ! -d "node_modules" ]; then echo "❌ node_modules 目录不存在,跳过缓存保存" exit 0 fi # 尝试多个缓存位置 CACHE_LOCATIONS=( "/toolcache" "/shared/cache" "/tmp" "/var/cache" ) CACHE_SAVED=false for cache_dir in "${CACHE_LOCATIONS[@]}"; do echo "🔍 尝试保存到: $cache_dir" if [ -w "$cache_dir" ] 2>/dev/null; then mkdir -p "$cache_dir" CACHE_FILE="$cache_dir/node_modules_${CACHE_KEY}.tar.gz" echo "📦 正在创建缓存文件: $CACHE_FILE" if tar -czf "$CACHE_FILE" node_modules; then echo "✅ 缓存已保存到: $CACHE_FILE" echo "📦 缓存文件大小: $(du -sh "$CACHE_FILE" | cut -f1)" CACHE_SAVED=true # 显示缓存目录内容 echo "🔍 缓存目录内容:" ls -la "$cache_dir/" | head -10 break else echo "❌ 保存到 $cache_dir 失败" continue fi else echo "❌ 目录 $cache_dir 不可写" fi done if [ "$CACHE_SAVED" = false ]; then echo "⚠️ 警告:所有缓存位置都无法保存,缓存功能可能不可用" fi echo "⏰ 等待30秒,让你检查缓存是否成功写入..." echo "🕐 当前时间: $(date)" echo "📊 缓存状态检查:" # 每5秒检查一次缓存状态,持续30秒 for i in {1..6}; do echo "=== 第 $i 次检查 (第 $((i*5)) 秒) ===" echo "时间: $(date)" for cache_dir in "${CACHE_LOCATIONS[@]}"; do if [ -d "$cache_dir" ]; then echo "📁 $cache_dir 目录内容:" ls -la "$cache_dir/" 2>/dev/null | head -5 || echo "无法访问 $cache_dir" CACHE_FILE="$cache_dir/node_modules_${CACHE_KEY}.tar.gz" if [ -f "$CACHE_FILE" ]; then echo "✅ 缓存文件存在: $CACHE_FILE" echo "📦 文件大小: $(du -sh "$CACHE_FILE" 2>/dev/null | cut -f1 || echo '无法获取大小')" echo "📅 修改时间: $(stat -c %y "$CACHE_FILE" 2>/dev/null || echo '无法获取时间')" else echo "❌ 缓存文件不存在: $CACHE_FILE" fi else echo "❌ 目录不存在: $cache_dir" fi echo "---" done if [ $i -lt 6 ]; then echo "⏳ 等待5秒..." sleep 5 fi done echo "✅ 30秒等待完成,缓存检查结束" echo "🕐 结束时间: $(date)" - name: 验证依赖 shell: bash run: | echo "🔍 验证项目依赖..." export PATH=$PATH:/usr/local/bin pnpm list --depth=0 echo "✅ 依赖验证完成" - name: 构建项目 shell: bash run: | echo "🔨 开始构建 Epic UI 前端项目..." export PATH=$PATH:/usr/local/bin pnpm build echo "✅ 构建完成" - name: 检查构建产物 shell: bash run: | echo "📦 构建产物信息:" if [ -d "dist" ]; then echo "✅ 找到dist目录" ls -la dist/ echo "dist目录大小: $(du -sh dist | cut -f1)" else echo "❌ 未找到dist目录" exit 1 fi - name: 显示项目信息 shell: bash run: | echo "📋 项目信息:" export PATH=$PATH:/usr/local/bin echo "Node.js 版本: $(node --version)" echo "npm 版本: $(npm --version)" echo "pnpm 版本: $(pnpm --version)" echo "构建时间: $(date)" echo "分支: ${{ gitea.ref }}" echo "提交: ${{ gitea.sha }}" - name: 准备部署 shell: bash run: | echo "🚀 准备部署..." echo "运行器操作系统: ${{ runner.os }}" echo "工作目录: ${{ gitea.workspace }}" echo "任务状态: ${{ job.status }}" mkdir -p deploy cp -r dist/* deploy/ echo "📦 部署包已准备完成" # - name: 复制构建产物到目标目录 # shell: bash # run: | # echo "📤 复制构建产物到 /opt/dockeropt/epic-ui ..." # mkdir -p /opt/dockeropt/epic-ui # cp -r dist/* /opt/dockeropt/epic-ui/ # echo "✅ 构建产物已复制到 /opt/dockeropt/epic-ui" # ls -la /opt/dockeropt/epic-ui/ # - name: 重启 Docker 容器 # shell: bash # run: | # echo "♻️ 重启 epic-ui 容器..." # if ! command -v docker &> /dev/null; then # if command -v apt-get &> /dev/null; then # apt-get update && apt-get install -y docker.io # elif command -v apk &> /dev/null; then # echo "http://mirrors.aliyun.com/alpine/v3.20/main" > /etc/apk/repositories # echo "http://mirrors.aliyun.com/alpine/v3.20/community" >> /etc/apk/repositories # apk update # apk add --no-cache docker-cli # elif command -v yum &> /dev/null; then # yum install -y docker # fi # fi # docker ps -a # docker restart epic-ui # echo "✅ epic-ui 容器已重启"