name: Build and Release on: push: tags: - 'v*' permissions: contents: write jobs: build: runs-on: ubuntu-latest steps: - name: Checkout repository uses: actions/checkout@v4 with: fetch-depth: 0 - name: Setup JDK 21 uses: actions/setup-java@v4 with: java-version: '21' distribution: 'temurin' - name: Make gradlew executable run: chmod +x ./gradlew - name: Run Forge data generation run: | echo "=== 运行 Forge 数据生成 ===" ./gradlew runData --no-daemon continue-on-error: false - name: Build with Gradle run: ./gradlew build --no-daemon - name: Prepare release files run: | mkdir -p release-files # 收集所有模块的构建产物 echo "=== 收集 common 模块构建产物 ===" if [ -d "common/build/libs" ]; then cp common/build/libs/*.jar release-files/ 2>/dev/null || echo "common 模块没有 jar 文件" fi echo "=== 收集 fabric 模块构建产物 ===" if [ -d "fabric/build/libs" ]; then cp fabric/build/libs/*-dev.jar release-files/ 2>/dev/null || true # 排除dev jar cp fabric/build/libs/*-sources.jar release-files/ 2>/dev/null || true cp fabric/build/libs/*-javadoc.jar release-files/ 2>/dev/null || true # 只复制主jar(没有sources/javadoc/dev classifier的jar) find fabric/build/libs -name "*.jar" ! -name "*-sources.jar" ! -name "*-javadoc.jar" ! -name "*-dev.jar" -exec cp {} release-files/ \; fi echo "=== 收集 forge 模块构建产物 ===" if [ -d "forge/build/libs" ]; then cp forge/build/libs/*-sources.jar release-files/ 2>/dev/null || true cp forge/build/libs/*-javadoc.jar release-files/ 2>/dev/null || true # 只复制主jar(没有sources/javadoc classifier的jar) find forge/build/libs -name "*.jar" ! -name "*-sources.jar" ! -name "*-javadoc.jar" -exec cp {} release-files/ \; fi echo "=== 准备发布的文件 ===" ls -la release-files/ - name: Upload release artifacts uses: actions/upload-artifact@v4 with: name: release-files path: release-files/ retention-days: 7 release: runs-on: ubuntu-latest needs: build if: startsWith(github.ref, 'refs/tags/v') steps: - name: Checkout with full history uses: actions/checkout@v4 with: fetch-depth: 0 - name: Download artifacts uses: actions/download-artifact@v4 with: name: release-files path: ./dist - name: Extract version info id: version_info run: | # 从tag中提取版本号(去掉v前缀) VERSION="${GITHUB_REF_NAME#v}" echo "version=$VERSION" >> $GITHUB_OUTPUT echo "minecraft_version=$(grep "^minecraft_version=" gradle.properties | cut -d'=' -f2)" >> $GITHUB_OUTPUT - name: Generate CZ-compliant changelog id: generate_changelog run: | CURRENT_TAG="${{ github.ref_name }}" PREV_TAG=$(git describe --tags --abbrev=0 $(git rev-list --tags --skip=1 --max-count=1) 2>/dev/null || echo "") # 创建临时文件 TEMP_FILE=$(mktemp) echo "# 🚀 版本 $CURRENT_TAG 发布" > $TEMP_FILE echo "" >> $TEMP_FILE echo "## 📋 变更摘要" >> $TEMP_FILE echo "" >> $TEMP_FILE if [ -z "$PREV_TAG" ]; then echo "### 初始版本发布" >> $TEMP_FILE echo "" >> $TEMP_FILE echo "这是项目的第一个正式版本。" >> $TEMP_FILE echo "" >> $TEMP_FILE # 获取所有提交并按类型分组 git log --pretty=format:"%s" --reverse | while read -r line; do echo "- $line" >> $TEMP_FILE done else echo "### 从 $PREV_TAG 到 $CURRENT_TAG 的变更" >> $TEMP_FILE echo "" >> $TEMP_FILE # 定义符合CZ规范的提交类型映射 declare -A commit_types commit_types=( ["✨ 新功能"]="^(feat|feature)(\(.*\))?:" ["🐛 修复"]="^(fix|bugfix)(\(.*\))?:" ["📝 文档"]="^(docs|documentation)(\(.*\))?:" ["🎨 样式"]="^(style)(\(.*\))?:" ["🔨 重构"]="^(refactor)(\(.*\))?:" ["⚡️ 性能"]="^(perf|performance)(\(.*\))?:" ["✅ 测试"]="^(test)(\(.*\))?:" ["🔧 构建"]="^(build)(\(.*\))?:" ["👷 CI"]="^(ci)(\(.*\))?:" ["📦 依赖"]="^(chore|deps)(\(.*\))?:" ["⏪ 回退"]="^(revert)(\(.*\))?:" ) # 获取所有提交 COMMITS=$(git log --pretty=format:"%s" $PREV_TAG..HEAD) # 处理每种类型的提交 for type_name in "${!commit_types[@]}"; do pattern="${commit_types[$type_name]}" # 提取匹配的提交 matched_commits=$(echo "$COMMITS" | grep -E "$pattern" || true) if [ -n "$matched_commits" ]; then echo "#### $type_name" >> $TEMP_FILE echo "" >> $TEMP_FILE # 处理每条提交,提取scope和subject echo "$matched_commits" | while read -r commit; do # 解析scope和subject if [[ $commit =~ ^[a-z]+\((.*)\):\ (.*) ]]; then scope="${BASH_REMATCH[1]}" subject="${BASH_REMATCH[2]}" echo "- **$scope**: $subject" >> $TEMP_FILE elif [[ $commit =~ ^[a-z]+:\ (.*) ]]; then subject="${BASH_REMATCH[1]}" echo "- $subject" >> $TEMP_FILE else echo "- $commit" >> $TEMP_FILE fi done echo "" >> $TEMP_FILE fi done # 处理破坏性变更(BREAKING CHANGE) breaking_changes=$(git log --pretty=format:"%b" $PREV_TAG..HEAD | grep -i "BREAKING CHANGE" || true) if [ -n "$breaking_changes" ]; then echo "#### ⚠️ 破坏性变更" >> $TEMP_FILE echo "" >> $TEMP_FILE echo "$breaking_changes" | while read -r line; do echo "- $line" >> $TEMP_FILE done echo "" >> $TEMP_FILE fi # 处理未分类的提交 uncategorized="$COMMITS" for pattern in "${commit_types[@]}"; do uncategorized=$(echo "$uncategorized" | grep -v -E "$pattern" || true) done if [ -n "$uncategorized" ]; then echo "#### 📝 其他更改" >> $TEMP_FILE echo "" >> $TEMP_FILE echo "$uncategorized" | while read -r commit; do echo "- $commit" >> $TEMP_FILE done echo "" >> $TEMP_FILE fi fi echo "## 📊 统计信息" >> $TEMP_FILE echo "" >> $TEMP_FILE if [ -z "$PREV_TAG" ]; then TOTAL_COMMITS=$(git rev-list --count HEAD) echo "- 总提交数: $TOTAL_COMMITS" >> $TEMP_FILE echo "- 首次发布" >> $TEMP_FILE else COMMITS=$(git rev-list --count $PREV_TAG..HEAD) echo "- 本次发布提交数: $COMMITS" >> $TEMP_FILE echo "- 上一个版本: $PREV_TAG" >> $TEMP_FILE fi echo "- 发布日期: $(date '+%Y年%m月%d日')" >> $TEMP_FILE echo "- 当前版本: $CURRENT_TAG" >> $TEMP_FILE echo "- Minecraft版本: ${{ steps.version_info.outputs.minecraft_version }}" >> $TEMP_FILE echo "" >> $TEMP_FILE echo "---" >> $TEMP_FILE echo "" >> $TEMP_FILE echo "## 📦 下载文件" >> $TEMP_FILE echo "" >> $TEMP_FILE echo "本次发布包含以下平台的构建文件:" >> $TEMP_FILE echo "" >> $TEMP_FILE # 列出所有发布的文件 echo "### Fabric 版本" >> $TEMP_FILE echo "\`\`\`" >> $TEMP_FILE ls -1 dist/*fabric*.jar 2>/dev/null | grep -v "sources\|javadoc" | xargs -n1 basename || echo "无 Fabric 主文件" >> $TEMP_FILE echo "\`\`\`" >> $TEMP_FILE echo "" >> $TEMP_FILE echo "### Forge 版本" >> $TEMP_FILE echo "\`\`\`" >> $TEMP_FILE ls -1 dist/*forge*.jar 2>/dev/null | grep -v "sources\|javadoc" | xargs -n1 basename || echo "无 Forge 主文件" >> $TEMP_FILE echo "\`\`\`" >> $TEMP_FILE echo "" >> $TEMP_FILE echo "### 通用模块" >> $TEMP_FILE echo "\`\`\`" >> $TEMP_FILE ls -1 dist/*common*.jar 2>/dev/null | grep -v "sources\|javadoc" | xargs -n1 basename || echo "无通用模块" >> $TEMP_FILE echo "\`\`\`" >> $TEMP_FILE echo "" >> $TEMP_FILE echo "### 源码和文档" >> $TEMP_FILE echo "\`\`\`" >> $TEMP_FILE ls -1 dist/*-sources.jar 2>/dev/null | xargs -n1 basename || echo "无源码文件" >> $TEMP_FILE ls -1 dist/*-javadoc.jar 2>/dev/null | xargs -n1 basename || echo "无文档文件" >> $TEMP_FILE echo "\`\`\`" >> $TEMP_FILE echo "" >> $TEMP_FILE echo "### 📜 详细提交历史" >> $TEMP_FILE echo "" >> $TEMP_FILE echo "
" >> $TEMP_FILE echo "点击展开查看完整提交历史" >> $TEMP_FILE echo "" >> $TEMP_FILE echo "\`\`\`" >> $TEMP_FILE # 显示所有提交的详细列表 if [ -z "$PREV_TAG" ]; then git log --pretty=format:"%h %s - %an (%ad)" --date=short --reverse >> $TEMP_FILE else git log --pretty=format:"%h %s - %an (%ad)" --date=short $PREV_TAG..HEAD >> $TEMP_FILE fi echo "\`\`\`" >> $TEMP_FILE echo "
" >> $TEMP_FILE # 将文件内容输出到变量 CHANGELOG_CONTENT=$(cat $TEMP_FILE) echo "changelog<> $GITHUB_OUTPUT echo "$CHANGELOG_CONTENT" >> $GITHUB_OUTPUT echo "EOF" >> $GITHUB_OUTPUT - name: Create Release uses: ncipollo/release-action@v1 with: artifacts: | dist/*.jar tag: ${{ github.ref_name }} name: "${{ steps.version_info.outputs.minecraft_version }} - ${{ github.ref_name }}" body: ${{ steps.generate_changelog.outputs.changelog }} draft: false prerelease: false token: ${{ secrets.GITHUB_TOKEN }} allowUpdates: true removeArtifacts: true