From ccf9d0d1d5e85ca929a284174d6151a233741f8f Mon Sep 17 00:00:00 2001 From: Alexander01998 Date: Tue, 19 Nov 2024 17:18:27 +0100 Subject: [PATCH 1/3] Update GHA workflows --- .github/workflows/dependency_graph.yml | 40 +++++++++++++ .github/workflows/gradle.yml | 79 +++++++++++++++++++------- .github/workflows/jsonsyntax.yml | 20 ------- .github/workflows/stale.yml | 4 +- 4 files changed, 100 insertions(+), 43 deletions(-) create mode 100644 .github/workflows/dependency_graph.yml delete mode 100644 .github/workflows/jsonsyntax.yml diff --git a/.github/workflows/dependency_graph.yml b/.github/workflows/dependency_graph.yml new file mode 100644 index 0000000..4d2765b --- /dev/null +++ b/.github/workflows/dependency_graph.yml @@ -0,0 +1,40 @@ +name: Update Gradle Dependency Graph + +on: + push: + branches: + # Submitting dependency graph reports on non-default branches does nothing + - "master" + tags-ignore: + - "**" + paths: + - "gradle**" + - "*.gradle" + workflow_dispatch: + +permissions: + contents: write + +jobs: + dependency-submission: + runs-on: ubuntu-latest + steps: + + - name: Checkout repository + uses: actions/checkout@v4 + + - name: Set up Java 21 + uses: actions/setup-java@v4 + with: + java-version: "21" + distribution: "microsoft" + + - name: Grant execute permission for gradlew + run: chmod +x gradlew + + - name: Generate and submit dependency graph + uses: gradle/actions/dependency-submission@v4 + with: + build-scan-publish: true + build-scan-terms-of-use-url: "https://gradle.com/help/legal-terms-of-use" + build-scan-terms-of-use-agree: "yes" diff --git a/.github/workflows/gradle.yml b/.github/workflows/gradle.yml index ce698ca..c01d84e 100644 --- a/.github/workflows/gradle.yml +++ b/.github/workflows/gradle.yml @@ -2,36 +2,41 @@ name: Java CI with Gradle on: push: + branches: + - "**" + tags-ignore: + - "**" paths: - - '**.java' - - '**.json' - - 'gradle**' - - 'build.gradle' + - "**.java" + - "**.json" + - "**.yml" + - "gradle**" + - "*.gradle" pull_request: paths: - - '**.java' - - '**.json' - - 'gradle**' - - 'build.gradle' - # Makes it possible to run this workflow manually from the Actions tab + - "**.java" + - "**.json" + - "**.yml" + - "gradle**" + - "*.gradle" workflow_dispatch: -permissions: - contents: write - jobs: build: runs-on: ubuntu-latest - + env: + VIRUSTOTAL_API_KEY: ${{ secrets.VIRUSTOTAL_API_KEY }} + IMGUR_CLIENT_ID: ${{ secrets.IMGUR_CLIENT_ID }} steps: + - name: Checkout repository uses: actions/checkout@v4 - name: Set up Java 21 uses: actions/setup-java@v4 with: - java-version: '21' - distribution: 'microsoft' + java-version: "21" + distribution: "microsoft" - name: Grant execute permission for gradlew run: chmod +x gradlew @@ -45,18 +50,48 @@ jobs: # Enable cache writing for NeoForge branches, since they don't benefit from the Fabric cache on master cache-read-only: ${{ github.ref != 'refs/heads/master' && !contains(github.ref, 'neoforge') }} - - name: Generate and submit dependency graph - if: ${{ github.event_name == 'push' }} - uses: gradle/actions/dependency-submission@v4 + - name: Compile Java code + run: ./gradlew remapJar --stacktrace --warning-mode=fail + + - name: Validate JSON files + run: ./gradlew spotlessJsonCheck || (echo "::error::JSON validation failed! Run './gradlew spotlessApply' to fix style issues, or check the full error message for syntax errors." && exit 1) - - name: Run tests and build + - name: Validate Java code style + run: ./gradlew spotlessCheck || (echo "::error::Java code style validation failed! To fix, run 'Clean Up' and then 'Format' in Eclipse, or './gradlew spotlessApply' in the terminal." && exit 1) + + - name: Run unit tests + run: ./gradlew test --stacktrace --warning-mode=fail + + - name: Validate access widener + run: ./gradlew validateAccessWidener --stacktrace --warning-mode=fail + + - name: Build run: ./gradlew build --stacktrace --warning-mode=fail - - name: VirusTotal scan - if: ${{ github.event_name == 'push' }} + - name: Upload to VirusTotal for analysis + id: virustotal + if: ${{ env.VIRUSTOTAL_API_KEY }} uses: crazy-max/ghaction-virustotal@v4 with: - vt_api_key: ${{ secrets.VIRUSTOTAL_API_KEY }} + vt_api_key: ${{ env.VIRUSTOTAL_API_KEY }} files: | ./build/libs/*.jar + # An error in this step means that the upload failed, not that a false + # positive was detected. continue-on-error: true + + - name: Add VirusTotal links to build summary + if: ${{ env.VIRUSTOTAL_API_KEY && steps.virustotal.outputs.analysis }} + shell: bash + run: | + echo "
" >> $GITHUB_STEP_SUMMARY + echo "🛡️ VirusTotal Scans" >> $GITHUB_STEP_SUMMARY + echo "" >> $GITHUB_STEP_SUMMARY + IFS=',' read -ra ANALYSIS <<< "${{ steps.virustotal.outputs.analysis }}" + for i in "${ANALYSIS[@]}"; do + filepath=${i%%=*} + url=${i#*=} + filename=$(basename "$filepath") + echo "- [$filename]($url)" >> $GITHUB_STEP_SUMMARY + done + echo "
" >> $GITHUB_STEP_SUMMARY diff --git a/.github/workflows/jsonsyntax.yml b/.github/workflows/jsonsyntax.yml deleted file mode 100644 index e6f9a4d..0000000 --- a/.github/workflows/jsonsyntax.yml +++ /dev/null @@ -1,20 +0,0 @@ -name: JSON syntax - -on: - push: - paths: - - '**.json' - pull_request: - paths: - - '**.json' - -jobs: - test: - runs-on: ubuntu-latest - steps: - - name: Checkout repository - uses: actions/checkout@v4 - - name: Check JSON syntax - uses: limitusus/json-syntax-check@v2 - with: - pattern: "\\.json$" diff --git a/.github/workflows/stale.yml b/.github/workflows/stale.yml index 3218d53..e0a1612 100644 --- a/.github/workflows/stale.yml +++ b/.github/workflows/stale.yml @@ -1,4 +1,5 @@ -name: "Close stale issues and pull requests" +name: Close stale issues and pull requests + on: schedule: - cron: "30 1 * * 1-5" @@ -11,6 +12,7 @@ jobs: stale: runs-on: ubuntu-latest steps: + - uses: actions/stale@v9 with: stale-issue-message: | From 8bfd53e4a9b313bbfef9daee6a2e446cedf0b90d Mon Sep 17 00:00:00 2001 From: Alexander01998 Date: Tue, 19 Nov 2024 17:24:13 +0100 Subject: [PATCH 2/3] Add publish workflow --- .github/workflows/publish.yml | 87 ++++++++++++++++++++++++++++++++ build.gradle | 95 +++++++++++++++++++++++++++++++---- 2 files changed, 172 insertions(+), 10 deletions(-) create mode 100644 .github/workflows/publish.yml diff --git a/.github/workflows/publish.yml b/.github/workflows/publish.yml new file mode 100644 index 0000000..8c38cf5 --- /dev/null +++ b/.github/workflows/publish.yml @@ -0,0 +1,87 @@ +name: Publish Release + +on: + workflow_dispatch: + inputs: + close_milestone: + description: "Close milestone" + required: true + type: boolean + default: true + upload_backups: + description: "Upload to backups server" + required: true + type: boolean + default: true + publish_github: + description: "Publish to GitHub" + required: true + type: boolean + default: true + publish_curseforge: + description: "Publish to CurseForge" + required: true + type: boolean + default: true + publish_modrinth: + description: "Publish to Modrinth" + required: true + type: boolean + default: true + +jobs: + publish: + runs-on: ubuntu-latest + env: + WI_BACKUPS_API_KEY: ${{ secrets.WI_BACKUPS_API_KEY }} + GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }} + CURSEFORGE_API_KEY: ${{ secrets.CURSEFORGE_API_KEY }} + MODRINTH_TOKEN: ${{ secrets.MODRINTH_TOKEN }} + + steps: + - uses: actions/checkout@v4 + + - name: Set up Java 21 + uses: actions/setup-java@v4 + with: + java-version: "21" + distribution: "microsoft" + + - name: Grant execute permission for gradlew + run: chmod +x gradlew + + - name: Setup Gradle + uses: gradle/actions/setup-gradle@v4 + + - name: Build + run: ./gradlew build --stacktrace --warning-mode=fail + + - name: Create and push tag + run: | + MOD_VERSION=$(grep "mod_version" gradle.properties | cut -d'=' -f2 | tr -d ' ') + git config --global user.name "Wurst-Bot" + git config --global user.email "contact.wurstimperium@gmail.com" + git tag $MOD_VERSION + git push origin $MOD_VERSION + + - name: Close milestone + if: ${{ inputs.close_milestone }} + run: ./gradlew closeMilestone --stacktrace + + - name: Upload backups + if: ${{ inputs.upload_backups }} + run: ./gradlew uploadBackups --stacktrace + + - name: Publish to GitHub + if: ${{ inputs.publish_github }} + env: + GITHUB_TOKEN: ${{ secrets.MCX_PUBLISH_TOKEN }} + run: ./gradlew github --stacktrace + + - name: Publish to CurseForge + if: ${{ inputs.publish_curseforge }} + run: ./gradlew publishCurseforge --stacktrace + + - name: Publish to Modrinth + if: ${{ inputs.publish_modrinth }} + run: ./gradlew publishModrinth --stacktrace diff --git a/build.gradle b/build.gradle index efe5cfb..cb6a8b8 100644 --- a/build.gradle +++ b/build.gradle @@ -1,6 +1,6 @@ buildscript { dependencies { - classpath 'org.kohsuke:github-api:1.326' + classpath "org.kohsuke:github-api:1.326" } } @@ -72,9 +72,9 @@ java { jar { from("LICENSE") { - rename { "${it}_${project.base.archivesName.get()}"} + rename {"${it}_${base.archivesName.get()}"} } - + exclude("/assets/mo_glass/textures/block/*.psd") } @@ -92,17 +92,25 @@ spotless { } } +def getGhVersion() { + return "v" + version.substring(0, version.indexOf("-MC")) +} + +def getChangelogUrl() { + def modSlug = archives_base_name.toLowerCase() + def versionSlug = version.substring(0, version.indexOf("-MC")).replace(".", "-") + return "https://www.wimods.net/${modSlug}/${modSlug}-${versionSlug}/" +} + publishMods { file = remapJar.archiveFile - def versionString = project.version as String - def ghVersion = "v" + versionString.substring(0, versionString.indexOf("-")) - def changelogUrl = "https://www.wimods.net/mo-glass/mo-glass-1-10/" - def archivesName = project.base.archivesName.get() as String + def archivesName = base.archivesName.get() as String additionalFiles.from( - file("${project.buildDir}/libs/${archivesName}-${versionString}-sources.jar"), + file("${project.buildDir}/libs/${archivesName}-${version}-sources.jar"), ) - type = ghVersion.contains("pre") ? BETA : STABLE + type = getGhVersion().contains("pre") ? BETA : STABLE modLoaders.add("fabric") + def changelogUrl = getChangelogUrl() curseforge { projectId = "353426" @@ -137,7 +145,7 @@ task github(dependsOn: build) { doLast { def github = GitHub.connectUsingOAuth(ENV.GITHUB_TOKEN as String) def repository = github.getRepository("Wurst-Imperium-MCX/Mo-Glass") - def ghVersion = "v" + version.substring(0, version.indexOf("-")) + def ghVersion = getGhVersion() def ghRelease = repository.getReleaseByTagName(ghVersion as String) if(ghRelease == null) { @@ -149,3 +157,70 @@ task github(dependsOn: build) { ghRelease.uploadAsset(remapSourcesJar.archiveFile.get().getAsFile(), "application/java-archive") } } + +import java.time.LocalDate +import org.kohsuke.github.GHIssueState +import org.kohsuke.github.GHMilestoneState +import java.time.ZoneId + +task closeMilestone { + onlyIf { + ENV.GITHUB_TOKEN + } + + doLast { + def github = GitHub.connectUsingOAuth(ENV.GITHUB_TOKEN as String) + def repository = github.getRepository("Wurst-Imperium/Mo-Glass") + def ghVersion = getGhVersion() + + // Weird API design: listMilestones() requires GHIssueState while everything else uses GHMilestoneState. + def milestone = repository.listMilestones(GHIssueState.ALL).find { it.title == ghVersion } + if (milestone == null) { + milestone = repository.createMilestone(ghVersion, "") + } + + if (milestone.getState() != GHMilestoneState.CLOSED) { + milestone.setDueOn(Date.from(LocalDate.now().atStartOfDay(ZoneId.systemDefault()).toInstant())) + milestone.setDescription(getChangelogUrl()) + milestone.close() + } + } +} + +task uploadBackups { + dependsOn build + + onlyIf { + ENV.WI_BACKUPS_API_KEY + } + + doLast { + def shortVersion = getGhVersion().substring(1) + def backupUrl = "https://api.wurstclient.net/artifact-backups/Mo-Glass/${shortVersion}" + + def connection = new URL(backupUrl).openConnection() as HttpURLConnection + def boundary = UUID.randomUUID().toString() + connection.setRequestMethod("POST") + connection.setRequestProperty("X-API-Key", ENV.WI_BACKUPS_API_KEY) + connection.setRequestProperty("Accept", "application/json") + connection.setRequestProperty("Content-Type", "multipart/form-data; boundary=$boundary") + connection.doOutput = true + + def output = connection.outputStream + [remapJar, remapSourcesJar].each { jarTask -> + def file = jarTask.archiveFile.get().asFile + output << "--${boundary}\r\n" + output << "Content-Disposition: form-data; name=\"files\"; filename=\"${file.name}\"\r\n" + output << "Content-Type: application/java-archive\r\n\r\n" + file.withInputStream { input -> + output << input + } + output << "\r\n" + } + output << "--${boundary}--\r\n" + output.flush() + + if(connection.responseCode != 200) + throw new GradleException("Failed to upload backups: ${connection.responseCode} ${connection.responseMessage}") + } +} From ac09361c014f72b56254ae1bf2a1e109ea0a1ed2 Mon Sep 17 00:00:00 2001 From: Alexander01998 Date: Wed, 20 Nov 2024 16:26:25 +0100 Subject: [PATCH 3/3] Fix missing v in auto-generated tag name --- .github/workflows/publish.yml | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/.github/workflows/publish.yml b/.github/workflows/publish.yml index 8c38cf5..5aa3ade 100644 --- a/.github/workflows/publish.yml +++ b/.github/workflows/publish.yml @@ -61,8 +61,8 @@ jobs: MOD_VERSION=$(grep "mod_version" gradle.properties | cut -d'=' -f2 | tr -d ' ') git config --global user.name "Wurst-Bot" git config --global user.email "contact.wurstimperium@gmail.com" - git tag $MOD_VERSION - git push origin $MOD_VERSION + git tag v$MOD_VERSION + git push origin v$MOD_VERSION - name: Close milestone if: ${{ inputs.close_milestone }}