mirror of
				https://github.com/gradle/gradle-build-action.git
				synced 2025-10-23 10:28:56 +08:00 
			
		
		
		
	Compare commits
	
		
			90 Commits
		
	
	
		
			v2
			...
			v3.0.0-rc.
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
|  | d4bd442194 | ||
|  | 4587492cf1 | ||
|  | ef85c4ed42 | ||
|  | 7e08fe288b | ||
|  | 4a8703fa34 | ||
|  | 4a39eedb8c | ||
|  | 272883a7ba | ||
|  | 2a8bfcf231 | ||
|  | e1ada08a9a | ||
|  | a8e3e5e2b4 | ||
|  | 2be01ca1c6 | ||
|  | a00827eebb | ||
|  | ad80850e98 | ||
|  | bd6d0a74d4 | ||
|  | 1b6cac1f97 | ||
|  | 90d7c1a069 | ||
|  | 4062866f05 | ||
|  | 83a95864e5 | ||
|  | 60c43cb563 | ||
|  | 75b3db10df | ||
|  | f1361c71c2 | ||
|  | 49ade81b5d | ||
|  | 79fa674432 | ||
|  | 46878035be | ||
|  | 42452daeb5 | ||
|  | 346645706f | ||
|  | 5516b39940 | ||
|  | 7099569988 | ||
|  | 610728fa8c | ||
|  | a835cbb991 | ||
|  | ee4d92bb22 | ||
|  | 173b6ae553 | ||
|  | a01f794d92 | ||
|  | 369fcc54d8 | ||
|  | 6523a87c8f | ||
|  | 11693a1169 | ||
|  | 0e6b90783e | ||
|  | 932abbbe13 | ||
|  | 1a18d0b2d3 | ||
|  | 7af89832c5 | ||
|  | b5ebb0cc96 | ||
|  | 3a75647ad4 | ||
|  | 4dda5928c7 | ||
|  | 6c7537229b | ||
|  | 650620f9f9 | ||
|  | d4e24dfc10 | ||
|  | 7c57ba1136 | ||
|  | 32bab5b15a | ||
|  | 270f30ba56 | ||
|  | c00a847f3f | ||
|  | e2aa3f332c | ||
|  | ff5d63e9dc | ||
|  | 4faac45dc8 | ||
|  | 45ed60450e | ||
|  | 72abd931ce | ||
|  | a4107da76d | ||
|  | d16a3f4093 | ||
|  | 333078158e | ||
|  | 34a07dced0 | ||
|  | 24e9e9dc6b | ||
|  | bc72ac9e9d | ||
|  | c791d32284 | ||
|  | ceb0c736c1 | ||
|  | 3f2ca32cb0 | ||
|  | 5d2dd0dea4 | ||
|  | e865911745 | ||
|  | 0538e78c32 | ||
|  | a4dabb3a70 | ||
|  | 51b7a82e8e | ||
|  | 92cb3fe7e8 | ||
|  | e843ea4565 | ||
|  | 14b4921945 | ||
|  | 9c7269b85b | ||
|  | a7b743845f | ||
|  | e6566cde89 | ||
|  | 89f8dcd819 | ||
|  | 93050d1483 | ||
|  | df38ec05e0 | ||
|  | b3f092e821 | ||
|  | 63ce84df08 | ||
|  | 57f3f23714 | ||
|  | 38785d7d62 | ||
|  | a738af78ea | ||
|  | ae24bf6608 | ||
|  | 334a4b8d4d | ||
|  | 009bd36b91 | ||
|  | 9d6738618d | ||
|  | f053e6b7e7 | ||
|  | c821b7c4f1 | ||
|  | 89e46180c6 | 
| @@ -1,3 +0,0 @@ | |||||||
| dist/ |  | ||||||
| lib/ |  | ||||||
| node_modules/ |  | ||||||
| @@ -1,56 +0,0 @@ | |||||||
| { |  | ||||||
|     "plugins": ["jest", "@typescript-eslint"], |  | ||||||
|     "extends": ["plugin:github/recommended"], |  | ||||||
|     "parser": "@typescript-eslint/parser", |  | ||||||
|     "parserOptions": { |  | ||||||
|       "ecmaVersion": 9, |  | ||||||
|       "sourceType": "module", |  | ||||||
|       "project": "./tsconfig.json" |  | ||||||
|     }, |  | ||||||
|     "rules": { |  | ||||||
|       "eslint-comments/no-use": "off", |  | ||||||
|       "import/no-namespace": "off", |  | ||||||
|       "i18n-text/no-en": "off", |  | ||||||
|       "no-unused-vars": "off", |  | ||||||
|       "no-shadow": "off", |  | ||||||
|       "sort-imports": "off", |  | ||||||
|       "@typescript-eslint/no-unused-vars": ["error", { "argsIgnorePattern": "^_" }], |  | ||||||
|       "@typescript-eslint/explicit-member-accessibility": ["error", {"accessibility": "no-public"}], |  | ||||||
|       "@typescript-eslint/no-require-imports": "error", |  | ||||||
|       "@typescript-eslint/array-type": "error", |  | ||||||
|       "@typescript-eslint/await-thenable": "error", |  | ||||||
|       "camelcase": "off", |  | ||||||
|       "@typescript-eslint/explicit-function-return-type": ["error", {"allowExpressions": true}], |  | ||||||
|       "@typescript-eslint/func-call-spacing": ["error", "never"], |  | ||||||
|       "@typescript-eslint/no-array-constructor": "error", |  | ||||||
|       "@typescript-eslint/no-empty-interface": "error", |  | ||||||
|       "@typescript-eslint/no-explicit-any": "error", |  | ||||||
|       "@typescript-eslint/no-extraneous-class": "error", |  | ||||||
|       "@typescript-eslint/no-for-in-array": "error", |  | ||||||
|       "@typescript-eslint/no-inferrable-types": "error", |  | ||||||
|       "@typescript-eslint/no-misused-new": "error", |  | ||||||
|       "@typescript-eslint/no-namespace": "error",  |  | ||||||
|       "@typescript-eslint/no-non-null-assertion": "off", |  | ||||||
|       "@typescript-eslint/no-shadow": "error", |  | ||||||
|       "@typescript-eslint/no-unnecessary-qualifier": "error", |  | ||||||
|       "@typescript-eslint/no-unnecessary-type-assertion": "error", |  | ||||||
|       "@typescript-eslint/no-useless-constructor": "error", |  | ||||||
|       "@typescript-eslint/no-var-requires": "error", |  | ||||||
|       "@typescript-eslint/prefer-for-of": "warn", |  | ||||||
|       "@typescript-eslint/prefer-function-type": "warn", |  | ||||||
|       "@typescript-eslint/prefer-includes": "error", |  | ||||||
|       "@typescript-eslint/prefer-string-starts-ends-with": "error", |  | ||||||
|       "@typescript-eslint/promise-function-async": "error", |  | ||||||
|       "@typescript-eslint/require-array-sort-compare": ["error", {"ignoreStringArrays":  true}], |  | ||||||
|       "@typescript-eslint/restrict-plus-operands": "error", |  | ||||||
|       "semi": "off", |  | ||||||
|       "@typescript-eslint/semi": ["error", "never"], |  | ||||||
|       "@typescript-eslint/type-annotation-spacing": "error", |  | ||||||
|       "@typescript-eslint/unbound-method": "error" |  | ||||||
|     }, |  | ||||||
|     "env": { |  | ||||||
|       "node": true, |  | ||||||
|       "es6": true, |  | ||||||
|       "jest/globals": true |  | ||||||
|     } |  | ||||||
|   } |  | ||||||
							
								
								
									
										16
									
								
								.github/actions/build-dist/action.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										16
									
								
								.github/actions/build-dist/action.yml
									
									
									
									
										vendored
									
									
								
							| @@ -1,16 +0,0 @@ | |||||||
| name: 'Build and upload distribution' |  | ||||||
| # Builds the action distribution an uploads as an artifact for later download |  | ||||||
| runs: |  | ||||||
|   using: "composite" |  | ||||||
|   steps:  |  | ||||||
|     - name: Build distribution |  | ||||||
|       shell: bash |  | ||||||
|       run: | |  | ||||||
|         npm install |  | ||||||
|         npm run build |  | ||||||
|     - name: Upload distribution |  | ||||||
|       uses: actions/upload-artifact@v3 |  | ||||||
|       with: |  | ||||||
|         name: dist |  | ||||||
|         path: dist/ |  | ||||||
|  |  | ||||||
							
								
								
									
										12
									
								
								.github/actions/download-dist/action.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										12
									
								
								.github/actions/download-dist/action.yml
									
									
									
									
										vendored
									
									
								
							| @@ -1,12 +0,0 @@ | |||||||
| name: 'Download dist' |  | ||||||
| # Downloads a 'dist' directory artifact that was uploaded in an earlier step |  | ||||||
| # We control this with an environment variable to allow for easier global configuration. |  | ||||||
| runs: |  | ||||||
|   using: "composite" |  | ||||||
|   steps:  |  | ||||||
|     - name: Download dist |  | ||||||
|       if: ${{ env.DOWNLOAD_DIST == 'true' }} |  | ||||||
|       uses: actions/download-artifact@v3 |  | ||||||
|       with: |  | ||||||
|         name: dist |  | ||||||
|         path: dist/ |  | ||||||
							
								
								
									
										18
									
								
								.github/dependabot.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										18
									
								
								.github/dependabot.yml
									
									
									
									
										vendored
									
									
								
							| @@ -15,18 +15,6 @@ updates: | |||||||
|         patterns: |         patterns: | ||||||
|         - "*" |         - "*" | ||||||
|  |  | ||||||
|   - package-ecosystem: "npm" |  | ||||||
|     directory: "/" |  | ||||||
|     schedule: |  | ||||||
|       interval: "weekly" |  | ||||||
|     ignore: |  | ||||||
|       - dependency-name: "@types/node" # Breaking change: update with next major release |  | ||||||
|       - dependency-name: "@octokit/rest" # Tied to node version |  | ||||||
|     groups: |  | ||||||
|       npm-dependencies: |  | ||||||
|         patterns: |  | ||||||
|         - "*" |  | ||||||
|        |  | ||||||
|   - package-ecosystem: "gradle" |   - package-ecosystem: "gradle" | ||||||
|     directory: ".github/workflow-samples/gradle-plugin" |     directory: ".github/workflow-samples/gradle-plugin" | ||||||
|     registries: |     registries: | ||||||
| @@ -63,9 +51,3 @@ updates: | |||||||
|       - gradle-plugin-portal |       - gradle-plugin-portal | ||||||
|     schedule: |     schedule: | ||||||
|       interval: "daily" |       interval: "daily" | ||||||
|   - package-ecosystem: "gradle" |  | ||||||
|     directory: "test/init-scripts" |  | ||||||
|     registries: |  | ||||||
|       - gradle-plugin-portal |  | ||||||
|     schedule: |  | ||||||
|       interval: "daily" |  | ||||||
|   | |||||||
| @@ -1,6 +1,6 @@ | |||||||
| plugins { | plugins { | ||||||
|     id "com.gradle.enterprise" version "3.15.1" |     id "com.gradle.enterprise" version "3.16.2" | ||||||
|     id "com.gradle.common-custom-user-data-gradle-plugin" version "1.12" |     id "com.gradle.common-custom-user-data-gradle-plugin" version "1.12.1" | ||||||
| } | } | ||||||
|  |  | ||||||
| gradleEnterprise { | gradleEnterprise { | ||||||
|   | |||||||
| @@ -8,7 +8,7 @@ repositories { | |||||||
|  |  | ||||||
| dependencies { | dependencies { | ||||||
|     api("org.apache.commons:commons-math3:3.6.1") |     api("org.apache.commons:commons-math3:3.6.1") | ||||||
|     implementation("com.google.guava:guava:32.1.3-jre") |     implementation("com.google.guava:guava:33.0.0-jre") | ||||||
|  |  | ||||||
|     testImplementation("org.junit.jupiter:junit-jupiter:5.10.1") |     testImplementation("org.junit.jupiter:junit-jupiter:5.10.1") | ||||||
| } | } | ||||||
|   | |||||||
| @@ -1,6 +1,6 @@ | |||||||
| plugins { | plugins { | ||||||
|     id("com.gradle.enterprise") version "3.15.1" |     id("com.gradle.enterprise") version "3.16.2" | ||||||
|     id("com.gradle.common-custom-user-data-gradle-plugin") version "1.12" |     id("com.gradle.common-custom-user-data-gradle-plugin") version "1.12.1" | ||||||
| } | } | ||||||
|  |  | ||||||
| gradleEnterprise { | gradleEnterprise { | ||||||
|   | |||||||
| @@ -1,5 +1,5 @@ | |||||||
| plugins { | plugins { | ||||||
|     id "com.gradle.build-scan" version "3.15.1"  |     id "com.gradle.build-scan" version "3.16.2" | ||||||
| } | } | ||||||
|  |  | ||||||
| gradleEnterprise { | gradleEnterprise { | ||||||
|   | |||||||
| @@ -1,5 +1,5 @@ | |||||||
| plugins { | plugins { | ||||||
|     id "com.gradle.enterprise" version "3.15.1" |     id "com.gradle.enterprise" version "3.16.2" | ||||||
| } | } | ||||||
|  |  | ||||||
| gradleEnterprise { | gradleEnterprise { | ||||||
|   | |||||||
							
								
								
									
										70
									
								
								.github/workflows/ci-codeql.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										70
									
								
								.github/workflows/ci-codeql.yml
									
									
									
									
										vendored
									
									
								
							| @@ -1,70 +0,0 @@ | |||||||
| # For most projects, this workflow file will not need changing; you simply need |  | ||||||
| # to commit it to your repository. |  | ||||||
| # |  | ||||||
| # You may wish to alter this file to override the set of languages analyzed, |  | ||||||
| # or to provide custom queries or build logic. |  | ||||||
| # |  | ||||||
| # ******** NOTE ******** |  | ||||||
| # We have attempted to detect the languages in your repository. Please check |  | ||||||
| # the `language` matrix defined below to confirm you have the correct set of |  | ||||||
| # supported CodeQL languages. |  | ||||||
| # |  | ||||||
| name: CI-codeql |  | ||||||
|  |  | ||||||
| on: |  | ||||||
|   push: |  | ||||||
|     branches: [ main ] |  | ||||||
|   pull_request: |  | ||||||
|     # The branches below must be a subset of the branches above |  | ||||||
|     branches: [ main ] |  | ||||||
|   schedule: |  | ||||||
|     - cron: '25 23 * * 2' |  | ||||||
|  |  | ||||||
| jobs: |  | ||||||
|   analyze: |  | ||||||
|     name: Analyze |  | ||||||
|     runs-on: ubuntu-latest |  | ||||||
|     permissions: |  | ||||||
|       actions: read |  | ||||||
|       contents: read |  | ||||||
|       security-events: write |  | ||||||
|  |  | ||||||
|     strategy: |  | ||||||
|       fail-fast: false |  | ||||||
|       matrix: |  | ||||||
|         language: [ 'javascript' ] |  | ||||||
|         # CodeQL supports [ 'cpp', 'csharp', 'go', 'java', 'javascript', 'python', 'ruby' ] |  | ||||||
|         # Learn more about CodeQL language support at https://git.io/codeql-language-support |  | ||||||
|  |  | ||||||
|     steps: |  | ||||||
|     - name: Checkout repository |  | ||||||
|       uses: actions/checkout@v4 |  | ||||||
|  |  | ||||||
|     # Initializes the CodeQL tools for scanning. |  | ||||||
|     - name: Initialize CodeQL |  | ||||||
|       uses: github/codeql-action/init@v2 |  | ||||||
|       with: |  | ||||||
|         languages: ${{ matrix.language }} |  | ||||||
|         # If you wish to specify custom queries, you can do so here or in a config file. |  | ||||||
|         # By default, queries listed here will override any specified in a config file. |  | ||||||
|         # Prefix the list here with "+" to use these queries and those in the config file. |  | ||||||
|         # queries: ./path/to/local/query, your-org/your-repo/queries@main |  | ||||||
|  |  | ||||||
|     # Autobuild attempts to build any compiled languages  (C/C++, C#, or Java). |  | ||||||
|     # If this step fails, then you should remove it and run the build manually (see below) |  | ||||||
|     - name: Autobuild |  | ||||||
|       uses: github/codeql-action/autobuild@v2 |  | ||||||
|  |  | ||||||
|     # ℹ️ Command-line programs to run using the OS shell. |  | ||||||
|     # 📚 https://git.io/JvXDl |  | ||||||
|  |  | ||||||
|     # ✏️ If the Autobuild fails above, remove it and uncomment the following three lines |  | ||||||
|     #    and modify them (or add more) to build your code if your project |  | ||||||
|     #    uses a compiled language |  | ||||||
|  |  | ||||||
|     #- run: | |  | ||||||
|     #   make bootstrap |  | ||||||
|     #   make release |  | ||||||
|  |  | ||||||
|     - name: Perform CodeQL Analysis |  | ||||||
|       uses: github/codeql-action/analyze@v2 |  | ||||||
							
								
								
									
										20
									
								
								.github/workflows/ci-dependency-review.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										20
									
								
								.github/workflows/ci-dependency-review.yml
									
									
									
									
										vendored
									
									
								
							| @@ -1,20 +0,0 @@ | |||||||
| # Dependency Review Action |  | ||||||
| # |  | ||||||
| # This Action will scan dependency manifest files that change as part of a Pull Request, surfacing known-vulnerable versions of the packages declared or updated in the PR. Once installed, if the workflow run is marked as required, PRs introducing known-vulnerable packages will be blocked from merging. |  | ||||||
| # |  | ||||||
| # Source repository: https://github.com/actions/dependency-review-action |  | ||||||
| # Public documentation: https://docs.github.com/en/code-security/supply-chain-security/understanding-your-software-supply-chain/about-dependency-review#dependency-review-enforcement |  | ||||||
| name: CI-dependency-review |  | ||||||
| on: [pull_request] |  | ||||||
|  |  | ||||||
| permissions: |  | ||||||
|   contents: read |  | ||||||
|  |  | ||||||
| jobs: |  | ||||||
|   dependency-review: |  | ||||||
|     runs-on: ubuntu-latest |  | ||||||
|     steps: |  | ||||||
|       - name: 'Checkout Repository' |  | ||||||
|         uses: actions/checkout@v4 |  | ||||||
|       - name: 'Dependency Review' |  | ||||||
|         uses: actions/dependency-review-action@v3 |  | ||||||
							
								
								
									
										33
									
								
								.github/workflows/ci-full-check.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										33
									
								
								.github/workflows/ci-full-check.yml
									
									
									
									
										vendored
									
									
								
							| @@ -9,9 +9,10 @@ on: | |||||||
|   push: |   push: | ||||||
|     branches:  |     branches:  | ||||||
|       - main |       - main | ||||||
|  |       - release/** | ||||||
|     paths: |     paths: | ||||||
|       - '.github/**' |       - '.github/**' | ||||||
|       - 'dist/**' |       - 'action.yml' | ||||||
|  |  | ||||||
| jobs: | jobs: | ||||||
|   action-inputs: |   action-inputs: | ||||||
| @@ -19,18 +20,20 @@ jobs: | |||||||
|     with: |     with: | ||||||
|       cache-key-prefix: ${{github.run_number}}- |       cache-key-prefix: ${{github.run_number}}- | ||||||
|  |  | ||||||
|   cache-cleanup: |  | ||||||
|     uses: ./.github/workflows/integ-test-cache-cleanup.yml |  | ||||||
|     with: |  | ||||||
|       cache-key-prefix: ${{github.run_number}}- |  | ||||||
|  |  | ||||||
|   caching-config: |   caching-config: | ||||||
|     uses: ./.github/workflows/integ-test-action-inputs-caching.yml |     uses: ./.github/workflows/integ-test-caching-config.yml | ||||||
|     with: |     with: | ||||||
|       cache-key-prefix: ${{github.run_number}}- |       cache-key-prefix: ${{github.run_number}}- | ||||||
|  |  | ||||||
|   dependency-graph: |   dependency-graph: | ||||||
|     uses: ./.github/workflows/integ-test-dependency-graph.yml |     uses: ./.github/workflows/integ-test-dependency-graph.yml | ||||||
|  |     permissions: | ||||||
|  |       contents: write | ||||||
|  |     with: | ||||||
|  |       cache-key-prefix: ${{github.run_number}}- | ||||||
|  |  | ||||||
|  |   dependency-graph-failures: | ||||||
|  |     uses: ./.github/workflows/integ-test-dependency-graph-failures.yml | ||||||
|     with: |     with: | ||||||
|       cache-key-prefix: ${{github.run_number}}- |       cache-key-prefix: ${{github.run_number}}- | ||||||
|  |  | ||||||
| @@ -44,22 +47,24 @@ jobs: | |||||||
|     with: |     with: | ||||||
|       cache-key-prefix: ${{github.run_number}}- |       cache-key-prefix: ${{github.run_number}}- | ||||||
|  |  | ||||||
|   gradle-enterprise-injection: |   develocity-injection: | ||||||
|     uses: ./.github/workflows/integ-test-inject-gradle-enterprise.yml |     uses: ./.github/workflows/integ-test-inject-develocity.yml | ||||||
|     with: |     with: | ||||||
|       cache-key-prefix: ${{github.run_number}}- |       cache-key-prefix: ${{github.run_number}}- | ||||||
|     secrets: |     secrets: | ||||||
|       GRADLE_ENTERPRISE_ACCESS_KEY: ${{ secrets.GE_SOLUTIONS_ACCESS_TOKEN }} |       DEVELOCITY_ACCESS_KEY: ${{ secrets.GE_SOLUTIONS_ACCESS_TOKEN }} | ||||||
|  |  | ||||||
|   provision-gradle-versions: |   provision-gradle-versions: | ||||||
|     uses: ./.github/workflows/integ-test-provision-gradle-versions.yml |     uses: ./.github/workflows/integ-test-provision-gradle-versions.yml | ||||||
|     with: |     with: | ||||||
|       cache-key-prefix: ${{github.run_number}}- |       cache-key-prefix: ${{github.run_number}}- | ||||||
|  |  | ||||||
|   # restore-configuration-cache: |   restore-configuration-cache: | ||||||
|   #   uses: ./.github/workflows/integ-test-restore-configuration-cache.yml |     uses: ./.github/workflows/integ-test-restore-configuration-cache.yml | ||||||
|   #   with: |     with: | ||||||
|   #     cache-key-prefix: ${{github.run_number}}- |       cache-key-prefix: ${{github.run_number}}- | ||||||
|  |     secrets: | ||||||
|  |       GRADLE_ENCRYPTION_KEY: ${{ secrets.GRADLE_ENCRYPTION_KEY }} | ||||||
|  |  | ||||||
|   restore-custom-gradle-home: |   restore-custom-gradle-home: | ||||||
|     uses: ./.github/workflows/integ-test-restore-custom-gradle-home.yml |     uses: ./.github/workflows/integ-test-restore-custom-gradle-home.yml | ||||||
|   | |||||||
							
								
								
									
										26
									
								
								.github/workflows/ci-init-script-check.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										26
									
								
								.github/workflows/ci-init-script-check.yml
									
									
									
									
										vendored
									
									
								
							| @@ -1,26 +0,0 @@ | |||||||
| name: CI-init-script-check |  | ||||||
|  |  | ||||||
| on: |  | ||||||
|   push: |  | ||||||
|     paths: |  | ||||||
|       - '.github/workflows/ci-init-script-check.yml' |  | ||||||
|       - 'src/resources/init-scripts/**' |  | ||||||
|       - 'test/init-scripts/**' |  | ||||||
|   workflow_dispatch: |  | ||||||
|  |  | ||||||
| jobs: |  | ||||||
|   test-init-scripts: |  | ||||||
|     runs-on: ubuntu-latest |  | ||||||
|     steps: |  | ||||||
|     - name: Checkout sources |  | ||||||
|       uses: actions/checkout@v4 |  | ||||||
|     - name: Setup Java |  | ||||||
|       uses: actions/setup-java@v4 |  | ||||||
|       with: |  | ||||||
|         distribution: temurin |  | ||||||
|         java-version: 8 |  | ||||||
|     - name: Setup Gradle |  | ||||||
|       uses: gradle/gradle-build-action@v2.10.0 # Use a released version to avoid breakages |  | ||||||
|     - name: Run integration tests |  | ||||||
|       working-directory: test/init-scripts |  | ||||||
|       run: ./gradlew check |  | ||||||
							
								
								
									
										91
									
								
								.github/workflows/ci-quick-check.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										91
									
								
								.github/workflows/ci-quick-check.yml
									
									
									
									
										vendored
									
									
								
							| @@ -3,140 +3,89 @@ name: CI-quick-check | |||||||
| on: | on: | ||||||
|   workflow_dispatch: |   workflow_dispatch: | ||||||
|   push: |   push: | ||||||
|     branches-ignore: main |     branches-ignore:  | ||||||
|  |     - main | ||||||
|  |     - release/** | ||||||
|  |  | ||||||
| jobs: | jobs: | ||||||
|   build-distribution: |  | ||||||
|     runs-on: ubuntu-latest |  | ||||||
|     steps: |  | ||||||
|     - name: Checkout sources |  | ||||||
|       uses: actions/checkout@v4 |  | ||||||
|     - name: Build and upload distribution |  | ||||||
|       uses: ./.github/actions/build-dist |  | ||||||
|  |  | ||||||
|   run-unit-tests: |  | ||||||
|     runs-on: ubuntu-latest |  | ||||||
|     steps: |  | ||||||
|     - name: Checkout sources |  | ||||||
|       uses: actions/checkout@v4 |  | ||||||
|     - name: Configure Gradle as default for unit test |  | ||||||
|       uses: ./ |  | ||||||
|       with: |  | ||||||
|         gradle-version: 8.5 |  | ||||||
|     - name: Run tests |  | ||||||
|       run: | |  | ||||||
|         npm install |  | ||||||
|         npm run all |  | ||||||
|  |  | ||||||
|   action-inputs: |   action-inputs: | ||||||
|     needs: build-distribution |  | ||||||
|     uses: ./.github/workflows/integ-test-action-inputs.yml |     uses: ./.github/workflows/integ-test-action-inputs.yml | ||||||
|     with: |     with: | ||||||
|       runner-os: '["ubuntu-latest"]' |       runner-os: '["ubuntu-latest"]' | ||||||
|       download-dist: true |  | ||||||
|  |  | ||||||
|   cache-cleanup: |  | ||||||
|     needs: build-distribution |  | ||||||
|     uses: ./.github/workflows/integ-test-cache-cleanup.yml |  | ||||||
|     with: |  | ||||||
|       runner-os: '["ubuntu-latest"]' |  | ||||||
|       download-dist: true |  | ||||||
|       cache-key-prefix: ${{github.run_number}}- # Requires a fresh cache entry each run |  | ||||||
|  |  | ||||||
|   caching-config: |   caching-config: | ||||||
|     needs: build-distribution |     uses: ./.github/workflows/integ-test-caching-config.yml | ||||||
|     uses: ./.github/workflows/integ-test-action-inputs-caching.yml |  | ||||||
|     with: |     with: | ||||||
|       runner-os: '["ubuntu-latest"]' |       runner-os: '["ubuntu-latest"]' | ||||||
|       download-dist: true |  | ||||||
|  |  | ||||||
|   dependency-graph: |   dependency-graph: | ||||||
|     needs: build-distribution |  | ||||||
|     uses: ./.github/workflows/integ-test-dependency-graph.yml |     uses: ./.github/workflows/integ-test-dependency-graph.yml | ||||||
|  |     permissions: | ||||||
|  |       contents: write | ||||||
|  |     with: | ||||||
|  |       runner-os: '["ubuntu-latest"]' | ||||||
|  |  | ||||||
|  |   dependency-graph-failures: | ||||||
|  |     uses: ./.github/workflows/integ-test-dependency-graph-failures.yml | ||||||
|     with: |     with: | ||||||
|       runner-os: '["ubuntu-latest"]' |       runner-os: '["ubuntu-latest"]' | ||||||
|       download-dist: true |  | ||||||
|  |  | ||||||
|   execution-with-caching: |   execution-with-caching: | ||||||
|     needs: build-distribution |  | ||||||
|     uses: ./.github/workflows/integ-test-execution-with-caching.yml |     uses: ./.github/workflows/integ-test-execution-with-caching.yml | ||||||
|     with: |     with: | ||||||
|       runner-os: '["ubuntu-latest"]' |       runner-os: '["ubuntu-latest"]' | ||||||
|       download-dist: true |  | ||||||
|  |  | ||||||
|   execution: |   execution: | ||||||
|     needs: build-distribution |  | ||||||
|     uses: ./.github/workflows/integ-test-execution.yml |     uses: ./.github/workflows/integ-test-execution.yml | ||||||
|     with: |     with: | ||||||
|       runner-os: '["ubuntu-latest"]' |       runner-os: '["ubuntu-latest"]' | ||||||
|       download-dist: true |  | ||||||
|  |  | ||||||
|   gradle-enterprise-injection: |   develocity-injection: | ||||||
|     needs: build-distribution |     uses: ./.github/workflows/integ-test-inject-develocity.yml | ||||||
|     uses: ./.github/workflows/integ-test-inject-gradle-enterprise.yml |  | ||||||
|     with: |     with: | ||||||
|       runner-os: '["ubuntu-latest"]' |       runner-os: '["ubuntu-latest"]' | ||||||
|       download-dist: true |  | ||||||
|     secrets: |     secrets: | ||||||
|       GRADLE_ENTERPRISE_ACCESS_KEY: ${{ secrets.GE_SOLUTIONS_ACCESS_TOKEN }} |       DEVELOCITY_ACCESS_KEY: ${{ secrets.GE_SOLUTIONS_ACCESS_TOKEN }} | ||||||
|  |  | ||||||
|   provision-gradle-versions: |   provision-gradle-versions: | ||||||
|     needs: build-distribution |  | ||||||
|     uses: ./.github/workflows/integ-test-provision-gradle-versions.yml |     uses: ./.github/workflows/integ-test-provision-gradle-versions.yml | ||||||
|     with: |     with: | ||||||
|       runner-os: '["ubuntu-latest"]' |       runner-os: '["ubuntu-latest"]' | ||||||
|       download-dist: true |  | ||||||
|  |  | ||||||
|   # restore-configuration-cache: |   restore-configuration-cache: | ||||||
|   #   needs: build-distribution |     uses: ./.github/workflows/integ-test-restore-configuration-cache.yml | ||||||
|   #   uses: ./.github/workflows/integ-test-restore-configuration-cache.yml |     with: | ||||||
|   #   with: |       runner-os: '["ubuntu-latest"]' | ||||||
|   #     runner-os: '["ubuntu-latest"]' |     secrets: | ||||||
|   #     download-dist: true |       GRADLE_ENCRYPTION_KEY: ${{ secrets.GRADLE_ENCRYPTION_KEY }} | ||||||
|  |  | ||||||
|   restore-containerized-gradle-home: |   restore-containerized-gradle-home: | ||||||
|     needs: build-distribution |  | ||||||
|     uses: ./.github/workflows/integ-test-restore-containerized-gradle-home.yml |     uses: ./.github/workflows/integ-test-restore-containerized-gradle-home.yml | ||||||
|     with: |  | ||||||
|       download-dist: true |  | ||||||
|  |  | ||||||
|   restore-custom-gradle-home: |   restore-custom-gradle-home: | ||||||
|     needs: build-distribution |  | ||||||
|     uses: ./.github/workflows/integ-test-restore-custom-gradle-home.yml |     uses: ./.github/workflows/integ-test-restore-custom-gradle-home.yml | ||||||
|     with: |  | ||||||
|       download-dist: true |  | ||||||
|  |  | ||||||
|   restore-gradle-home: |   restore-gradle-home: | ||||||
|     needs: build-distribution |  | ||||||
|     uses: ./.github/workflows/integ-test-restore-gradle-home.yml |     uses: ./.github/workflows/integ-test-restore-gradle-home.yml | ||||||
|     with: |     with: | ||||||
|       runner-os: '["ubuntu-latest"]' |       runner-os: '["ubuntu-latest"]' | ||||||
|       download-dist: true |  | ||||||
|  |  | ||||||
|   restore-java-toolchain: |   restore-java-toolchain: | ||||||
|     needs: build-distribution |  | ||||||
|     uses: ./.github/workflows/integ-test-restore-java-toolchain.yml |     uses: ./.github/workflows/integ-test-restore-java-toolchain.yml | ||||||
|     with: |     with: | ||||||
|       runner-os: '["ubuntu-latest"]' |       runner-os: '["ubuntu-latest"]' | ||||||
|       download-dist: true |  | ||||||
|  |  | ||||||
|   sample-kotlin-dsl: |   sample-kotlin-dsl: | ||||||
|     needs: build-distribution |  | ||||||
|     uses: ./.github/workflows/integ-test-sample-kotlin-dsl.yml |     uses: ./.github/workflows/integ-test-sample-kotlin-dsl.yml | ||||||
|     with: |     with: | ||||||
|       runner-os: '["ubuntu-latest"]' |       runner-os: '["ubuntu-latest"]' | ||||||
|       download-dist: true |  | ||||||
|  |  | ||||||
|   sample-gradle-plugin: |   sample-gradle-plugin: | ||||||
|     needs: build-distribution |  | ||||||
|     uses: ./.github/workflows/integ-test-sample-gradle-plugin.yml |     uses: ./.github/workflows/integ-test-sample-gradle-plugin.yml | ||||||
|     with: |     with: | ||||||
|       runner-os: '["ubuntu-latest"]' |       runner-os: '["ubuntu-latest"]' | ||||||
|       download-dist: true |  | ||||||
|  |  | ||||||
|   toolchain-detection: |   toolchain-detection: | ||||||
|     needs: build-distribution |  | ||||||
|     uses: ./.github/workflows/integ-test-detect-java-toolchains.yml |     uses: ./.github/workflows/integ-test-detect-java-toolchains.yml | ||||||
|     with: |     with: | ||||||
|       runner-os: '["ubuntu-latest"]' |       runner-os: '["ubuntu-latest"]' | ||||||
|       download-dist: true |  | ||||||
|   | |||||||
							
								
								
									
										40
									
								
								.github/workflows/ci-verify-outputs.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										40
									
								
								.github/workflows/ci-verify-outputs.yml
									
									
									
									
										vendored
									
									
								
							| @@ -1,40 +0,0 @@ | |||||||
| name: CI-verify-outputs |  | ||||||
|  |  | ||||||
| on: |  | ||||||
|   pull_request: |  | ||||||
|     types: |  | ||||||
|       - assigned |  | ||||||
|       - review_requested |  | ||||||
|   push: |  | ||||||
|     branches:  |  | ||||||
|       - main |  | ||||||
|       - dependabot/** |  | ||||||
|  |  | ||||||
| jobs: |  | ||||||
|   check: |  | ||||||
|     runs-on: ubuntu-latest |  | ||||||
|     steps: |  | ||||||
|     - name: Checkout sources |  | ||||||
|       uses: actions/checkout@v4 |  | ||||||
|     - name: Build |  | ||||||
|       run: | |  | ||||||
|         npm -v |  | ||||||
|         node -v |  | ||||||
|         npm install |  | ||||||
|         npm run build |  | ||||||
|  |  | ||||||
|     - name: Compare the expected and actual dist/ directories |  | ||||||
|       run: | |  | ||||||
|         if [ "$(git diff --ignore-space-at-eol dist/ | wc -l)" -gt "0" ]; then |  | ||||||
|           echo "Detected uncommitted changes after build.  See status below:" |  | ||||||
|           git diff |  | ||||||
|           exit 1 |  | ||||||
|         fi |  | ||||||
|       id: diff |  | ||||||
|  |  | ||||||
|     # If index.js was different than expected, upload the expected version as an artifact |  | ||||||
|     - uses: actions/upload-artifact@v3 |  | ||||||
|       if: ${{ failure() && steps.diff.conclusion == 'failure' }} |  | ||||||
|       with: |  | ||||||
|         name: dist |  | ||||||
|         path: dist/ |  | ||||||
							
								
								
									
										28
									
								
								.github/workflows/demo-job-summary.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										28
									
								
								.github/workflows/demo-job-summary.yml
									
									
									
									
										vendored
									
									
								
							| @@ -8,16 +8,11 @@ env: | |||||||
|   GRADLE_BUILD_ACTION_CACHE_DEBUG_ENABLED: true |   GRADLE_BUILD_ACTION_CACHE_DEBUG_ENABLED: true | ||||||
|  |  | ||||||
| jobs: | jobs: | ||||||
|   run-gradle-builds: |   many-gradle-builds: | ||||||
|     runs-on: ubuntu-latest |     runs-on: ubuntu-latest | ||||||
|     steps: |     steps: | ||||||
|     - name: Checkout sources |     - name: Checkout sources | ||||||
|       uses: actions/checkout@v4 |       uses: actions/checkout@v4 | ||||||
|     - name: Build distribution |  | ||||||
|       shell: bash |  | ||||||
|       run: | |  | ||||||
|         npm install |  | ||||||
|         npm run build |  | ||||||
|     - name: Setup Gradle |     - name: Setup Gradle | ||||||
|       uses: ./ |       uses: ./ | ||||||
|     - name: Build kotlin-dsl project |     - name: Build kotlin-dsl project | ||||||
| @@ -42,16 +37,27 @@ jobs: | |||||||
|       continue-on-error: true |       continue-on-error: true | ||||||
|       run: ./gradlew not-a-real-task |       run: ./gradlew not-a-real-task | ||||||
|  |  | ||||||
|  |   successful-builds-with-no-summary: | ||||||
|  |     runs-on: ubuntu-latest | ||||||
|  |     steps: | ||||||
|  |     - name: Checkout sources | ||||||
|  |       uses: actions/checkout@v4 | ||||||
|  |     - name: Setup Gradle | ||||||
|  |       uses: ./ | ||||||
|  |       with: | ||||||
|  |         add-job-summary: on-failure | ||||||
|  |     - name: Build kotlin-dsl project | ||||||
|  |       working-directory: .github/workflow-samples/kotlin-dsl | ||||||
|  |       run: ./gradlew assemble | ||||||
|  |     - name: Build kotlin-dsl project without Build Scan® | ||||||
|  |       working-directory: .github/workflow-samples/kotlin-dsl | ||||||
|  |       run: ./gradlew assemble check --no-scan | ||||||
|  |  | ||||||
|   pre-existing-gradle-home: |   pre-existing-gradle-home: | ||||||
|     runs-on: ubuntu-latest |     runs-on: ubuntu-latest | ||||||
|     steps: |     steps: | ||||||
|     - name: Checkout sources |     - name: Checkout sources | ||||||
|       uses: actions/checkout@v4 |       uses: actions/checkout@v4 | ||||||
|     - name: Build distribution |  | ||||||
|       shell: bash |  | ||||||
|       run: | |  | ||||||
|         npm install |  | ||||||
|         npm run build |  | ||||||
|     - name: Pre-create Gradle User Home |     - name: Pre-create Gradle User Home | ||||||
|       shell: bash |       shell: bash | ||||||
|       run: | |       run: | | ||||||
|   | |||||||
							
								
								
									
										46
									
								
								.github/workflows/demo-pr-build-scan-comment.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										46
									
								
								.github/workflows/demo-pr-build-scan-comment.yml
									
									
									
									
										vendored
									
									
								
							| @@ -2,26 +2,50 @@ name: Demo adding Build Scan® comment to PR | |||||||
| on: | on: | ||||||
|   pull_request: |   pull_request: | ||||||
|     types: [assigned, review_requested] |     types: [assigned, review_requested] | ||||||
|  |  | ||||||
|  | permissions: | ||||||
|  |   pull-requests: write | ||||||
|  |  | ||||||
| jobs: | jobs: | ||||||
|   gradle: |   successful-build-with-always-comment: | ||||||
|     runs-on: ubuntu-latest |     runs-on: ubuntu-latest | ||||||
|     steps: |     steps: | ||||||
|     - name: Checkout project sources |     - name: Checkout project sources | ||||||
|       uses: actions/checkout@v4 |       uses: actions/checkout@v4 | ||||||
|     - name: Setup Gradle |     - name: Setup Gradle | ||||||
|       uses: ./ |       uses: ./ | ||||||
|  |       with: | ||||||
|  |         add-job-summary-as-pr-comment: always | ||||||
|     - name: Run build with Gradle wrapper |     - name: Run build with Gradle wrapper | ||||||
|       id: gradle |       id: gradle | ||||||
|       working-directory: .github/workflow-samples/kotlin-dsl |       working-directory: .github/workflow-samples/kotlin-dsl | ||||||
|       run: ./gradlew build --scan |       run: ./gradlew build --scan | ||||||
|     - name: "Add Build Scan URL as PR comment" |  | ||||||
|       uses: actions/github-script@v7 |   successful-build-with-comment-on-failure: | ||||||
|  |     runs-on: ubuntu-latest | ||||||
|  |     steps: | ||||||
|  |     - name: Checkout project sources | ||||||
|  |       uses: actions/checkout@v4 | ||||||
|  |     - name: Setup Gradle | ||||||
|  |       uses: ./ | ||||||
|       with: |       with: | ||||||
|         github-token: ${{secrets.GITHUB_TOKEN}} |         add-job-summary-as-pr-comment: on-failure | ||||||
|         script: | |     - name: Run build with Gradle wrapper | ||||||
|           github.rest.issues.createComment({ |       id: gradle | ||||||
|             issue_number: context.issue.number, |       working-directory: .github/workflow-samples/kotlin-dsl | ||||||
|             owner: context.repo.owner, |       run: ./gradlew build --scan | ||||||
|             repo: context.repo.repo, |  | ||||||
|             body: 'PR ready for review: ${{ steps.gradle.outputs.build-scan-url }}' |   failing-build-with-comment-on-failure: | ||||||
|           }) |     runs-on: ubuntu-latest | ||||||
|  |     steps: | ||||||
|  |     - name: Checkout project sources | ||||||
|  |       uses: actions/checkout@v4 | ||||||
|  |     - name: Setup Gradle | ||||||
|  |       uses: ./ | ||||||
|  |       with: | ||||||
|  |         add-job-summary-as-pr-comment: on-failure | ||||||
|  |     - name: Run build with Gradle wrapper | ||||||
|  |       id: gradle | ||||||
|  |       working-directory: .github/workflow-samples/kotlin-dsl | ||||||
|  |       run: ./gradlew no-a-real-task --scan | ||||||
|  |       continue-on-error: true | ||||||
|   | |||||||
| @@ -8,12 +8,8 @@ on: | |||||||
|       runner-os: |       runner-os: | ||||||
|         type: string |         type: string | ||||||
|         default: '["ubuntu-latest", "windows-latest", "macos-latest"]' |         default: '["ubuntu-latest", "windows-latest", "macos-latest"]' | ||||||
|       download-dist: |  | ||||||
|         type: boolean |  | ||||||
|         default: false |  | ||||||
|  |  | ||||||
| env: | env: | ||||||
|   DOWNLOAD_DIST: ${{ inputs.download-dist }} |  | ||||||
|   GRADLE_BUILD_ACTION_CACHE_KEY_PREFIX: action-inputs-${{ inputs.cache-key-prefix }} |   GRADLE_BUILD_ACTION_CACHE_KEY_PREFIX: action-inputs-${{ inputs.cache-key-prefix }} | ||||||
|   GRADLE_BUILD_ACTION_CACHE_DEBUG_ENABLED: true |   GRADLE_BUILD_ACTION_CACHE_DEBUG_ENABLED: true | ||||||
|  |  | ||||||
| @@ -26,8 +22,6 @@ jobs: | |||||||
|     steps: |     steps: | ||||||
|     - name: Checkout sources |     - name: Checkout sources | ||||||
|       uses: actions/checkout@v4 |       uses: actions/checkout@v4 | ||||||
|     - name: Download distribution if required |  | ||||||
|       uses: ./.github/actions/download-dist |  | ||||||
|     - name: Invoke with multi-line arguments |     - name: Invoke with multi-line arguments | ||||||
|       uses: ./ |       uses: ./ | ||||||
|       with: |       with: | ||||||
|   | |||||||
							
								
								
									
										87
									
								
								.github/workflows/integ-test-cache-cleanup.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										87
									
								
								.github/workflows/integ-test-cache-cleanup.yml
									
									
									
									
										vendored
									
									
								
							| @@ -1,87 +0,0 @@ | |||||||
| name: Test cache cleanup |  | ||||||
|  |  | ||||||
| on: |  | ||||||
|   workflow_call: |  | ||||||
|     inputs: |  | ||||||
|       cache-key-prefix: |  | ||||||
|         type: string |  | ||||||
|       runner-os: |  | ||||||
|         type: string |  | ||||||
|         default: '["ubuntu-latest", "windows-latest", "macos-latest"]' |  | ||||||
|       download-dist: |  | ||||||
|         type: boolean |  | ||||||
|         default: false |  | ||||||
|  |  | ||||||
| env: |  | ||||||
|   DOWNLOAD_DIST: ${{ inputs.download-dist }} |  | ||||||
|   GRADLE_BUILD_ACTION_CACHE_KEY_PREFIX: integ-test-cache-cleanup-${{ inputs.cache-key-prefix }} |  | ||||||
|   GRADLE_BUILD_ACTION_CACHE_DEBUG_ENABLED: true |  | ||||||
|  |  | ||||||
| jobs: |  | ||||||
|   full-build: |  | ||||||
|     strategy: |  | ||||||
|       matrix: |  | ||||||
|         os: ${{fromJSON(inputs.runner-os)}} |  | ||||||
|     runs-on: ${{ matrix.os }} |  | ||||||
|     steps: |  | ||||||
|     - name: Checkout sources |  | ||||||
|       uses: actions/checkout@v4 |  | ||||||
|     - name: Download distribution if required |  | ||||||
|       uses: ./.github/actions/download-dist |  | ||||||
|     - name: Setup Gradle |  | ||||||
|       uses: ./ |  | ||||||
|       with: |  | ||||||
|         cache-read-only: false # For testing, allow writing cache entries on non-default branches |  | ||||||
|     - name: Build with 3.1 |  | ||||||
|       working-directory: test/jest/resources/cache-cleanup |  | ||||||
|       run: gradle --no-daemon --build-cache -Dcommons_math3_version="3.1" build |  | ||||||
|  |  | ||||||
|   # Second build will use the cache from the first build, but cleanup should remove unused artifacts |  | ||||||
|   assemble-build: |  | ||||||
|     needs: full-build |  | ||||||
|     strategy: |  | ||||||
|       matrix: |  | ||||||
|         os: ${{fromJSON(inputs.runner-os)}} |  | ||||||
|     runs-on: ${{ matrix.os }} |  | ||||||
|     steps: |  | ||||||
|     - name: Checkout sources |  | ||||||
|       uses: actions/checkout@v4 |  | ||||||
|     - name: Download distribution if required |  | ||||||
|       uses: ./.github/actions/download-dist |  | ||||||
|     - name: Setup Gradle |  | ||||||
|       uses: ./ |  | ||||||
|       with: |  | ||||||
|         cache-read-only: false |  | ||||||
|         gradle-home-cache-cleanup: true |  | ||||||
|     - name: Build with 3.1.1 |  | ||||||
|       working-directory: test/jest/resources/cache-cleanup |  | ||||||
|       run: gradle --no-daemon --build-cache -Dcommons_math3_version="3.1.1" build |  | ||||||
|  |  | ||||||
|   check-clean-cache: |  | ||||||
|     needs: assemble-build |  | ||||||
|     strategy: |  | ||||||
|       matrix: |  | ||||||
|         os: ${{fromJSON(inputs.runner-os)}} |  | ||||||
|     runs-on: ${{ matrix.os }} |  | ||||||
|     steps: |  | ||||||
|     - name: Checkout sources |  | ||||||
|       uses: actions/checkout@v4 |  | ||||||
|     - name: Download distribution if required |  | ||||||
|       uses: ./.github/actions/download-dist |  | ||||||
|     - name: Setup Gradle |  | ||||||
|       uses: ./ |  | ||||||
|       with: |  | ||||||
|         cache-read-only: true |  | ||||||
|     - name: Report Gradle User Home |  | ||||||
|       run: du -hc ~/.gradle/caches/modules-2 |  | ||||||
|     - name: Verify cleaned cache |  | ||||||
|       shell: bash |  | ||||||
|       run: | |  | ||||||
|         if [ ! -e ~/.gradle/caches/modules-2/files-2.1/org.apache.commons/commons-math3/3.1.1 ]; then |  | ||||||
|           echo "::error ::Should find commons-math3 3.1.1 in cache" |  | ||||||
|           exit 1 |  | ||||||
|         fi |  | ||||||
|         if [ -e ~/.gradle/caches/modules-2/files-2.1/org.apache.commons/commons-math3/3.1 ]; then |  | ||||||
|           echo "::error ::Should NOT find commons-math3 3.1 in cache" |  | ||||||
|           exit 1 |  | ||||||
|         fi |  | ||||||
| @@ -1,4 +1,4 @@ | |||||||
| name: Test action inputs for caching | name: Test caching configuration | ||||||
| 
 | 
 | ||||||
| on: | on: | ||||||
|   workflow_call: |   workflow_call: | ||||||
| @@ -8,12 +8,8 @@ on: | |||||||
|       runner-os: |       runner-os: | ||||||
|         type: string |         type: string | ||||||
|         default: '["ubuntu-latest", "windows-latest", "macos-latest"]' |         default: '["ubuntu-latest", "windows-latest", "macos-latest"]' | ||||||
|       download-dist: |  | ||||||
|         type: boolean |  | ||||||
|         default: false |  | ||||||
| 
 | 
 | ||||||
| env: | env: | ||||||
|   DOWNLOAD_DIST: ${{ inputs.download-dist }} |  | ||||||
|   GRADLE_BUILD_ACTION_CACHE_KEY_PREFIX: action-inputs-caching-${{ inputs.cache-key-prefix }} |   GRADLE_BUILD_ACTION_CACHE_KEY_PREFIX: action-inputs-caching-${{ inputs.cache-key-prefix }} | ||||||
|   GRADLE_BUILD_ACTION_CACHE_DEBUG_ENABLED: true |   GRADLE_BUILD_ACTION_CACHE_DEBUG_ENABLED: true | ||||||
| 
 | 
 | ||||||
| @@ -26,8 +22,6 @@ jobs: | |||||||
|     steps: |     steps: | ||||||
|     - name: Checkout sources |     - name: Checkout sources | ||||||
|       uses: actions/checkout@v4 |       uses: actions/checkout@v4 | ||||||
|     - name: Download distribution if required |  | ||||||
|       uses: ./.github/actions/download-dist |  | ||||||
|     - name: Setup Gradle |     - name: Setup Gradle | ||||||
|       uses: ./ |       uses: ./ | ||||||
|       with: |       with: | ||||||
| @@ -38,7 +32,8 @@ jobs: | |||||||
|             enterprise |             enterprise | ||||||
|         # Exclude build-cache from main cache entry |         # Exclude build-cache from main cache entry | ||||||
|         gradle-home-cache-excludes: | |         gradle-home-cache-excludes: | | ||||||
|             caches/build-cache-1 |             caches/build-cache-* | ||||||
|  |             caches/*/executionHistory | ||||||
|     - name: Build using Gradle wrapper |     - name: Build using Gradle wrapper | ||||||
|       working-directory: .github/workflow-samples/groovy-dsl |       working-directory: .github/workflow-samples/groovy-dsl | ||||||
|       run: ./gradlew test |       run: ./gradlew test | ||||||
| @@ -53,8 +48,6 @@ jobs: | |||||||
|     steps: |     steps: | ||||||
|     - name: Checkout sources |     - name: Checkout sources | ||||||
|       uses: actions/checkout@v4 |       uses: actions/checkout@v4 | ||||||
|     - name: Download distribution if required |  | ||||||
|       uses: ./.github/actions/download-dist |  | ||||||
|     - name: Setup Gradle |     - name: Setup Gradle | ||||||
|       uses: ./ |       uses: ./ | ||||||
|       with: |       with: | ||||||
| @@ -63,7 +56,8 @@ jobs: | |||||||
|             caches |             caches | ||||||
|             enterprise |             enterprise | ||||||
|         gradle-home-cache-excludes: | |         gradle-home-cache-excludes: | | ||||||
|             caches/build-cache-1 |             caches/build-cache-* | ||||||
|  |             caches/*/executionHistory | ||||||
|         cache-read-only: true |         cache-read-only: true | ||||||
|     - name: Execute Gradle build with --offline |     - name: Execute Gradle build with --offline | ||||||
|       working-directory: .github/workflow-samples/groovy-dsl |       working-directory: .github/workflow-samples/groovy-dsl | ||||||
| @@ -78,8 +72,6 @@ jobs: | |||||||
|     steps: |     steps: | ||||||
|     - name: Checkout sources |     - name: Checkout sources | ||||||
|       uses: actions/checkout@v4 |       uses: actions/checkout@v4 | ||||||
|     - name: Download distribution if required |  | ||||||
|       uses: ./.github/actions/download-dist |  | ||||||
|     - name: Setup Gradle |     - name: Setup Gradle | ||||||
|       uses: ./ |       uses: ./ | ||||||
|       with: |       with: | ||||||
| @@ -101,8 +93,6 @@ jobs: | |||||||
|     steps: |     steps: | ||||||
|     - name: Checkout sources |     - name: Checkout sources | ||||||
|       uses: actions/checkout@v4 |       uses: actions/checkout@v4 | ||||||
|     - name: Download distribution if required |  | ||||||
|       uses: ./.github/actions/download-dist |  | ||||||
|     - name: Create dummy Gradle User Home |     - name: Create dummy Gradle User Home | ||||||
|       run: mkdir -p ~/.gradle/caches |       run: mkdir -p ~/.gradle/caches | ||||||
|     - name: Setup Gradle |     - name: Setup Gradle | ||||||
| @@ -129,8 +119,6 @@ jobs: | |||||||
|     steps: |     steps: | ||||||
|     - name: Checkout sources |     - name: Checkout sources | ||||||
|       uses: actions/checkout@v4 |       uses: actions/checkout@v4 | ||||||
|     - name: Download distribution if required |  | ||||||
|       uses: ./.github/actions/download-dist |  | ||||||
|     - name: Setup Gradle |     - name: Setup Gradle | ||||||
|       uses: ./ |       uses: ./ | ||||||
|       with: |       with: | ||||||
| @@ -150,8 +138,6 @@ jobs: | |||||||
|     steps: |     steps: | ||||||
|     - name: Checkout sources |     - name: Checkout sources | ||||||
|       uses: actions/checkout@v4 |       uses: actions/checkout@v4 | ||||||
|     - name: Download distribution if required |  | ||||||
|       uses: ./.github/actions/download-dist |  | ||||||
|     - name: Setup Gradle |     - name: Setup Gradle | ||||||
|       uses: ./ |       uses: ./ | ||||||
|       with: |       with: | ||||||
							
								
								
									
										91
									
								
								.github/workflows/integ-test-dependency-graph-failures.yml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										91
									
								
								.github/workflows/integ-test-dependency-graph-failures.yml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,91 @@ | |||||||
|  | name: Test dependency graph | ||||||
|  |  | ||||||
|  | on: | ||||||
|  |   workflow_call: | ||||||
|  |     inputs: | ||||||
|  |       cache-key-prefix: | ||||||
|  |         type: string | ||||||
|  |       runner-os: | ||||||
|  |         type: string | ||||||
|  |         default: '["ubuntu-latest", "windows-latest", "macos-latest"]' | ||||||
|  |  | ||||||
|  | env: | ||||||
|  |   GRADLE_BUILD_ACTION_CACHE_KEY_PREFIX: dependency-graph-${{ inputs.cache-key-prefix }} | ||||||
|  |   GRADLE_BUILD_ACTION_CACHE_DEBUG_ENABLED: true | ||||||
|  |  | ||||||
|  | jobs: | ||||||
|  |   unsupported-gradle-version-warning: | ||||||
|  |     runs-on: ubuntu-latest | ||||||
|  |     steps: | ||||||
|  |     - name: Checkout sources | ||||||
|  |       uses: actions/checkout@v4 | ||||||
|  |     - name: Setup Gradle for dependency-graph generate | ||||||
|  |       uses: ./ | ||||||
|  |       with: | ||||||
|  |         gradle-version: 7.0.1 | ||||||
|  |         dependency-graph: generate | ||||||
|  |         dependency-graph-continue-on-failure: true | ||||||
|  |     - name: Run with unsupported Gradle version | ||||||
|  |       working-directory: .github/workflow-samples/groovy-dsl | ||||||
|  |       run: | | ||||||
|  |         if gradle help | grep -q 'warning::Dependency Graph is not supported for Gradle 7.0.1. No dependency snapshot will be generated.'; | ||||||
|  |         then | ||||||
|  |           echo "Got the expected warning" | ||||||
|  |         else | ||||||
|  |           echo "Did not get the expected warning" | ||||||
|  |           exit 1 | ||||||
|  |         fi | ||||||
|  |  | ||||||
|  |   unsupported-gradle-version-failure: | ||||||
|  |     runs-on: ubuntu-latest | ||||||
|  |     steps: | ||||||
|  |     - name: Checkout sources | ||||||
|  |       uses: actions/checkout@v4 | ||||||
|  |     - name: Setup Gradle for dependency-graph generate | ||||||
|  |       uses: ./ | ||||||
|  |       with: | ||||||
|  |         gradle-version: 7.0.1 | ||||||
|  |         dependency-graph: generate | ||||||
|  |         dependency-graph-continue-on-failure: false | ||||||
|  |     - name: Run with unsupported Gradle version | ||||||
|  |       working-directory: .github/workflow-samples/groovy-dsl | ||||||
|  |       run: | | ||||||
|  |         if gradle help; then | ||||||
|  |           echo "Expected build to fail with Gradle 7.0.1" | ||||||
|  |           exit 1 | ||||||
|  |         fi | ||||||
|  |  | ||||||
|  |   insufficient-permissions-warning: | ||||||
|  |     runs-on: ubuntu-latest | ||||||
|  |     permissions: | ||||||
|  |       contents: read | ||||||
|  |     steps: | ||||||
|  |     - name: Checkout sources | ||||||
|  |       uses: actions/checkout@v4 | ||||||
|  |     - name: Setup Gradle for dependency-graph generate | ||||||
|  |       uses: ./ | ||||||
|  |       with: | ||||||
|  |         dependency-graph: generate-and-submit | ||||||
|  |         dependency-graph-continue-on-failure: true | ||||||
|  |     - name: Run with insufficient permissions | ||||||
|  |       working-directory: .github/workflow-samples/groovy-dsl | ||||||
|  |       run: ./gradlew help | ||||||
|  |       # This test is primarily for demonstration: it's unclear how to check for warnings emitted in the post-action | ||||||
|  |  | ||||||
|  |   SHOULD_FAIL-insufficient-permissions-failure: | ||||||
|  |     runs-on: ubuntu-latest | ||||||
|  |     permissions: | ||||||
|  |       contents: read | ||||||
|  |     continue-on-error: true | ||||||
|  |     steps: | ||||||
|  |     - name: Checkout sources | ||||||
|  |       uses: actions/checkout@v4 | ||||||
|  |     - name: Setup Gradle for dependency-graph generate | ||||||
|  |       uses: ./ | ||||||
|  |       with: | ||||||
|  |         dependency-graph: generate-and-submit | ||||||
|  |         dependency-graph-continue-on-failure: false | ||||||
|  |     - name: Run with insufficient permissions | ||||||
|  |       working-directory: .github/workflow-samples/groovy-dsl | ||||||
|  |       run: ./gradlew help | ||||||
|  |       # This test is primarily for demonstration: it's unclear how to check for a failure in the post-action | ||||||
| @@ -8,12 +8,11 @@ on: | |||||||
|       runner-os: |       runner-os: | ||||||
|         type: string |         type: string | ||||||
|         default: '["ubuntu-latest", "windows-latest", "macos-latest"]' |         default: '["ubuntu-latest", "windows-latest", "macos-latest"]' | ||||||
|       download-dist: |  | ||||||
|         type: boolean | permissions: | ||||||
|         default: false |   contents: write | ||||||
|  |  | ||||||
| env: | env: | ||||||
|   DOWNLOAD_DIST: ${{ inputs.download-dist }} |  | ||||||
|   GRADLE_BUILD_ACTION_CACHE_KEY_PREFIX: dependency-graph-${{ inputs.cache-key-prefix }} |   GRADLE_BUILD_ACTION_CACHE_KEY_PREFIX: dependency-graph-${{ inputs.cache-key-prefix }} | ||||||
|   GRADLE_BUILD_ACTION_CACHE_DEBUG_ENABLED: true |   GRADLE_BUILD_ACTION_CACHE_DEBUG_ENABLED: true | ||||||
|  |  | ||||||
| @@ -26,12 +25,10 @@ jobs: | |||||||
|     steps: |     steps: | ||||||
|     - name: Checkout sources |     - name: Checkout sources | ||||||
|       uses: actions/checkout@v4 |       uses: actions/checkout@v4 | ||||||
|     - name: Download distribution if required |  | ||||||
|       uses: ./.github/actions/download-dist |  | ||||||
|     - name: Setup Gradle for dependency-graph generate |     - name: Setup Gradle for dependency-graph generate | ||||||
|       uses: ./ |       uses: ./ | ||||||
|       with: |       with: | ||||||
|         dependency-graph: generate |         dependency-graph: generate-and-upload | ||||||
|     - name: Run gradle build |     - name: Run gradle build | ||||||
|       run: ./gradlew build |       run: ./gradlew build | ||||||
|       working-directory: .github/workflow-samples/groovy-dsl |       working-directory: .github/workflow-samples/groovy-dsl | ||||||
| @@ -44,8 +41,6 @@ jobs: | |||||||
|     steps: |     steps: | ||||||
|     - name: Checkout sources |     - name: Checkout sources | ||||||
|       uses: actions/checkout@v4 |       uses: actions/checkout@v4 | ||||||
|     - name: Download distribution if required |  | ||||||
|       uses: ./.github/actions/download-dist |  | ||||||
|     - name: Setup Gradle for dependency-graph generate |     - name: Setup Gradle for dependency-graph generate | ||||||
|       uses: ./ |       uses: ./ | ||||||
|       with: |       with: | ||||||
| @@ -55,13 +50,11 @@ jobs: | |||||||
|       working-directory: .github/workflow-samples/kotlin-dsl |       working-directory: .github/workflow-samples/kotlin-dsl | ||||||
|    |    | ||||||
|   submit: |   submit: | ||||||
|     needs: [groovy-generate, kotlin-generate] |     needs: [groovy-generate] | ||||||
|     runs-on: "ubuntu-latest" |     runs-on: "ubuntu-latest" | ||||||
|     steps: |     steps: | ||||||
|     - name: Checkout sources |     - name: Checkout sources | ||||||
|       uses: actions/checkout@v4 |       uses: actions/checkout@v4 | ||||||
|     - name: Download distribution if required |  | ||||||
|       uses: ./.github/actions/download-dist |  | ||||||
|     - name: Submit dependency graphs |     - name: Submit dependency graphs | ||||||
|       uses: ./ |       uses: ./ | ||||||
|       with: |       with: | ||||||
| @@ -75,23 +68,25 @@ jobs: | |||||||
|     steps: |     steps: | ||||||
|     - name: Checkout sources |     - name: Checkout sources | ||||||
|       uses: actions/checkout@v4 |       uses: actions/checkout@v4 | ||||||
|     - name: Download distribution if required |  | ||||||
|       uses: ./.github/actions/download-dist |  | ||||||
|     - name: Setup Gradle for dependency-graph generate |     - name: Setup Gradle for dependency-graph generate | ||||||
|       uses: ./ |       uses: ./ | ||||||
|       with: |       with: | ||||||
|         dependency-graph: generate |         dependency-graph: generate-and-submit | ||||||
|     - id: gradle-assemble |     - id: gradle-assemble | ||||||
|       run: ./gradlew assemble |       run: ./gradlew assemble | ||||||
|       working-directory: .github/workflow-samples/groovy-dsl |       working-directory: .github/workflow-samples/groovy-dsl | ||||||
|     - id: gradle-build |     - id: gradle-build | ||||||
|       run: ./gradlew build |       run: ./gradlew build | ||||||
|       working-directory: .github/workflow-samples/groovy-dsl |       working-directory: .github/workflow-samples/groovy-dsl | ||||||
|  |     - id: gradle-build-again | ||||||
|  |       run: ./gradlew build | ||||||
|  |       working-directory: .github/workflow-samples/groovy-dsl | ||||||
|     - name: Check generated dependency graphs |     - name: Check generated dependency graphs | ||||||
|       shell: bash |       shell: bash | ||||||
|       run: | |       run: | | ||||||
|         echo "gradle-assemble report file: ${{ steps.gradle-assemble.outputs.dependency-graph-file }}" |         echo "gradle-assemble report file: ${{ steps.gradle-assemble.outputs.dependency-graph-file }}" | ||||||
|         echo "gradle-build report file: ${{ steps.gradle-build.outputs.dependency-graph-file }}" |         echo "gradle-build report file: ${{ steps.gradle-build.outputs.dependency-graph-file }}" | ||||||
|  |         echo "gradle-build-again report file: ${{ steps.gradle-build-again.outputs.dependency-graph-file }}" | ||||||
|         ls -l dependency-graph-reports |         ls -l dependency-graph-reports | ||||||
|         if [ ! -e "${{ steps.gradle-assemble.outputs.dependency-graph-file }}" ]; then |         if [ ! -e "${{ steps.gradle-assemble.outputs.dependency-graph-file }}" ]; then | ||||||
|             echo "Did not find gradle-assemble dependency graph file" |             echo "Did not find gradle-assemble dependency graph file" | ||||||
| @@ -101,3 +96,39 @@ jobs: | |||||||
|             echo "Did not find gradle-build dependency graph files" |             echo "Did not find gradle-build dependency graph files" | ||||||
|             exit 1 |             exit 1 | ||||||
|         fi |         fi | ||||||
|  |         if [ ! -e "${{ steps.gradle-build-again.outputs.dependency-graph-file }}" ]; then | ||||||
|  |             echo "Did not find gradle-build-again dependency graph files" | ||||||
|  |             exit 1 | ||||||
|  |         fi | ||||||
|  |          | ||||||
|  |   config-cache: | ||||||
|  |     runs-on: ubuntu-latest | ||||||
|  |     steps: | ||||||
|  |     - name: Checkout sources | ||||||
|  |       uses: actions/checkout@v4 | ||||||
|  |     - name: Setup Gradle for dependency-graph generate | ||||||
|  |       uses: ./ | ||||||
|  |       with: | ||||||
|  |         dependency-graph: generate-and-submit | ||||||
|  |     - id: config-cache-store | ||||||
|  |       run: ./gradlew assemble --configuration-cache | ||||||
|  |       working-directory: .github/workflow-samples/groovy-dsl | ||||||
|  |     - name: Check and delete generated dependency graph | ||||||
|  |       shell: bash | ||||||
|  |       run: | | ||||||
|  |         if [ ! -e "${{ steps.config-cache-store.outputs.dependency-graph-file }}" ]; then | ||||||
|  |             echo "Did not find config-cache-store dependency graph files" | ||||||
|  |             exit 1 | ||||||
|  |         fi | ||||||
|  |         rm ${{ steps.config-cache-store.outputs.dependency-graph-file }} | ||||||
|  |     - id: config-cache-reuse | ||||||
|  |       run: ./gradlew assemble --configuration-cache | ||||||
|  |       working-directory: .github/workflow-samples/groovy-dsl | ||||||
|  |     - name: Check no dependency graph is generated | ||||||
|  |       shell: bash | ||||||
|  |       run: | | ||||||
|  |         if [ ! -z "$(ls -A dependency-graph-reports)" ]; then | ||||||
|  |             echo "Expected no dependency graph files to be generated" | ||||||
|  |             ls -l dependency-graph-reports | ||||||
|  |             exit 1 | ||||||
|  |         fi         | ||||||
|   | |||||||
| @@ -8,12 +8,8 @@ on: | |||||||
|       runner-os: |       runner-os: | ||||||
|         type: string |         type: string | ||||||
|         default: '["ubuntu-latest", "windows-latest", "macos-latest"]' |         default: '["ubuntu-latest", "windows-latest", "macos-latest"]' | ||||||
|       download-dist: |  | ||||||
|         type: boolean |  | ||||||
|         default: false |  | ||||||
|  |  | ||||||
| env: | env: | ||||||
|   DOWNLOAD_DIST: ${{ inputs.download-dist }} |  | ||||||
|   GRADLE_BUILD_ACTION_CACHE_KEY_PREFIX: detect-java-toolchain-${{ inputs.cache-key-prefix }} |   GRADLE_BUILD_ACTION_CACHE_KEY_PREFIX: detect-java-toolchain-${{ inputs.cache-key-prefix }} | ||||||
|   GRADLE_BUILD_ACTION_CACHE_DEBUG_ENABLED: true |   GRADLE_BUILD_ACTION_CACHE_DEBUG_ENABLED: true | ||||||
|  |  | ||||||
| @@ -21,21 +17,20 @@ jobs: | |||||||
|   # Test that pre-installed runner JDKs are detected |   # Test that pre-installed runner JDKs are detected | ||||||
|   pre-installed-toolchains: |   pre-installed-toolchains: | ||||||
|     strategy: |     strategy: | ||||||
|  |       fail-fast: false | ||||||
|       matrix: |       matrix: | ||||||
|         os: ${{fromJSON(inputs.runner-os)}} |         os: ${{fromJSON(inputs.runner-os)}} | ||||||
|     runs-on: ${{ matrix.os }} |     runs-on: ${{ matrix.os }} | ||||||
|     steps: |     steps: | ||||||
|     - name: Checkout sources |     - name: Checkout sources | ||||||
|       uses: actions/checkout@v4 |       uses: actions/checkout@v4 | ||||||
|     - name: Download distribution if required |  | ||||||
|       uses: ./.github/actions/download-dist |  | ||||||
|     - name: Setup Gradle |     - name: Setup Gradle | ||||||
|       uses: ./ |       uses: ./ | ||||||
|     - name: List detected toolchains |     - name: List detected toolchains | ||||||
|       shell: bash |       shell: bash | ||||||
|       working-directory: .github/workflow-samples/groovy-dsl |       working-directory: .github/workflow-samples/groovy-dsl | ||||||
|       run: |  |       run: |  | ||||||
|         gradle -q javaToolchains > output.txt |         gradle --info javaToolchains > output.txt | ||||||
|         cat output.txt |         cat output.txt | ||||||
|     - name: Verify detected toolchains |     - name: Verify detected toolchains | ||||||
|       shell: bash |       shell: bash | ||||||
| @@ -44,18 +39,18 @@ jobs: | |||||||
|         grep -q 'Eclipse Temurin JDK 1.8' output.txt || (echo "::error::Did not detect preinstalled JDK 1.8" && exit 1) |         grep -q 'Eclipse Temurin JDK 1.8' output.txt || (echo "::error::Did not detect preinstalled JDK 1.8" && exit 1) | ||||||
|         grep -q 'Eclipse Temurin JDK 11' output.txt || (echo "::error::Did not detect preinstalled JDK 11" && exit 1) |         grep -q 'Eclipse Temurin JDK 11' output.txt || (echo "::error::Did not detect preinstalled JDK 11" && exit 1) | ||||||
|         grep -q 'Eclipse Temurin JDK 17' output.txt || (echo "::error::Did not detect preinstalled JDK 17" && exit 1) |         grep -q 'Eclipse Temurin JDK 17' output.txt || (echo "::error::Did not detect preinstalled JDK 17" && exit 1) | ||||||
|  |         grep -q 'Eclipse Temurin JDK 21' output.txt || (echo "::error::Did not detect preinstalled JDK 21" && exit 1) | ||||||
|  |  | ||||||
|   # Test that JDKs provisioned by setup-java are detected |   # Test that JDKs provisioned by setup-java are detected | ||||||
|   setup-java-installed-toolchain: |   setup-java-installed-toolchain: | ||||||
|     strategy: |     strategy: | ||||||
|  |       fail-fast: false | ||||||
|       matrix: |       matrix: | ||||||
|         os: ${{fromJSON(inputs.runner-os)}} |         os: ${{fromJSON(inputs.runner-os)}} | ||||||
|     runs-on: ${{ matrix.os }} |     runs-on: ${{ matrix.os }} | ||||||
|     steps: |     steps: | ||||||
|     - name: Checkout sources |     - name: Checkout sources | ||||||
|       uses: actions/checkout@v4 |       uses: actions/checkout@v4 | ||||||
|     - name: Download distribution if required |  | ||||||
|       uses: ./.github/actions/download-dist |  | ||||||
|     - name: Setup Java 20 |     - name: Setup Java 20 | ||||||
|       uses: actions/setup-java@v4 |       uses: actions/setup-java@v4 | ||||||
|       with: |       with: | ||||||
| @@ -72,42 +67,19 @@ jobs: | |||||||
|       shell: bash |       shell: bash | ||||||
|       working-directory: .github/workflow-samples/groovy-dsl |       working-directory: .github/workflow-samples/groovy-dsl | ||||||
|       run: |  |       run: |  | ||||||
|         gradle -q javaToolchains > output.txt |         gradle --info javaToolchains > output.txt | ||||||
|         cat output.txt |         cat output.txt | ||||||
|     - name: Verify detected toolchains |     - name: Verify setup JDKs are detected | ||||||
|       shell: bash |       shell: bash | ||||||
|       working-directory: .github/workflow-samples/groovy-dsl |       working-directory: .github/workflow-samples/groovy-dsl | ||||||
|       run: |  |       run: |  | ||||||
|         grep -q 'Eclipse Temurin JDK 16' output.txt || (echo "::error::Did not detect setup-java installed JDK 16" && exit 1) |         grep -q 'Eclipse Temurin JDK 16' output.txt || (echo "::error::Did not detect setup-java installed JDK 16" && exit 1) | ||||||
|         grep -q 'Eclipse Temurin JDK 20' output.txt || (echo "::error::Did not detect setup-java installed JDK 20" && exit 1) |         grep -q 'Eclipse Temurin JDK 20' output.txt || (echo "::error::Did not detect setup-java installed JDK 20" && exit 1) | ||||||
|  |     - name: Verify pre-installed toolchains are detected | ||||||
|   # Test that predefined JDK detection property is not overwritten by action |  | ||||||
|   check-no-overwrite: |  | ||||||
|     strategy: |  | ||||||
|       matrix: |  | ||||||
|         os: ${{fromJSON(inputs.runner-os)}} |  | ||||||
|     runs-on: ${{ matrix.os }} |  | ||||||
|     steps: |  | ||||||
|     - name: Checkout sources |  | ||||||
|       uses: actions/checkout@v4 |  | ||||||
|     - name: Download distribution if required |  | ||||||
|       uses: ./.github/actions/download-dist |  | ||||||
|     - name: Configure java installations env var in Gradle User Home |  | ||||||
|       shell: bash |       shell: bash | ||||||
|  |       working-directory: .github/workflow-samples/groovy-dsl | ||||||
|       run: |  |       run: |  | ||||||
|         mkdir -p ~/.gradle |         grep -q 'Eclipse Temurin JDK 1.8' output.txt || (echo "::error::Did not detect preinstalled JDK 1.8" && exit 1) | ||||||
|         echo "org.gradle.java.installations.fromEnv=XXXXX" > ~/.gradle/gradle.properties |         grep -q 'Eclipse Temurin JDK 11' output.txt || (echo "::error::Did not detect preinstalled JDK 11" && exit 1) | ||||||
|     - name: Setup Gradle |         grep -q 'Eclipse Temurin JDK 17' output.txt || (echo "::error::Did not detect preinstalled JDK 17" && exit 1) | ||||||
|       uses: ./ |         grep -q 'Eclipse Temurin JDK 21' output.txt || (echo "::error::Did not detect preinstalled JDK 21" && exit 1) | ||||||
|     - name: Check gradle.properties |  | ||||||
|       shell: bash |  | ||||||
|       run: | |  | ||||||
|         cat ~/.gradle/gradle.properties |  | ||||||
|         if grep -q 'org.gradle.java.installations.fromEnv=JAVA_HOME' ~/.gradle/gradle.properties ; then |  | ||||||
|           echo 'Found overwritten fromEnv' |  | ||||||
|           exit 1 |  | ||||||
|         fi |  | ||||||
|         if ! grep -q 'org.gradle.java.installations.fromEnv=XXXXX' ~/.gradle/gradle.properties ; then |  | ||||||
|           echo 'Did NOT find original fromEnv' |  | ||||||
|           exit 1 |  | ||||||
|         fi |  | ||||||
|   | |||||||
| @@ -8,12 +8,8 @@ on: | |||||||
|       runner-os: |       runner-os: | ||||||
|         type: string |         type: string | ||||||
|         default: '["ubuntu-latest", "windows-latest", "macos-latest"]' |         default: '["ubuntu-latest", "windows-latest", "macos-latest"]' | ||||||
|       download-dist: |  | ||||||
|         type: boolean |  | ||||||
|         default: false |  | ||||||
|  |  | ||||||
| env: | env: | ||||||
|   DOWNLOAD_DIST: ${{ inputs.download-dist }} |  | ||||||
|   GRADLE_BUILD_ACTION_CACHE_KEY_PREFIX: execution-with-caching-${{ inputs.cache-key-prefix }} |   GRADLE_BUILD_ACTION_CACHE_KEY_PREFIX: execution-with-caching-${{ inputs.cache-key-prefix }} | ||||||
|   GRADLE_BUILD_ACTION_CACHE_DEBUG_ENABLED: true |   GRADLE_BUILD_ACTION_CACHE_DEBUG_ENABLED: true | ||||||
|  |  | ||||||
| @@ -26,8 +22,6 @@ jobs: | |||||||
|     steps: |     steps: | ||||||
|     - name: Checkout sources |     - name: Checkout sources | ||||||
|       uses: actions/checkout@v4 |       uses: actions/checkout@v4 | ||||||
|     - name: Download distribution if required |  | ||||||
|       uses: ./.github/actions/download-dist |  | ||||||
|     - name: Execute Gradle build |     - name: Execute Gradle build | ||||||
|       uses: ./ |       uses: ./ | ||||||
|       with: |       with: | ||||||
| @@ -45,8 +39,6 @@ jobs: | |||||||
|     steps: |     steps: | ||||||
|     - name: Checkout sources |     - name: Checkout sources | ||||||
|       uses: actions/checkout@v4 |       uses: actions/checkout@v4 | ||||||
|     - name: Download distribution if required |  | ||||||
|       uses: ./.github/actions/download-dist |  | ||||||
|     - name: Execute Gradle build |     - name: Execute Gradle build | ||||||
|       uses: ./ |       uses: ./ | ||||||
|       with: |       with: | ||||||
|   | |||||||
							
								
								
									
										14
									
								
								.github/workflows/integ-test-execution.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										14
									
								
								.github/workflows/integ-test-execution.yml
									
									
									
									
										vendored
									
									
								
							| @@ -8,12 +8,8 @@ on: | |||||||
|       runner-os: |       runner-os: | ||||||
|         type: string |         type: string | ||||||
|         default: '["ubuntu-latest", "windows-latest", "macos-latest"]' |         default: '["ubuntu-latest", "windows-latest", "macos-latest"]' | ||||||
|       download-dist: |  | ||||||
|         type: boolean |  | ||||||
|         default: false |  | ||||||
|  |  | ||||||
| env: | env: | ||||||
|   DOWNLOAD_DIST: ${{ inputs.download-dist }} |  | ||||||
|   GRADLE_BUILD_ACTION_CACHE_KEY_PREFIX: execution-${{ inputs.cache-key-prefix }} |   GRADLE_BUILD_ACTION_CACHE_KEY_PREFIX: execution-${{ inputs.cache-key-prefix }} | ||||||
|   GRADLE_BUILD_ACTION_CACHE_DEBUG_ENABLED: true |   GRADLE_BUILD_ACTION_CACHE_DEBUG_ENABLED: true | ||||||
|  |  | ||||||
| @@ -31,8 +27,6 @@ jobs: | |||||||
|     steps: |     steps: | ||||||
|     - name: Checkout sources |     - name: Checkout sources | ||||||
|       uses: actions/checkout@v4 |       uses: actions/checkout@v4 | ||||||
|     - name: Download distribution if required |  | ||||||
|       uses: ./.github/actions/download-dist |  | ||||||
|     - name: Test use defined Gradle version |     - name: Test use defined Gradle version | ||||||
|       uses: ./ |       uses: ./ | ||||||
|       with: |       with: | ||||||
| @@ -46,12 +40,6 @@ jobs: | |||||||
|         gradle-version: release-candidate |         gradle-version: release-candidate | ||||||
|         build-root-directory: .github/workflow-samples/no-wrapper |         build-root-directory: .github/workflow-samples/no-wrapper | ||||||
|         arguments: help |         arguments: help | ||||||
|     - name: Test use defined Gradle executable |  | ||||||
|       uses: ./ |  | ||||||
|       with: |  | ||||||
|         gradle-executable: .github/workflow-samples/groovy-dsl/gradlew${{ matrix.script-suffix }} |  | ||||||
|         build-root-directory: .github/workflow-samples/no-wrapper |  | ||||||
|         arguments: help |  | ||||||
|  |  | ||||||
|   gradle-versions: |   gradle-versions: | ||||||
|     strategy: |     strategy: | ||||||
| @@ -69,8 +57,6 @@ jobs: | |||||||
|     steps: |     steps: | ||||||
|     - name: Checkout sources |     - name: Checkout sources | ||||||
|       uses: actions/checkout@v4 |       uses: actions/checkout@v4 | ||||||
|     - name: Download distribution if required |  | ||||||
|       uses: ./.github/actions/download-dist |  | ||||||
|     - name: Setup Java |     - name: Setup Java | ||||||
|       uses: actions/setup-java@v4 |       uses: actions/setup-java@v4 | ||||||
|       with: |       with: | ||||||
|   | |||||||
							
								
								
									
										89
									
								
								.github/workflows/integ-test-inject-develocity.yml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										89
									
								
								.github/workflows/integ-test-inject-develocity.yml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,89 @@ | |||||||
|  | name: Test develocity injection | ||||||
|  |  | ||||||
|  | on: | ||||||
|  |   workflow_call: | ||||||
|  |     inputs: | ||||||
|  |       cache-key-prefix: | ||||||
|  |         type: string | ||||||
|  |       runner-os: | ||||||
|  |         type: string | ||||||
|  |         default: '["ubuntu-latest", "windows-latest", "macos-latest"]' | ||||||
|  |     secrets: | ||||||
|  |       DEVELOCITY_ACCESS_KEY: | ||||||
|  |         required: true | ||||||
|  |  | ||||||
|  | env: | ||||||
|  |   GRADLE_BUILD_ACTION_CACHE_KEY_PREFIX: provision-gradle-versions-${{ inputs.cache-key-prefix }} | ||||||
|  |   GRADLE_BUILD_ACTION_CACHE_DEBUG_ENABLED: true | ||||||
|  |  | ||||||
|  | jobs: | ||||||
|  |   inject-develocity: | ||||||
|  |     env: | ||||||
|  |       DEVELOCITY_INJECTION_ENABLED: true | ||||||
|  |       DEVELOCITY_URL: https://ge.solutions-team.gradle.com | ||||||
|  |       DEVELOCITY_PLUGIN_VERSION: 3.16.2 | ||||||
|  |       DEVELOCITY_CCUD_PLUGIN_VERSION: 1.12.1 | ||||||
|  |       GRADLE_ENTERPRISE_ACCESS_KEY: ${{ secrets.DEVELOCITY_ACCESS_KEY }} # This env var has not (yet) been renamed/aliased in GE plugin 3.16.2 | ||||||
|  |     strategy: | ||||||
|  |       matrix: | ||||||
|  |         gradle: [current, 7.6.2, 6.9.4, 5.6.4] | ||||||
|  |         os: ${{fromJSON(inputs.runner-os)}} | ||||||
|  |     runs-on: ubuntu-latest | ||||||
|  |     steps: | ||||||
|  |     - name: Checkout sources | ||||||
|  |       uses: actions/checkout@v4 | ||||||
|  |     - name: Setup Java | ||||||
|  |       uses: actions/setup-java@v4 | ||||||
|  |       with: | ||||||
|  |         distribution: temurin | ||||||
|  |         java-version: 8 | ||||||
|  |     - name: Setup Gradle | ||||||
|  |       id: setup-gradle | ||||||
|  |       uses: ./ | ||||||
|  |       with: | ||||||
|  |         cache-read-only: false # For testing, allow writing cache entries on non-default branches | ||||||
|  |         gradle-version: ${{ matrix.gradle }} | ||||||
|  |     - name: Run Gradle build | ||||||
|  |       id: gradle | ||||||
|  |       working-directory: .github/workflow-samples/no-ge | ||||||
|  |       run: gradle help | ||||||
|  |     - name: Check Build Scan url | ||||||
|  |       if: ${{ !steps.gradle.outputs.build-scan-url }} | ||||||
|  |       uses: actions/github-script@v7 | ||||||
|  |       with: | ||||||
|  |         script: | | ||||||
|  |           core.setFailed('No Build Scan detected')    | ||||||
|  |  | ||||||
|  |   build-scan-publish: | ||||||
|  |     strategy: | ||||||
|  |       matrix: | ||||||
|  |         gradle: [current, 7.6.2, 6.9.4, 5.6.4] | ||||||
|  |     runs-on: ubuntu-latest | ||||||
|  |     steps: | ||||||
|  |     - name: Checkout sources | ||||||
|  |       uses: actions/checkout@v4 | ||||||
|  |     - name: Setup Java | ||||||
|  |       uses: actions/setup-java@v4 | ||||||
|  |       with: | ||||||
|  |         distribution: temurin | ||||||
|  |         java-version: 8 | ||||||
|  |     - name: Setup Gradle | ||||||
|  |       id: setup-gradle | ||||||
|  |       uses: ./ | ||||||
|  |       with: | ||||||
|  |         cache-read-only: false # For testing, allow writing cache entries on non-default branches | ||||||
|  |         gradle-version: ${{ matrix.gradle }} | ||||||
|  |         build-scan-publish: true | ||||||
|  |         build-scan-terms-of-service-url: "https://gradle.com/terms-of-service" | ||||||
|  |         build-scan-terms-of-service-agree: "yes" | ||||||
|  |     - name: Run Gradle build | ||||||
|  |       id: gradle | ||||||
|  |       working-directory: .github/workflow-samples/no-ge | ||||||
|  |       run: gradle help | ||||||
|  |     - name: Check Build Scan url | ||||||
|  |       if: ${{ !steps.gradle.outputs.build-scan-url }} | ||||||
|  |       uses: actions/github-script@v7 | ||||||
|  |       with: | ||||||
|  |         script: | | ||||||
|  |           core.setFailed('No Build Scan detected')    | ||||||
|  |  | ||||||
| @@ -1,60 +0,0 @@ | |||||||
| name: Test gradle enterprise injection |  | ||||||
|  |  | ||||||
| on: |  | ||||||
|   workflow_call: |  | ||||||
|     inputs: |  | ||||||
|       cache-key-prefix: |  | ||||||
|         type: string |  | ||||||
|       runner-os: |  | ||||||
|         type: string |  | ||||||
|         default: '["ubuntu-latest", "windows-latest", "macos-latest"]' |  | ||||||
|       download-dist: |  | ||||||
|         type: boolean |  | ||||||
|         default: false |  | ||||||
|     secrets: |  | ||||||
|       GRADLE_ENTERPRISE_ACCESS_KEY: |  | ||||||
|         required: true |  | ||||||
|  |  | ||||||
| env: |  | ||||||
|   DOWNLOAD_DIST: ${{ inputs.download-dist }} |  | ||||||
|   GRADLE_BUILD_ACTION_CACHE_KEY_PREFIX: provision-gradle-versions-${{ inputs.cache-key-prefix }} |  | ||||||
|   GRADLE_BUILD_ACTION_CACHE_DEBUG_ENABLED: true |  | ||||||
|   GRADLE_ENTERPRISE_INJECTION_ENABLED: true |  | ||||||
|   GRADLE_ENTERPRISE_URL: https://ge.solutions-team.gradle.com |  | ||||||
|   GRADLE_ENTERPRISE_PLUGIN_VERSION: 3.15.1 |  | ||||||
|   GRADLE_ENTERPRISE_CCUD_PLUGIN_VERSION: 1.12 |  | ||||||
|   GRADLE_ENTERPRISE_ACCESS_KEY: ${{ secrets.GRADLE_ENTERPRISE_ACCESS_KEY }} |  | ||||||
|  |  | ||||||
| jobs: |  | ||||||
|   inject-gradle-enterprise: |  | ||||||
|     strategy: |  | ||||||
|       matrix: |  | ||||||
|         gradle: [current, 7.6.2, 6.9.4, 5.6.4] |  | ||||||
|         os: ${{fromJSON(inputs.runner-os)}} |  | ||||||
|     runs-on: ubuntu-latest |  | ||||||
|     steps: |  | ||||||
|     - name: Checkout sources |  | ||||||
|       uses: actions/checkout@v4 |  | ||||||
|     - name: Download distribution if required |  | ||||||
|       uses: ./.github/actions/download-dist |  | ||||||
|     - name: Setup Java |  | ||||||
|       uses: actions/setup-java@v4 |  | ||||||
|       with: |  | ||||||
|         distribution: temurin |  | ||||||
|         java-version: 8 |  | ||||||
|     - name: Setup Gradle |  | ||||||
|       id: setup-gradle |  | ||||||
|       uses: ./ |  | ||||||
|       with: |  | ||||||
|         cache-read-only: false # For testing, allow writing cache entries on non-default branches |  | ||||||
|         gradle-version: ${{ matrix.gradle }} |  | ||||||
|     - name: Run Gradle build |  | ||||||
|       id: gradle |  | ||||||
|       working-directory: .github/workflow-samples/no-ge |  | ||||||
|       run: gradle help |  | ||||||
|     - name: Check Build Scan url |  | ||||||
|       if: ${{ !steps.gradle.outputs.build-scan-url }} |  | ||||||
|       uses: actions/github-script@v7 |  | ||||||
|       with: |  | ||||||
|         script: | |  | ||||||
|           core.setFailed('No Build Scan detected')    |  | ||||||
| @@ -8,12 +8,8 @@ on: | |||||||
|       runner-os: |       runner-os: | ||||||
|         type: string |         type: string | ||||||
|         default: '["ubuntu-latest", "windows-latest", "macos-latest"]' |         default: '["ubuntu-latest", "windows-latest", "macos-latest"]' | ||||||
|       download-dist: |  | ||||||
|         type: boolean |  | ||||||
|         default: false |  | ||||||
|  |  | ||||||
| env: | env: | ||||||
|   DOWNLOAD_DIST: ${{ inputs.download-dist }} |  | ||||||
|   GRADLE_BUILD_ACTION_CACHE_KEY_PREFIX: provision-gradle-versions-${{ inputs.cache-key-prefix }} |   GRADLE_BUILD_ACTION_CACHE_KEY_PREFIX: provision-gradle-versions-${{ inputs.cache-key-prefix }} | ||||||
|   GRADLE_BUILD_ACTION_CACHE_DEBUG_ENABLED: true |   GRADLE_BUILD_ACTION_CACHE_DEBUG_ENABLED: true | ||||||
|  |  | ||||||
| @@ -31,8 +27,6 @@ jobs: | |||||||
|     steps: |     steps: | ||||||
|     - name: Checkout sources |     - name: Checkout sources | ||||||
|       uses: actions/checkout@v4 |       uses: actions/checkout@v4 | ||||||
|     - name: Download distribution if required |  | ||||||
|       uses: ./.github/actions/download-dist |  | ||||||
|     - name: Setup Gradle with v6.9 |     - name: Setup Gradle with v6.9 | ||||||
|       uses: ./ |       uses: ./ | ||||||
|       with: |       with: | ||||||
| @@ -83,8 +77,6 @@ jobs: | |||||||
|     steps: |     steps: | ||||||
|     - name: Checkout sources |     - name: Checkout sources | ||||||
|       uses: actions/checkout@v4 |       uses: actions/checkout@v4 | ||||||
|     - name: Download distribution if required |  | ||||||
|       uses: ./.github/actions/download-dist |  | ||||||
|     - name: Setup Java |     - name: Setup Java | ||||||
|       uses: actions/setup-java@v4 |       uses: actions/setup-java@v4 | ||||||
|       with: |       with: | ||||||
|   | |||||||
| @@ -8,12 +8,11 @@ on: | |||||||
|       runner-os: |       runner-os: | ||||||
|         type: string |         type: string | ||||||
|         default: '["ubuntu-latest", "windows-latest", "macos-latest"]' |         default: '["ubuntu-latest", "windows-latest", "macos-latest"]' | ||||||
|       download-dist: |     secrets: | ||||||
|         type: boolean |       GRADLE_ENCRYPTION_KEY: | ||||||
|         default: false |         required: true | ||||||
|  |  | ||||||
| env: | env: | ||||||
|   DOWNLOAD_DIST: ${{ inputs.download-dist }} |  | ||||||
|   GRADLE_BUILD_ACTION_CACHE_KEY_PREFIX: restore-configuration-cache-${{ inputs.cache-key-prefix }} |   GRADLE_BUILD_ACTION_CACHE_KEY_PREFIX: restore-configuration-cache-${{ inputs.cache-key-prefix }} | ||||||
|   GRADLE_BUILD_ACTION_CACHE_DEBUG_ENABLED: true |   GRADLE_BUILD_ACTION_CACHE_DEBUG_ENABLED: true | ||||||
|  |  | ||||||
| @@ -28,15 +27,20 @@ jobs: | |||||||
|     steps: |     steps: | ||||||
|     - name: Checkout sources |     - name: Checkout sources | ||||||
|       uses: actions/checkout@v4 |       uses: actions/checkout@v4 | ||||||
|     - name: Download distribution if required |     - name: Setup Java to ensure consistency | ||||||
|       uses: ./.github/actions/download-dist |       uses: actions/setup-java@v4 | ||||||
|  |       with: | ||||||
|  |         distribution: 'liberica' | ||||||
|  |         java-version: '21' | ||||||
|     - name: Setup Gradle |     - name: Setup Gradle | ||||||
|       uses: ./ |       uses: ./ | ||||||
|       with: |       with: | ||||||
|         cache-read-only: false # For testing, allow writing cache entries on non-default branches |         cache-read-only: false # For testing, allow writing cache entries on non-default branches | ||||||
|  |         cache-encryption-key: ${{ secrets.GRADLE_ENCRYPTION_KEY }} | ||||||
|  |         gradle-version: 8.6-rc-1 | ||||||
|     - name: Groovy build with configuration-cache enabled |     - name: Groovy build with configuration-cache enabled | ||||||
|       working-directory: .github/workflow-samples/groovy-dsl |       working-directory: .github/workflow-samples/groovy-dsl | ||||||
|       run: ./gradlew test --configuration-cache |       run: gradle test --configuration-cache | ||||||
|  |  | ||||||
|   verify-build-groovy: |   verify-build-groovy: | ||||||
|     env: |     env: | ||||||
| @@ -49,16 +53,21 @@ jobs: | |||||||
|     steps: |     steps: | ||||||
|     - name: Checkout sources |     - name: Checkout sources | ||||||
|       uses: actions/checkout@v4 |       uses: actions/checkout@v4 | ||||||
|     - name: Download distribution if required |     - name: Setup Java to ensure consistency | ||||||
|       uses: ./.github/actions/download-dist |       uses: actions/setup-java@v4 | ||||||
|  |       with: | ||||||
|  |         distribution: 'liberica' | ||||||
|  |         java-version: '21' | ||||||
|     - name: Setup Gradle |     - name: Setup Gradle | ||||||
|       uses: ./ |       uses: ./ | ||||||
|       with: |       with: | ||||||
|         cache-read-only: true |         cache-read-only: true | ||||||
|  |         cache-encryption-key: ${{ secrets.GRADLE_ENCRYPTION_KEY }} | ||||||
|  |         gradle-version: 8.6-rc-1 | ||||||
|     - name: Groovy build with configuration-cache enabled |     - name: Groovy build with configuration-cache enabled | ||||||
|       id: execute |       id: execute | ||||||
|       working-directory: .github/workflow-samples/groovy-dsl |       working-directory: .github/workflow-samples/groovy-dsl | ||||||
|       run: ./gradlew test --configuration-cache |       run: gradle test --configuration-cache | ||||||
|     - name: Check that configuration-cache was used |     - name: Check that configuration-cache was used | ||||||
|       uses: actions/github-script@v7 |       uses: actions/github-script@v7 | ||||||
|       with: |       with: | ||||||
| @@ -80,17 +89,22 @@ jobs: | |||||||
|     steps: |     steps: | ||||||
|     - name: Checkout sources |     - name: Checkout sources | ||||||
|       uses: actions/checkout@v4 |       uses: actions/checkout@v4 | ||||||
|     - name: Download distribution if required |     - name: Setup Java to ensure consistency | ||||||
|       uses: ./.github/actions/download-dist |       uses: actions/setup-java@v4 | ||||||
|  |       with: | ||||||
|  |         distribution: 'liberica' | ||||||
|  |         java-version: '21' | ||||||
|     - name: Setup Gradle with no extracted cache entries restored |     - name: Setup Gradle with no extracted cache entries restored | ||||||
|       uses: ./ |       uses: ./ | ||||||
|       env:  |       env:  | ||||||
|         GRADLE_BUILD_ACTION_SKIP_RESTORE: "generated-gradle-jars|wrapper-zips|java-toolchains|instrumented-jars|dependencies|kotlin-dsl" |         GRADLE_BUILD_ACTION_SKIP_RESTORE: "generated-gradle-jars|wrapper-zips|java-toolchains|instrumented-jars|dependencies|kotlin-dsl" | ||||||
|       with: |       with: | ||||||
|         cache-read-only: true |         cache-read-only: true | ||||||
|  |         cache-encryption-key: ${{ secrets.GRADLE_ENCRYPTION_KEY }} | ||||||
|  |         gradle-version: 8.6-rc-1 | ||||||
|     - name: Check execute Gradle build with configuration cache enabled (but not restored) |     - name: Check execute Gradle build with configuration cache enabled (but not restored) | ||||||
|       working-directory: .github/workflow-samples/groovy-dsl |       working-directory: .github/workflow-samples/groovy-dsl | ||||||
|       run: ./gradlew test --configuration-cache |       run: gradle test --configuration-cache | ||||||
|  |  | ||||||
|   seed-build-kotlin: |   seed-build-kotlin: | ||||||
|     env: |     env: | ||||||
| @@ -102,15 +116,20 @@ jobs: | |||||||
|     steps: |     steps: | ||||||
|     - name: Checkout sources |     - name: Checkout sources | ||||||
|       uses: actions/checkout@v4 |       uses: actions/checkout@v4 | ||||||
|     - name: Download distribution if required |     - name: Setup Java to ensure consistency | ||||||
|       uses: ./.github/actions/download-dist |       uses: actions/setup-java@v4 | ||||||
|  |       with: | ||||||
|  |         distribution: 'liberica' | ||||||
|  |         java-version: '21' | ||||||
|     - name: Setup Gradle |     - name: Setup Gradle | ||||||
|       uses: ./ |       uses: ./ | ||||||
|       with: |       with: | ||||||
|         cache-read-only: false # For testing, allow writing cache entries on non-default branches |         cache-read-only: false # For testing, allow writing cache entries on non-default branches | ||||||
|  |         cache-encryption-key: ${{ secrets.GRADLE_ENCRYPTION_KEY }} | ||||||
|  |         gradle-version: 8.6-rc-1 | ||||||
|     - name: Execute 'help' with configuration-cache enabled |     - name: Execute 'help' with configuration-cache enabled | ||||||
|       working-directory: .github/workflow-samples/kotlin-dsl |       working-directory: .github/workflow-samples/kotlin-dsl | ||||||
|       run: ./gradlew help --configuration-cache |       run: gradle help --configuration-cache | ||||||
|  |  | ||||||
|   modify-build-kotlin: |   modify-build-kotlin: | ||||||
|     env: |     env: | ||||||
| @@ -123,15 +142,20 @@ jobs: | |||||||
|     steps: |     steps: | ||||||
|     - name: Checkout sources |     - name: Checkout sources | ||||||
|       uses: actions/checkout@v4 |       uses: actions/checkout@v4 | ||||||
|     - name: Download distribution if required |     - name: Setup Java to ensure consistency | ||||||
|       uses: ./.github/actions/download-dist |       uses: actions/setup-java@v4 | ||||||
|  |       with: | ||||||
|  |         distribution: 'liberica' | ||||||
|  |         java-version: '21' | ||||||
|     - name: Setup Gradle |     - name: Setup Gradle | ||||||
|       uses: ./ |       uses: ./ | ||||||
|       with: |       with: | ||||||
|         cache-read-only: false # For testing, allow writing cache entries on non-default branches |         cache-read-only: false # For testing, allow writing cache entries on non-default branches | ||||||
|  |         cache-encryption-key: ${{ secrets.GRADLE_ENCRYPTION_KEY }} | ||||||
|  |         gradle-version: 8.6-rc-1 | ||||||
|     - name: Execute 'test' with configuration-cache enabled |     - name: Execute 'test' with configuration-cache enabled | ||||||
|       working-directory: .github/workflow-samples/kotlin-dsl |       working-directory: .github/workflow-samples/kotlin-dsl | ||||||
|       run: ./gradlew test --configuration-cache |       run: gradle test --configuration-cache | ||||||
|  |  | ||||||
|   # Test restore configuration-cache from the third build invocation |   # Test restore configuration-cache from the third build invocation | ||||||
|   verify-build-kotlin: |   verify-build-kotlin: | ||||||
| @@ -145,16 +169,21 @@ jobs: | |||||||
|     steps: |     steps: | ||||||
|     - name: Checkout sources |     - name: Checkout sources | ||||||
|       uses: actions/checkout@v4 |       uses: actions/checkout@v4 | ||||||
|     - name: Download distribution if required |     - name: Setup Java to ensure consistency | ||||||
|       uses: ./.github/actions/download-dist |       uses: actions/setup-java@v4 | ||||||
|  |       with: | ||||||
|  |         distribution: 'liberica' | ||||||
|  |         java-version: '21' | ||||||
|     - name: Setup Gradle |     - name: Setup Gradle | ||||||
|       uses: ./ |       uses: ./ | ||||||
|       with: |       with: | ||||||
|         cache-read-only: true |         cache-read-only: true | ||||||
|  |         cache-encryption-key: ${{ secrets.GRADLE_ENCRYPTION_KEY }} | ||||||
|  |         gradle-version: 8.6-rc-1 | ||||||
|     - name: Execute 'test' again with configuration-cache enabled |     - name: Execute 'test' again with configuration-cache enabled | ||||||
|       id: execute |       id: execute | ||||||
|       working-directory: .github/workflow-samples/kotlin-dsl |       working-directory: .github/workflow-samples/kotlin-dsl | ||||||
|       run: ./gradlew test --configuration-cache |       run: gradle test --configuration-cache | ||||||
|     - name: Check that configuration-cache was used |     - name: Check that configuration-cache was used | ||||||
|       uses: actions/github-script@v7 |       uses: actions/github-script@v7 | ||||||
|       with: |       with: | ||||||
|   | |||||||
| @@ -5,12 +5,8 @@ on: | |||||||
|     inputs: |     inputs: | ||||||
|       cache-key-prefix: |       cache-key-prefix: | ||||||
|         type: string |         type: string | ||||||
|       download-dist: |  | ||||||
|         type: boolean |  | ||||||
|         default: false |  | ||||||
|  |  | ||||||
| env: | env: | ||||||
|   DOWNLOAD_DIST: ${{ inputs.download-dist }} |  | ||||||
|   GRADLE_BUILD_ACTION_CACHE_KEY_PREFIX: restore-custom-gradle-home-${{ inputs.cache-key-prefix }} |   GRADLE_BUILD_ACTION_CACHE_KEY_PREFIX: restore-custom-gradle-home-${{ inputs.cache-key-prefix }} | ||||||
|   GRADLE_BUILD_ACTION_CACHE_DEBUG_ENABLED: true |   GRADLE_BUILD_ACTION_CACHE_DEBUG_ENABLED: true | ||||||
|  |  | ||||||
| @@ -21,8 +17,6 @@ jobs: | |||||||
|     steps: |     steps: | ||||||
|     - name: Checkout sources |     - name: Checkout sources | ||||||
|       uses: actions/checkout@v4 |       uses: actions/checkout@v4 | ||||||
|     - name: Download distribution if required |  | ||||||
|       uses: ./.github/actions/download-dist |  | ||||||
|     - name: Setup Java |     - name: Setup Java | ||||||
|       uses: actions/setup-java@v4 |       uses: actions/setup-java@v4 | ||||||
|       with: |       with: | ||||||
| @@ -44,8 +38,6 @@ jobs: | |||||||
|     steps: |     steps: | ||||||
|     - name: Checkout sources |     - name: Checkout sources | ||||||
|       uses: actions/checkout@v4 |       uses: actions/checkout@v4 | ||||||
|     - name: Download distribution if required |  | ||||||
|       uses: ./.github/actions/download-dist |  | ||||||
|     - name: Setup Java |     - name: Setup Java | ||||||
|       uses: actions/setup-java@v4 |       uses: actions/setup-java@v4 | ||||||
|       with: |       with: | ||||||
|   | |||||||
| @@ -5,12 +5,8 @@ on: | |||||||
|     inputs: |     inputs: | ||||||
|       cache-key-prefix: |       cache-key-prefix: | ||||||
|         type: string |         type: string | ||||||
|       download-dist: |  | ||||||
|         type: boolean |  | ||||||
|         default: false |  | ||||||
|  |  | ||||||
| env: | env: | ||||||
|   DOWNLOAD_DIST: ${{ inputs.download-dist }} |  | ||||||
|   GRADLE_BUILD_ACTION_CACHE_KEY_PREFIX: restore-custom-gradle-home-${{ inputs.cache-key-prefix }} |   GRADLE_BUILD_ACTION_CACHE_KEY_PREFIX: restore-custom-gradle-home-${{ inputs.cache-key-prefix }} | ||||||
|   GRADLE_BUILD_ACTION_CACHE_DEBUG_ENABLED: true |   GRADLE_BUILD_ACTION_CACHE_DEBUG_ENABLED: true | ||||||
|  |  | ||||||
| @@ -24,8 +20,6 @@ jobs: | |||||||
|         echo "GRADLE_USER_HOME=$GITHUB_WORKSPACE/gradle-user-home" >> $GITHUB_ENV |         echo "GRADLE_USER_HOME=$GITHUB_WORKSPACE/gradle-user-home" >> $GITHUB_ENV | ||||||
|     - name: Checkout sources |     - name: Checkout sources | ||||||
|       uses: actions/checkout@v4 |       uses: actions/checkout@v4 | ||||||
|     - name: Download distribution if required |  | ||||||
|       uses: ./.github/actions/download-dist |  | ||||||
|     - name: Setup Gradle |     - name: Setup Gradle | ||||||
|       uses: ./ |       uses: ./ | ||||||
|       with: |       with: | ||||||
| @@ -45,8 +39,6 @@ jobs: | |||||||
|         echo "GRADLE_USER_HOME=$GITHUB_WORKSPACE/gradle-user-home" >> $GITHUB_ENV |         echo "GRADLE_USER_HOME=$GITHUB_WORKSPACE/gradle-user-home" >> $GITHUB_ENV | ||||||
|     - name: Checkout sources |     - name: Checkout sources | ||||||
|       uses: actions/checkout@v4 |       uses: actions/checkout@v4 | ||||||
|     - name: Download distribution if required |  | ||||||
|       uses: ./.github/actions/download-dist |  | ||||||
|     - name: Setup Gradle |     - name: Setup Gradle | ||||||
|       uses: ./ |       uses: ./ | ||||||
|       with: |       with: | ||||||
| @@ -66,8 +58,6 @@ jobs: | |||||||
|         echo "GRADLE_USER_HOME=$GITHUB_WORKSPACE/gradle-user-home" >> $GITHUB_ENV |         echo "GRADLE_USER_HOME=$GITHUB_WORKSPACE/gradle-user-home" >> $GITHUB_ENV | ||||||
|     - name: Checkout sources |     - name: Checkout sources | ||||||
|       uses: actions/checkout@v4 |       uses: actions/checkout@v4 | ||||||
|     - name: Download distribution if required |  | ||||||
|       uses: ./.github/actions/download-dist |  | ||||||
|     - name: Setup Gradle |     - name: Setup Gradle | ||||||
|       uses: ./ |       uses: ./ | ||||||
|       with: |       with: | ||||||
|   | |||||||
| @@ -8,12 +8,8 @@ on: | |||||||
|       runner-os: |       runner-os: | ||||||
|         type: string |         type: string | ||||||
|         default: '["ubuntu-latest", "windows-latest", "macos-latest"]' |         default: '["ubuntu-latest", "windows-latest", "macos-latest"]' | ||||||
|       download-dist: |  | ||||||
|         type: boolean |  | ||||||
|         default: false |  | ||||||
|  |  | ||||||
| env: | env: | ||||||
|   DOWNLOAD_DIST: ${{ inputs.download-dist }} |  | ||||||
|   GRADLE_BUILD_ACTION_CACHE_KEY_PREFIX: restore-gradle-home-${{ inputs.cache-key-prefix }} |   GRADLE_BUILD_ACTION_CACHE_KEY_PREFIX: restore-gradle-home-${{ inputs.cache-key-prefix }} | ||||||
|   GRADLE_BUILD_ACTION_CACHE_KEY_JOB: restore-gradle-home |   GRADLE_BUILD_ACTION_CACHE_KEY_JOB: restore-gradle-home | ||||||
|   GRADLE_BUILD_ACTION_CACHE_DEBUG_ENABLED: true |   GRADLE_BUILD_ACTION_CACHE_DEBUG_ENABLED: true | ||||||
| @@ -27,8 +23,6 @@ jobs: | |||||||
|     steps: |     steps: | ||||||
|     - name: Checkout sources |     - name: Checkout sources | ||||||
|       uses: actions/checkout@v4 |       uses: actions/checkout@v4 | ||||||
|     - name: Download distribution if required |  | ||||||
|       uses: ./.github/actions/download-dist |  | ||||||
|     - name: Setup Gradle |     - name: Setup Gradle | ||||||
|       uses: ./ |       uses: ./ | ||||||
|       with: |       with: | ||||||
| @@ -47,8 +41,6 @@ jobs: | |||||||
|     steps: |     steps: | ||||||
|     - name: Checkout sources |     - name: Checkout sources | ||||||
|       uses: actions/checkout@v4 |       uses: actions/checkout@v4 | ||||||
|     - name: Download distribution if required |  | ||||||
|       uses: ./.github/actions/download-dist |  | ||||||
|     - name: Setup Gradle |     - name: Setup Gradle | ||||||
|       uses: ./ |       uses: ./ | ||||||
|       with: |       with: | ||||||
| @@ -67,8 +59,6 @@ jobs: | |||||||
|     steps: |     steps: | ||||||
|     - name: Checkout sources |     - name: Checkout sources | ||||||
|       uses: actions/checkout@v4 |       uses: actions/checkout@v4 | ||||||
|     - name: Download distribution if required |  | ||||||
|       uses: ./.github/actions/download-dist |  | ||||||
|     - name: Setup Gradle |     - name: Setup Gradle | ||||||
|       uses: ./ |       uses: ./ | ||||||
|       with: |       with: | ||||||
| @@ -87,8 +77,6 @@ jobs: | |||||||
|     steps: |     steps: | ||||||
|     - name: Checkout sources |     - name: Checkout sources | ||||||
|       uses: actions/checkout@v4 |       uses: actions/checkout@v4 | ||||||
|     - name: Download distribution if required |  | ||||||
|       uses: ./.github/actions/download-dist |  | ||||||
|     - name: Setup Gradle with no extracted cache entries restored |     - name: Setup Gradle with no extracted cache entries restored | ||||||
|       uses: ./ |       uses: ./ | ||||||
|       env:  |       env:  | ||||||
| @@ -109,8 +97,6 @@ jobs: | |||||||
|     steps: |     steps: | ||||||
|     - name: Checkout sources |     - name: Checkout sources | ||||||
|       uses: actions/checkout@v4 |       uses: actions/checkout@v4 | ||||||
|     - name: Download distribution if required |  | ||||||
|       uses: ./.github/actions/download-dist |  | ||||||
|     - name: Pre-create Gradle User Home |     - name: Pre-create Gradle User Home | ||||||
|       shell: bash |       shell: bash | ||||||
|       run: | |       run: | | ||||||
|   | |||||||
| @@ -8,12 +8,8 @@ on: | |||||||
|       runner-os: |       runner-os: | ||||||
|         type: string |         type: string | ||||||
|         default: '["ubuntu-latest", "windows-latest", "macos-latest"]' |         default: '["ubuntu-latest", "windows-latest", "macos-latest"]' | ||||||
|       download-dist: |  | ||||||
|         type: boolean |  | ||||||
|         default: false |  | ||||||
|  |  | ||||||
| env: | env: | ||||||
|   DOWNLOAD_DIST: ${{ inputs.download-dist }} |  | ||||||
|   GRADLE_BUILD_ACTION_CACHE_KEY_PREFIX: restore-java-toolchain-${{ inputs.cache-key-prefix }} |   GRADLE_BUILD_ACTION_CACHE_KEY_PREFIX: restore-java-toolchain-${{ inputs.cache-key-prefix }} | ||||||
|   GRADLE_BUILD_ACTION_CACHE_DEBUG_ENABLED: true |   GRADLE_BUILD_ACTION_CACHE_DEBUG_ENABLED: true | ||||||
|  |  | ||||||
| @@ -26,8 +22,6 @@ jobs: | |||||||
|     steps: |     steps: | ||||||
|     - name: Checkout sources |     - name: Checkout sources | ||||||
|       uses: actions/checkout@v4 |       uses: actions/checkout@v4 | ||||||
|     - name: Download distribution if required |  | ||||||
|       uses: ./.github/actions/download-dist |  | ||||||
|     - name: Setup Gradle |     - name: Setup Gradle | ||||||
|       uses: ./ |       uses: ./ | ||||||
|       with: |       with: | ||||||
| @@ -46,8 +40,6 @@ jobs: | |||||||
|     steps: |     steps: | ||||||
|     - name: Checkout sources |     - name: Checkout sources | ||||||
|       uses: actions/checkout@v4 |       uses: actions/checkout@v4 | ||||||
|     - name: Download distribution if required |  | ||||||
|       uses: ./.github/actions/download-dist |  | ||||||
|     - name: Setup Gradle |     - name: Setup Gradle | ||||||
|       uses: ./ |       uses: ./ | ||||||
|       with: |       with: | ||||||
|   | |||||||
| @@ -8,12 +8,8 @@ on: | |||||||
|       runner-os: |       runner-os: | ||||||
|         type: string |         type: string | ||||||
|         default: '["ubuntu-latest", "windows-latest", "macos-latest"]' |         default: '["ubuntu-latest", "windows-latest", "macos-latest"]' | ||||||
|       download-dist: |  | ||||||
|         type: boolean |  | ||||||
|         default: false |  | ||||||
|  |  | ||||||
| env: | env: | ||||||
|   DOWNLOAD_DIST: ${{ inputs.download-dist }} |  | ||||||
|   GRADLE_BUILD_ACTION_CACHE_KEY_PREFIX: sample-gradle-plugin-${{ inputs.cache-key-prefix }} |   GRADLE_BUILD_ACTION_CACHE_KEY_PREFIX: sample-gradle-plugin-${{ inputs.cache-key-prefix }} | ||||||
|   GRADLE_BUILD_ACTION_CACHE_DEBUG_ENABLED: true |   GRADLE_BUILD_ACTION_CACHE_DEBUG_ENABLED: true | ||||||
|  |  | ||||||
| @@ -26,8 +22,6 @@ jobs: | |||||||
|     steps: |     steps: | ||||||
|     - name: Checkout sources |     - name: Checkout sources | ||||||
|       uses: actions/checkout@v4 |       uses: actions/checkout@v4 | ||||||
|     - name: Download distribution if required |  | ||||||
|       uses: ./.github/actions/download-dist |  | ||||||
|     - name: Setup Gradle |     - name: Setup Gradle | ||||||
|       uses: ./ |       uses: ./ | ||||||
|       with: |       with: | ||||||
| @@ -45,8 +39,6 @@ jobs: | |||||||
|     steps: |     steps: | ||||||
|     - name: Checkout sources |     - name: Checkout sources | ||||||
|       uses: actions/checkout@v4 |       uses: actions/checkout@v4 | ||||||
|     - name: Download distribution if required |  | ||||||
|       uses: ./.github/actions/download-dist |  | ||||||
|     - name: Setup Gradle |     - name: Setup Gradle | ||||||
|       uses: ./ |       uses: ./ | ||||||
|       with: |       with: | ||||||
|   | |||||||
| @@ -8,12 +8,8 @@ on: | |||||||
|       runner-os: |       runner-os: | ||||||
|         type: string |         type: string | ||||||
|         default: '["ubuntu-latest", "windows-latest", "macos-latest"]' |         default: '["ubuntu-latest", "windows-latest", "macos-latest"]' | ||||||
|       download-dist: |  | ||||||
|         type: boolean |  | ||||||
|         default: false |  | ||||||
|  |  | ||||||
| env: | env: | ||||||
|   DOWNLOAD_DIST: ${{ inputs.download-dist }} |  | ||||||
|   GRADLE_BUILD_ACTION_CACHE_KEY_PREFIX: sample-kotlin-dsl-${{ inputs.cache-key-prefix }} |   GRADLE_BUILD_ACTION_CACHE_KEY_PREFIX: sample-kotlin-dsl-${{ inputs.cache-key-prefix }} | ||||||
|   GRADLE_BUILD_ACTION_CACHE_DEBUG_ENABLED: true |   GRADLE_BUILD_ACTION_CACHE_DEBUG_ENABLED: true | ||||||
|  |  | ||||||
| @@ -26,8 +22,6 @@ jobs: | |||||||
|     steps: |     steps: | ||||||
|     - name: Checkout sources |     - name: Checkout sources | ||||||
|       uses: actions/checkout@v4 |       uses: actions/checkout@v4 | ||||||
|     - name: Download distribution if required |  | ||||||
|       uses: ./.github/actions/download-dist |  | ||||||
|     - name: Setup Gradle |     - name: Setup Gradle | ||||||
|       uses: ./ |       uses: ./ | ||||||
|       with: |       with: | ||||||
| @@ -45,8 +39,6 @@ jobs: | |||||||
|     steps: |     steps: | ||||||
|     - name: Checkout sources |     - name: Checkout sources | ||||||
|       uses: actions/checkout@v4 |       uses: actions/checkout@v4 | ||||||
|     - name: Download distribution if required |  | ||||||
|       uses: ./.github/actions/download-dist |  | ||||||
|     - name: Setup Gradle |     - name: Setup Gradle | ||||||
|       uses: ./ |       uses: ./ | ||||||
|       with: |       with: | ||||||
|   | |||||||
							
								
								
									
										28
									
								
								.github/workflows/purge-old-workflow-runs.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										28
									
								
								.github/workflows/purge-old-workflow-runs.yml
									
									
									
									
										vendored
									
									
								
							| @@ -1,28 +0,0 @@ | |||||||
| name: Purge old workflow runs |  | ||||||
| on: |  | ||||||
|   workflow_dispatch: |  | ||||||
|     inputs: |  | ||||||
|       days: |  | ||||||
|         description: 'Purge runs older than days' |  | ||||||
|         required: true |  | ||||||
|         default: 30 |  | ||||||
|       minimum_runs: |  | ||||||
|         description: 'The minimum runs to keep for each workflow.' |  | ||||||
|         required: true |  | ||||||
|         default: 6 |  | ||||||
|       delete_workflow_pattern: |  | ||||||
|         description: 'The name of the workflow. if not set then it will target all workflows.' |  | ||||||
|         required: false |  | ||||||
|  |  | ||||||
| jobs: |  | ||||||
|   del_runs: |  | ||||||
|     runs-on: ubuntu-latest |  | ||||||
|     steps: |  | ||||||
|       - name: Purge workflow runs |  | ||||||
|         uses: Mattraks/delete-workflow-runs@v2 |  | ||||||
|         with: |  | ||||||
|           token: ${{ github.token }} |  | ||||||
|           repository: ${{ github.repository }} |  | ||||||
|           retain_days: ${{ github.event.inputs.days }} |  | ||||||
|           keep_minimum_runs: ${{ github.event.inputs.minimum_runs }} |  | ||||||
|           delete_workflow_pattern: ${{ github.event.inputs.delete_workflow_pattern }} |  | ||||||
| @@ -1,3 +0,0 @@ | |||||||
| dist/ |  | ||||||
| lib/ |  | ||||||
| node_modules/ |  | ||||||
| @@ -1,11 +0,0 @@ | |||||||
| { |  | ||||||
|     "printWidth": 120, |  | ||||||
|     "tabWidth": 4, |  | ||||||
|     "useTabs": false, |  | ||||||
|     "semi": false, |  | ||||||
|     "singleQuote": true, |  | ||||||
|     "trailingComma": "none", |  | ||||||
|     "bracketSpacing": false, |  | ||||||
|     "arrowParens": "avoid", |  | ||||||
|     "parser": "typescript" |  | ||||||
|   } |  | ||||||
| @@ -1,3 +0,0 @@ | |||||||
| # Configuration file for asdf version manager |  | ||||||
| nodejs 16.18.1 |  | ||||||
| gradle 8.5 |  | ||||||
| @@ -1,14 +0,0 @@ | |||||||
| ### How to merge a Dependabot PR |  | ||||||
|  |  | ||||||
| The "distribution" for a GitHub Action is checked into the repository itself.  |  | ||||||
| In the case of the `gradle-build-action`, the transpiled sources are committed to the `dist` directory.  |  | ||||||
| Any production dependencies are inlined into the distribution.  |  | ||||||
| So if a Dependabot PR updates a production dependency (or a dev dependency that changes the distribution, like the Typescript compiler),  |  | ||||||
| then a manual step is required to rebuild the dist and commit. |  | ||||||
|  |  | ||||||
| The simplest process to follow is: |  | ||||||
| 1. Checkout the dependabot branch locally eg: `git checkout dependabot/npm_and_yarn/actions/github-5.1.0` |  | ||||||
| 2. Run `npm install` to download and the new dependencies and install locally |  | ||||||
| 3. Run `npm run build` to regenerate the distribution |  | ||||||
| 4. Push the changes to the dependabot branch |  | ||||||
| 5. If/when the checks pass, you can merge the dependabot PR |  | ||||||
							
								
								
									
										645
									
								
								README.md
									
									
									
									
									
								
							
							
						
						
									
										645
									
								
								README.md
									
									
									
									
									
								
							| @@ -1,10 +1,26 @@ | |||||||
|  | > [!IMPORTANT] | ||||||
|  | > As of `v3` this action has been superceded by `gradle/actions/setup-gradle`. | ||||||
|  | > Any workflow that uses `gradle/gradle-build-action@v3` will transparently delegate to `gradle/actions/setup-gradle@v3`. | ||||||
|  | > | ||||||
|  | > Users are encouraged to update their workflows, replacing: | ||||||
|  | > ``` | ||||||
|  | > uses: gradle/gradle-build-action@v3 | ||||||
|  | > ``` | ||||||
|  | > | ||||||
|  | > with | ||||||
|  | > ``` | ||||||
|  | > uses: gradle/actions/setup-gradle@v3 | ||||||
|  | > ``` | ||||||
|  | > | ||||||
|  | > See the [setup-gradle documentation](https://github.com/gradle/actions/tree/main/setup-gradle) for up-to-date documentation for `gradle/actons/setup-gradle`.  | ||||||
|  |  | ||||||
| # Execute Gradle builds in GitHub Actions workflows | # Execute Gradle builds in GitHub Actions workflows | ||||||
|  |  | ||||||
| This GitHub Action can be used to configure Gradle and optionally execute a Gradle build on any platform supported by GitHub Actions. | This GitHub Action can be used to configure Gradle and optionally execute a Gradle build on any platform supported by GitHub Actions. | ||||||
|  |  | ||||||
| ## Why use the `gradle-build-action`? | ## Why use the `gradle-build-action`? | ||||||
|  |  | ||||||
| It is possible to directly invoke Gradle in your workflow, and the `actions/setup-java@v3` action provides a simple way to cache Gradle dependencies.  | It is possible to directly invoke Gradle in your workflow, and the `actions/setup-java@v4` action provides a simple way to cache Gradle dependencies.  | ||||||
|  |  | ||||||
| However, the `gradle-build-action` offers a number of advantages over this approach: | However, the `gradle-build-action` offers a number of advantages over this approach: | ||||||
|  |  | ||||||
| @@ -36,7 +52,7 @@ jobs: | |||||||
|     runs-on: ${{ matrix.os }} |     runs-on: ${{ matrix.os }} | ||||||
|     steps: |     steps: | ||||||
|     - uses: actions/checkout@v4 |     - uses: actions/checkout@v4 | ||||||
|     - uses: actions/setup-java@v3 |     - uses: actions/setup-java@v4 | ||||||
|       with: |       with: | ||||||
|         distribution: temurin |         distribution: temurin | ||||||
|         java-version: 11 |         java-version: 11 | ||||||
| @@ -85,7 +101,7 @@ jobs: | |||||||
|     runs-on: ubuntu-latest |     runs-on: ubuntu-latest | ||||||
|     steps: |     steps: | ||||||
|     - uses: actions/checkout@v4 |     - uses: actions/checkout@v4 | ||||||
|     - uses: actions/setup-java@v3 |     - uses: actions/setup-java@v4 | ||||||
|       with: |       with: | ||||||
|         distribution: temurin |         distribution: temurin | ||||||
|         java-version: 11 |         java-version: 11 | ||||||
| @@ -97,7 +113,6 @@ jobs: | |||||||
|     - run: echo "The release-candidate version was ${{ steps.setup-gradle.outputs.gradle-version }}" |     - run: echo "The release-candidate version was ${{ steps.setup-gradle.outputs.gradle-version }}" | ||||||
| ``` | ``` | ||||||
|  |  | ||||||
|  |  | ||||||
| ## Caching build state between Jobs | ## Caching build state between Jobs | ||||||
|  |  | ||||||
| The `gradle-build-action` will use the GitHub Actions cache to save and restore reusable state that may be speed up a subsequent build invocation. This includes most content that is downloaded from the internet as part of a build, as well as expensive to create content like compiled build scripts, transformed Jar files, etc. | The `gradle-build-action` will use the GitHub Actions cache to save and restore reusable state that may be speed up a subsequent build invocation. This includes most content that is downloaded from the internet as part of a build, as well as expensive to create content like compiled build scripts, transformed Jar files, etc. | ||||||
| @@ -121,7 +136,7 @@ cache-disabled: true | |||||||
|  |  | ||||||
| By default, the `gradle-build-action` will only write to the cache from Jobs on the default (`main`/`master`) branch. | By default, the `gradle-build-action` will only write to the cache from Jobs on the default (`main`/`master`) branch. | ||||||
| Jobs on other branches will read entries from the cache but will not write updated entries.  | Jobs on other branches will read entries from the cache but will not write updated entries.  | ||||||
| See [Optimizing cache effectiveness](#optimizing-cache-effectiveness) for a more detailed explanation. | See [Optimizing cache effectiveness](#select-which-branches-should-write-to-the-cache) for a more detailed explanation. | ||||||
|  |  | ||||||
| In some circumstances it makes sense to change this default, and to configure a workflow Job to read existing cache entries but not to write changes back. | In some circumstances it makes sense to change this default, and to configure a workflow Job to read existing cache entries but not to write changes back. | ||||||
|  |  | ||||||
| @@ -161,6 +176,31 @@ If you want override the default and have the `gradle-build-action` caches overw | |||||||
| cache-overwrite-existing: true | cache-overwrite-existing: true | ||||||
| ``` | ``` | ||||||
|  |  | ||||||
|  | ### Saving configuration-cache data | ||||||
|  |  | ||||||
|  | When Gradle is executed with the [configuration-cache](https://docs.gradle.org/current/userguide/configuration_cache.html) enabled, the configuration-cache data is stored | ||||||
|  | in the project directory, at `<project-dir>/.gradle/configuration-cache`. Due to the way the configuration-cache works, [this file may contain stored credentials and other | ||||||
|  | secrets](https://docs.gradle.org/release-nightly/userguide/configuration_cache.html#config_cache:secrets), and this data needs to be encrypted in order to be safely stored in the GitHub Actions cache. | ||||||
|  |  | ||||||
|  | In order to benefit from configuration caching in your GitHub Actions workflow, you must: | ||||||
|  | - Execute your build with Gradle 8.6 or newer. This can be achieved directly, or via the Gradle Wrapper. | ||||||
|  | - Enable the configuration cache for your build. | ||||||
|  | - Generate a [valid Gradle encryption key](https://docs.gradle.org/8.6-rc-1/userguide/configuration_cache.html#config_cache:secrets:configuring_encryption_key) and save it as a [GitHub Actions secret](https://docs.github.com/en/actions/security-guides/using-secrets-in-github-actions). | ||||||
|  | - Provide the secret key via the `cache-encryption-key` action parameter. | ||||||
|  |  | ||||||
|  | ```yaml | ||||||
|  | jobs: | ||||||
|  |   gradle-with-configuration-cache: | ||||||
|  |     runs-on: ubuntu-latest | ||||||
|  |     steps: | ||||||
|  |     - uses: actions/checkout@v4 | ||||||
|  |     - uses: gradle/gradle-build-action@v3 | ||||||
|  |       with: | ||||||
|  |         gradle-version: 8.6-rc-1 | ||||||
|  |         cache-encryption-key: ${{ secrets.GradleEncryptionKey }} | ||||||
|  |     - run: gradle build --configuration-cache | ||||||
|  | ``` | ||||||
|  |  | ||||||
| ### Incompatibility with other caching mechanisms | ### Incompatibility with other caching mechanisms | ||||||
|  |  | ||||||
| When using `gradle-build-action` we recommend that you avoid using other mechanisms to save and restore the Gradle User Home.  | When using `gradle-build-action` we recommend that you avoid using other mechanisms to save and restore the Gradle User Home.  | ||||||
| @@ -223,11 +263,11 @@ For this reason, it's very difficult to create a cache key that will determinist | |||||||
|  |  | ||||||
| The Gradle User Home cache key is composed of: | The Gradle User Home cache key is composed of: | ||||||
| - The current operating system (`RUNNER_OS`) | - The current operating system (`RUNNER_OS`) | ||||||
| - The workflow name and Job ID | - The Job id | ||||||
| - A hash of the Job matrix parameters | - A hash of the Job matrix parameters and the workflow name | ||||||
| - The git SHA for the latest commit | - The git SHA for the latest commit | ||||||
|  |  | ||||||
| Specifically, the cache key is: `${cache-protocol}-gradle|${runner-os}|${workflow-name}-${job-id}[${hash-of-job-matrix}]-${git-sha}` | Specifically, the cache key is: `${cache-protocol}-gradle|${runner-os}|${job-id}[${hash-of-job-matrix-and-workflow-name}]-${git-sha}` | ||||||
|  |  | ||||||
| As such, the cache key is likely to change on each subsequent run of GitHub actions.  | As such, the cache key is likely to change on each subsequent run of GitHub actions.  | ||||||
| This allows the most recent state to always be available in the GitHub actions cache. | This allows the most recent state to always be available in the GitHub actions cache. | ||||||
| @@ -235,9 +275,9 @@ This allows the most recent state to always be available in the GitHub actions c | |||||||
| ### Finding a matching cache entry | ### Finding a matching cache entry | ||||||
|  |  | ||||||
| In most cases, no exact match will exist for the cache key. Instead, the Gradle User Home will be restored for the closest matching cache entry, using a set of "restore keys". The entries will be matched with the following precedence: | In most cases, no exact match will exist for the cache key. Instead, the Gradle User Home will be restored for the closest matching cache entry, using a set of "restore keys". The entries will be matched with the following precedence: | ||||||
| - An exact match on OS, workflow, job, matrix and Git SHA | - An exact match on OS, job id, workflow name, matrix and Git SHA | ||||||
| - The most recent entry saved for the same OS, workflow, job and matrix values | - The most recent entry saved for the same OS, job id, workflow name and matrix values | ||||||
| - The most recent entry saved for the same OS, workflow and job | - The most recent entry saved for the same OS and job id | ||||||
| - The most recent entry saved for the same OS | - The most recent entry saved for the same OS | ||||||
|  |  | ||||||
| Due to branch scoping of cache entries, the above match will be first performed for entries from the same branch, and then for the default ('main') branch. | Due to branch scoping of cache entries, the above match will be first performed for entries from the same branch, and then for the default ('main') branch. | ||||||
| @@ -290,19 +330,22 @@ There are some techniques that can be used to avoid/mitigate this issue: | |||||||
| ### Select which branches should write to the cache | ### Select which branches should write to the cache | ||||||
|  |  | ||||||
| GitHub cache entries are not shared between builds on different branches.  | GitHub cache entries are not shared between builds on different branches.  | ||||||
| This means that each PR branch will have it's own Gradle User Home cache, and will not benefit from cache entries written by other PR branches. | Workflow runs can restore caches created in either the current branch or the default branch (usually main). | ||||||
| An exception to this is that cache entries written in parent and upstream branches are visible to child branches, and cache entries for the default (`master`/`main`) branch can be read by actions invoked for any other branch. | This means that each branch will have it's own Gradle User Home cache scope, and will not benefit from cache entries written for other (non-default) branches. | ||||||
|  |  | ||||||
| By default, the `gradle-build-action` will only _write_ to the cache for builds run on the default (`master`/`main`) branch.  | By default, the `gradle-build-action` will only _write_ to the cache for builds run on the default (`master`/`main`) branch.  | ||||||
| Jobs run on other branches will only read from the cache. In most cases, this is the desired behaviour,  | Jobs run on other branches will only read from the cache. In most cases, this is the desired behavior.  | ||||||
| because Jobs run against other branches will benefit from the cache Gradle User Home from `main`,  | This is because Jobs run on other branches will benefit from the cache Gradle User Home from `main`,  | ||||||
| without writing private cache entries that could lead to evicting shared entries. | without writing private cache entries that which could lead to evicting these shared entries. | ||||||
|  |  | ||||||
| If you have other long-lived development branches that would benefit from writing to the cache,  | If you have other long-lived development branches that would benefit from writing to the cache,  | ||||||
| you can configure these by overriding the `cache-read-only` action parameter.  | you can configure this by disabling the `cache-read-only` action parameter for these branches.  | ||||||
| See [Using the cache read-only](#using-the-cache-read-only) for more details. | See [Using the cache read-only](#using-the-cache-read-only) for more details. | ||||||
|  |  | ||||||
| Similarly, you could use `cache-read-only` for certain jobs in the workflow, and instead have these jobs reuse the cache content from upstream jobs. | Note there are some cases where writing cache entries is typically unhelpful (these are disabled by default): | ||||||
|  | - For `pull_request` triggered runs, the cache scope is limited to the merge ref (`refs/pull/.../merge`) and can only be restored by re-runs of the same pull request. | ||||||
|  | - For `merge_group` triggered runs, the cache scope is limited to a temporary branch with a special prefix created to validate pull request changes, and won't be available on subsequent Merge Queue executions. | ||||||
|  |  | ||||||
|    |    | ||||||
| ### Exclude content from Gradle User Home cache | ### Exclude content from Gradle User Home cache | ||||||
|  |  | ||||||
| @@ -344,62 +387,57 @@ gradle-home-cache-cleanup: true | |||||||
|  |  | ||||||
| ## Build reporting | ## Build reporting | ||||||
|  |  | ||||||
| The `gradle-build-action` collects information about any Gradle executions that occur in a workflow, and reports these via | The `gradle-build-action` collects information about any Gradle executions that occur in a workflow, including the root project, | ||||||
| a Job Summary, visible in the GitHub Actions UI. For each Gradle execution, details about the invocation are listed, together with | requested tasks, build outcome and any Build Scan link generated. Details of cache entries read and written are also collected. | ||||||
| a link to any Build Scan® published. | These details are compiled into a Job Summary, which is visible in the GitHub Actions UI. | ||||||
|  |  | ||||||
| Generation of a Job Summary is enabled by default. If this is not desired, it can be disable as follows: | Generation of a Job Summary is enabled by default for all Jobs using the `gradle-build-action`. This feature can be configured | ||||||
|  | so that a Job Summary is never generated, or so that a Job Summary is only generated on build failure: | ||||||
| ```yaml | ```yaml | ||||||
| generate-job-summary: false | add-job-summary: 'on-failure' # Valid values are 'always' (default), 'never', and 'on-failure' | ||||||
| ``` | ``` | ||||||
|  |  | ||||||
| Note that the action collects information about Gradle invocations via an [Initialization Script](https://docs.gradle.org/current/userguide/init_scripts.html#sec:using_an_init_script) | ### Adding Job Summary as a Pull Request comment | ||||||
| located at `USER_HOME/.gradle/init.d/build-result-capture.init.gradle`. |  | ||||||
| If you are using init scripts for the [Gradle Enterprise Gradle Plugin](https://plugins.gradle.org/plugin/com.gradle.enterprise) like |  | ||||||
| [`scans-init.gradle` or `gradle-enterprise-init.gradle`](https://docs.gradle.com/enterprise/gradle-plugin/#scans_gradle_com), |  | ||||||
| you'll need to ensure these files are applied prior to `build-result-capture.init.gradle`. |  | ||||||
| Since Gradle applies init scripts in alphabetical order, one way to ensure this is via file naming. |  | ||||||
|  |  | ||||||
| ### Build Scan® link as Step output | It is sometimes more convenient to view the results of a GitHub Actions Job directly from the Pull Request that triggered | ||||||
|  | the Job. For this purpose you can configure the action so that Job Summary data is added as a Pull Request comment. | ||||||
|  |  | ||||||
| As well as reporting the [Build Scan](https://gradle.com/build-scans/) link in the Job Summary, |  | ||||||
| the `gradle-build-action` action makes this link available as a Step output named `build-scan-url`. |  | ||||||
|  |  | ||||||
| You can then use that link in subsequent actions of your workflow. For example: |  | ||||||
|  |  | ||||||
| ```yaml | ```yaml | ||||||
| # .github/workflows/gradle-build-pr.yml | name: CI | ||||||
| name: Run Gradle on PRs | on: | ||||||
| on: pull_request |   pull_request: | ||||||
|  |  | ||||||
|  | permissions: | ||||||
|  |   pull-requests: write | ||||||
|  |  | ||||||
| jobs: | jobs: | ||||||
|   gradle: |   run-gradle-build: | ||||||
|     runs-on: ubuntu-latest |     runs-on: ubuntu-latest | ||||||
|     steps: |     steps: | ||||||
|     - name: Checkout project sources |     - name: Checkout project sources | ||||||
|       uses: actions/checkout@v4 |       uses: actions/checkout@v4 | ||||||
|     - name: Setup Gradle |     - name: Setup Gradle | ||||||
|       uses: gradle/gradle-build-action@v2 |       uses: gradle/gradle-build-action@v3 | ||||||
|     - name: Run build with Gradle wrapper |  | ||||||
|       id: gradle |  | ||||||
|       run: ./gradlew build --scan |  | ||||||
|     - name: "Add Build Scan URL as PR comment" |  | ||||||
|       uses: actions/github-script@v5 |  | ||||||
|       if: github.event_name == 'pull_request' && failure() |  | ||||||
|       with: |       with: | ||||||
|         github-token: ${{secrets.GITHUB_TOKEN}} |         add-job-summary-as-pr-comment: on-failure # Valid values are 'never' (default), 'always', and 'on-failure' | ||||||
|         script: | |     - run: ./gradlew build --scan | ||||||
|           github.rest.issues.createComment({ |  | ||||||
|             issue_number: context.issue.number, |  | ||||||
|             owner: context.repo.owner, |  | ||||||
|             repo: context.repo.repo, |  | ||||||
|             body: '❌ ${{ github.workflow }} failed: ${{ steps.gradle.outputs.build-scan-url }}' |  | ||||||
|           }) |  | ||||||
| ``` | ``` | ||||||
|  |  | ||||||
| ### Saving build outputs | Note that in order to add a Pull Request comment, the workflow must be configured with the `pull-requests: write` permission. | ||||||
|  |  | ||||||
| By default, a GitHub Actions workflow using `gradle-build-action` will record the log output and any Build Scan links for your build, |  | ||||||
| but any output files generated by the build will not be saved. | ### Build Scan® link as Step output | ||||||
|  |  | ||||||
|  | As well as reporting all [Build Scan](https://gradle.com/build-scans/) links in the Job Summary, | ||||||
|  | the `gradle-build-action` action makes this link available an an output of any Step that executes Gradle. | ||||||
|  |  | ||||||
|  | The output name is `build-scan-url`. You can then use the build scan link in subsequent actions of your workflow.  | ||||||
|  |  | ||||||
|  | ### Saving arbitrary build outputs | ||||||
|  |  | ||||||
|  | By default, a GitHub Actions workflow using `gradle-build-action` will record the log output and any Build Scan  | ||||||
|  | links for your build, but any output files generated by the build will not be saved. | ||||||
|  |  | ||||||
| To save selected files from your build execution, you can use the core [Upload-Artifact](https://github.com/actions/upload-artifact) action. | To save selected files from your build execution, you can use the core [Upload-Artifact](https://github.com/actions/upload-artifact) action. | ||||||
| For example: | For example: | ||||||
| @@ -423,102 +461,13 @@ jobs: | |||||||
|         path: build/reports/ |         path: build/reports/ | ||||||
| ``` | ``` | ||||||
|  |  | ||||||
| ## Use the action to invoke Gradle | ### Use of custom init-scripts in Gradle User Home | ||||||
|  |  | ||||||
| If the `gradle-build-action` is configured with an `arguments` input, then Gradle will execute a Gradle build with the arguments provided. NOTE: We recommend using the `gradle-build-action` as a "Setup Gradle" step as described above, with Gradle being invoked via a regular `run` command. | Note that the action collects information about Gradle invocations via an [Initialization Script](https://docs.gradle.org/current/userguide/init_scripts.html#sec:using_an_init_script) | ||||||
|  | located at `USER_HOME/.gradle/init.d/gradle-build-action.build-result-capture.init.gradle`. | ||||||
|  |  | ||||||
| If no `arguments` are provided, the action will not execute Gradle, but will still cache Gradle state and configure build-scan capture for all subsequent Gradle executions. | If you are adding any custom init scripts to the `USER_HOME/.gradle/init.d` directory, it may be necessary to ensure these files are applied prior to `gradle-build-action.build-result-capture.init.gradle`. | ||||||
|  | Since Gradle applies init scripts in alphabetical order, one way to ensure this is via file naming. | ||||||
| ```yaml |  | ||||||
| name: Run Gradle on PRs |  | ||||||
| on: pull_request |  | ||||||
| jobs: |  | ||||||
|   gradle: |  | ||||||
|     strategy: |  | ||||||
|       matrix: |  | ||||||
|         os: [ubuntu-latest, macos-latest, windows-latest] |  | ||||||
|     runs-on: ${{ matrix.os }} |  | ||||||
|     steps: |  | ||||||
|     - uses: actions/checkout@v4 |  | ||||||
|     - uses: actions/setup-java@v3 |  | ||||||
|       with: |  | ||||||
|         distribution: temurin |  | ||||||
|         java-version: 11 |  | ||||||
|      |  | ||||||
|     - name: Setup and execute Gradle 'test' task |  | ||||||
|       uses: gradle/gradle-build-action@v2 |  | ||||||
|       with: |  | ||||||
|         arguments: test |  | ||||||
| ``` |  | ||||||
|  |  | ||||||
| ### Multiple Gradle executions in the same Job |  | ||||||
|  |  | ||||||
| It is possible to configure multiple Gradle executions to run sequentially in the same job.  |  | ||||||
| The initial Action step will perform the Gradle setup. |  | ||||||
|  |  | ||||||
| ```yaml |  | ||||||
| - uses: gradle/gradle-build-action@v2 |  | ||||||
|   with: |  | ||||||
|     arguments: assemble |  | ||||||
| - uses: gradle/gradle-build-action@v2 |  | ||||||
|   with: |  | ||||||
|     arguments: check |  | ||||||
| ``` |  | ||||||
|  |  | ||||||
| ### Gradle command-line arguments |  | ||||||
|  |  | ||||||
| The `arguments` input can be used to pass arbitrary arguments to the `gradle` command line. |  | ||||||
| Arguments can be supplied in a single line, or as a multi-line input. |  | ||||||
|  |  | ||||||
| Here are some valid examples: |  | ||||||
| ```yaml |  | ||||||
| arguments: build |  | ||||||
| arguments: check --scan |  | ||||||
| arguments: some arbitrary tasks |  | ||||||
| arguments: build -PgradleProperty=foo |  | ||||||
| arguments: | |  | ||||||
|     build |  | ||||||
|     --scan |  | ||||||
|     -PgradleProperty=foo |  | ||||||
|     -DsystemProperty=bar |  | ||||||
| ``` |  | ||||||
|  |  | ||||||
| If you need to pass environment variables, use the GitHub Actions workflow syntax: |  | ||||||
|  |  | ||||||
| ```yaml |  | ||||||
| - uses: gradle/gradle-build-action@v2 |  | ||||||
|   env: |  | ||||||
|     CI: true |  | ||||||
|   with: |  | ||||||
|     arguments: build |  | ||||||
| ``` |  | ||||||
|  |  | ||||||
| ### Gradle build located in a subdirectory |  | ||||||
|  |  | ||||||
| By default, the action will execute Gradle in the root directory of your project.  |  | ||||||
| Use the `build-root-directory` input to target a Gradle build in a subdirectory. |  | ||||||
|  |  | ||||||
| ```yaml |  | ||||||
| - uses: gradle/gradle-build-action@v2 |  | ||||||
|   with: |  | ||||||
|     arguments: build |  | ||||||
|     build-root-directory: some/subdirectory |  | ||||||
| ``` |  | ||||||
|  |  | ||||||
| ### Using a specific Gradle executable |  | ||||||
|  |  | ||||||
| The action will first look for a Gradle wrapper script in the root directory of your project.  |  | ||||||
| If not found, `gradle` will be executed from the PATH. |  | ||||||
| Use the `gradle-executable` input to execute using a specific Gradle installation. |  | ||||||
|  |  | ||||||
| ```yaml |  | ||||||
|  - uses: gradle/gradle-build-action@v2 |  | ||||||
|    with: |  | ||||||
|      arguments: build |  | ||||||
|      gradle-executable: /path/to/installed/gradle |  | ||||||
| ``` |  | ||||||
|  |  | ||||||
| This mechanism can also be used to target a Gradle wrapper script that is located in a non-default location. |  | ||||||
|  |  | ||||||
| ## Support for GitHub Enterprise Server (GHES) | ## Support for GitHub Enterprise Server (GHES) | ||||||
|  |  | ||||||
| @@ -530,347 +479,36 @@ You can use the `gradle-build-action` on GitHub Enterprise Server, and benefit f | |||||||
|  |  | ||||||
| # GitHub Dependency Graph support | # GitHub Dependency Graph support | ||||||
|  |  | ||||||
| The `gradle-build-action` has support for submitting a [GitHub Dependency Graph](https://docs.github.com/en/code-security/supply-chain-security/understanding-your-software-supply-chain/about-the-dependency-graph) snapshot via the [GitHub Dependency Submission API](https://docs.github.com/en/rest/dependency-graph/dependency-submission?apiVersion=2022-11-28). |  | ||||||
|  |  | ||||||
| The dependency graph snapshot is generated via integration with the [GitHub Dependency Graph Gradle Plugin](https://plugins.gradle.org/plugin/org.gradle.github-dependency-graph-gradle-plugin), and saved as a workflow artifact. The generated snapshot files can be submitted either in the same job, or in a subsequent job (in the same or a dependent workflow). |  | ||||||
|  |  | ||||||
| The generated dependency graph snapshot reports all of the dependencies that were resolved during a build execution, and is used by GitHub to generate [Dependabot Alerts](https://docs.github.com/en/code-security/dependabot/dependabot-alerts/about-dependabot-alerts) for vulnerable dependencies, as well as to populate the [Dependency Graph insights view](https://docs.github.com/en/code-security/supply-chain-security/understanding-your-software-supply-chain/exploring-the-dependencies-of-a-repository#viewing-the-dependency-graph). |  | ||||||
|  |  | ||||||
| ## Enable Dependency Graph generation for a workflow |  | ||||||
|   |  | ||||||
| You enable GitHub Dependency Graph support by setting the `dependency-graph` action parameter. Valid values are: |  | ||||||
|  |  | ||||||
| | Option | Behaviour | |  | ||||||
| | --- | --- | |  | ||||||
| | `disabled`            | Do not generate a dependency graph for any build invocations.<p>This is the default. | |  | ||||||
| | `generate`            | Generate a dependency graph snapshot for each build invocation, saving as a workflow artifact. | |  | ||||||
| | `generate-and-submit` | As per `generate`, but any generated dependency graph snapshots will be submitted at the end of the job. | |  | ||||||
| | `download-and-submit` | Download any previously saved dependency graph snapshots, submitting them via the Dependency Submission API. This can be useful to collect all snapshots in a matrix of builds and submit them in one step. | |  | ||||||
|  |  | ||||||
| Example of a CI workflow that generates and submits a dependency graph: |  | ||||||
| ```yaml |  | ||||||
| name: CI build |  | ||||||
| on: |  | ||||||
|   push: |  | ||||||
|    |  | ||||||
| permissions: |  | ||||||
|   contents: write |  | ||||||
|  |  | ||||||
| jobs: |  | ||||||
|   build: |  | ||||||
|     runs-on: ubuntu-latest |  | ||||||
|     steps: |  | ||||||
|     - uses: actions/checkout@v4 |  | ||||||
|     - name: Setup Gradle to generate and submit dependency graphs |  | ||||||
|       uses: gradle/gradle-build-action@v2 |  | ||||||
|       with: |  | ||||||
|         dependency-graph: generate-and-submit |  | ||||||
|     - name: Run the usual CI build (dependency-graph will be generated and submitted post-job) |  | ||||||
|       run: ./gradlew build |  | ||||||
| ``` |  | ||||||
|  |  | ||||||
| The `contents: write` permission is required in order to submit (but not generate) the dependency graph file.  |  | ||||||
| Depending on [repository settings](https://docs.github.com/en/actions/security-guides/automatic-token-authentication#permissions-for-the-github_token), this permission may be available by default or may need to be explicitly enabled in the workflow file (as above). |  | ||||||
|  |  | ||||||
| > [!IMPORTANT] | > [!IMPORTANT] | ||||||
| > The above configuration will work for workflows that run as a result of commits to a repository branch,  | > The simplest (and recommended) way to generate a dependency graph is via a separate workflow  | ||||||
| > but not when a workflow is triggered by a PR from a repository fork. | > using `gradle/actions/dependency-submission`. This action will attempt to detect all dependencies used by your build | ||||||
| > This is because the `contents: write` permission is not available when executing a workflow  | > without building and testing the project itself. | ||||||
| > for a PR submitted from a forked repository. | > | ||||||
| > For a configuration that supports this setup, see [Dependency Graphs for pull request workflows](#dependency-graphs-for-pull-request-workflows). | > See the [dependency-submission documentation](https://github.com/gradle/actions/blob/main/dependency-submission/README.md) for up-to-date documentation. | ||||||
|  |  | ||||||
| ### Using a custom plugin repository | For documentation on directly generating a dependency graph from a Gradle execution, see the | ||||||
|  | [setup-gradle docs](https://github.com/gradle/actions/blob/main/setup-gradle/README.md#github-dependency-graph-support) on this topic. | ||||||
|  |  | ||||||
| By default, the action downloads the `github-dependency-graph-gradle-plugin` from the Gradle Plugin Portal (https://plugins.gradle.org). If your GitHub Actions environment does not have access to this URL, you can specify a custom plugin repository to use.  | # Develocity plugin injection | ||||||
| Do so by setting the `GRADLE_PLUGIN_REPOSITORY_URL` environment variable with your Gradle invocation. |  | ||||||
|  |  | ||||||
| ```yaml | The `gradle-build-action` provides support for injecting and configuring the Develocity Gradle plugin into any Gradle build, without any modification to the project sources. | ||||||
| jobs: |  | ||||||
|   build: |  | ||||||
|     runs-on: ubuntu-latest |  | ||||||
|     steps: |  | ||||||
|     - uses: actions/checkout@v4 |  | ||||||
|     - name: Setup Gradle to generate and submit dependency graphs |  | ||||||
|       uses: gradle/gradle-build-action@v2 |  | ||||||
|       with: |  | ||||||
|         dependency-graph: generate-and-submit |  | ||||||
|     - name: Run a build, resolving the 'dependency-graph' plugin from the plugin portal proxy |  | ||||||
|       run: ./gradlew build |  | ||||||
|       env: |  | ||||||
|         GRADLE_PLUGIN_REPOSITORY_URL: "https://gradle-plugins-proxy.mycorp.com" |  | ||||||
| ``` |  | ||||||
|  |  | ||||||
| ### Integrating the `dependency-review-action` |  | ||||||
|  |  | ||||||
| The GitHub [dependency-review-action](https://github.com/actions/dependency-review-action) helps you  |  | ||||||
| understand dependency changes (and the security impact of these changes) for a pull request. |  | ||||||
| For the `dependency-review-action` to succeed, it must run _after_ the dependency graph has been submitted for a PR. |  | ||||||
|  |  | ||||||
| When using `generate-and-submit`, dependency graph files are submitted at the end of the job, after all steps have been |  | ||||||
| executed. For this reason, the `dependency-review-action` must be executed in a dependent job, |  | ||||||
| and not as a subsequent step in the job that generates the dependency graph. |  | ||||||
|  |  | ||||||
| Example of a pull request workflow that executes a build for a pull request and runs the `dependency-review-action`: |  | ||||||
|  |  | ||||||
| ```yaml |  | ||||||
| name: PR check |  | ||||||
|  |  | ||||||
| on: |  | ||||||
|   pull_request: |  | ||||||
|    |  | ||||||
| permissions: |  | ||||||
|   contents: write |  | ||||||
|   # Note that this permission will not be available if the PR is from a forked repository |  | ||||||
|  |  | ||||||
| jobs: |  | ||||||
|   build: |  | ||||||
|     runs-on: ubuntu-latest |  | ||||||
|     steps: |  | ||||||
|     - uses: actions/checkout@v4 |  | ||||||
|     - name: Setup Gradle to generate and submit dependency graphs |  | ||||||
|       uses: gradle/gradle-build-action@v2 |  | ||||||
|       with: |  | ||||||
|         dependency-graph: generate-and-submit |  | ||||||
|     - name: Run a build and generate the dependency graph which will be submitted post-job |  | ||||||
|       run: ./gradlew build |  | ||||||
|  |  | ||||||
|   dependency-review: |  | ||||||
|     needs: build |  | ||||||
|     runs-on: ubuntu-latest |  | ||||||
|     - name: Perform dependency review |  | ||||||
|       uses: actions/dependency-review-action@v3 |  | ||||||
| ``` |  | ||||||
|  |  | ||||||
| See [Dependency Graphs for pull request workflows](#dependency-graphs-for-pull-request-workflows) for a more complex |  | ||||||
| (and less functional) example that will work for pull requests submitted from forked repositories. |  | ||||||
|  |  | ||||||
| ## Limiting the scope of the dependency graph |  | ||||||
|  |  | ||||||
| At times it is helpful to limit the dependencies reported to GitHub, in order to security alerts for dependencies that don't form a critical part of your product. |  | ||||||
| For example, a vulnerability in the tool you use to generate documentation is unlikely to be as important as a vulnerability in one of your runtime dependencies. |  | ||||||
|  |  | ||||||
| There are a number of techniques you can employ to limit the scope of the generated dependency graph: |  | ||||||
| - [Don't generate a dependency graph for all Gradle executions](#choosing-which-gradle-invocations-will-generate-a-dependency-graph) |  | ||||||
| - [For a Gradle execution, filter which Gradle projects and configurations will contribute dependencies](#filtering-which-gradle-configurations-contribute-to-the-dependency-graph) |  | ||||||
| - [Use a separate workflow that only resolves the required dependencies](#use-a-dedicated-workflow-for-dependency-graph-generation) |  | ||||||
|  |  | ||||||
| > [!NOTE] |  | ||||||
| > Ideally, all dependencies involved in building and testing a project will be extracted and reported in a dependency graph.  |  | ||||||
| > These dependencies would be assigned to different scopes (eg development, runtime, testing) and the GitHub UI would make it easy to opt-in to security alerts for different dependency scopes. |  | ||||||
| > However, this functionality does not yet exist. |  | ||||||
|  |  | ||||||
| ### Choosing which Gradle invocations will generate a dependency graph |  | ||||||
|  |  | ||||||
| Once you enable the dependency graph support for a workflow job (via the `dependency-graph` parameter), dependencies will be collected and reported for all subsequent Gradle invocations. |  | ||||||
| If you have a Gradle build step that you want to exclude from dependency graph generation, you can set the `GITHUB_DEPENDENCY_GRAPH_ENABLED` environment variable to `false`. |  | ||||||
|  |  | ||||||
| ```yaml |  | ||||||
| jobs: |  | ||||||
|   build: |  | ||||||
|     runs-on: ubuntu-latest |  | ||||||
|     steps: |  | ||||||
|     - uses: actions/checkout@v4 |  | ||||||
|     - name: Setup Gradle to generate and submit dependency graphs |  | ||||||
|       uses: gradle/gradle-build-action@v2 |  | ||||||
|       with: |  | ||||||
|         dependency-graph: generate-and-submit |  | ||||||
|     - name: Build the app, generating a graph of dependencies required |  | ||||||
|       run: ./gradlew :my-app:assemble |  | ||||||
|     - name: Run all checks, disabling dependency graph generation |  | ||||||
|       run: ./gradlew check |  | ||||||
|       env: |  | ||||||
|         GITHUB_DEPENDENCY_GRAPH_ENABLED: false |  | ||||||
| ``` |  | ||||||
|  |  | ||||||
| ### Filtering which Gradle Configurations contribute to the dependency graph |  | ||||||
|  |  | ||||||
| If you do not want the dependency graph to include every dependency configuration in every project in your build, you can limit the |  | ||||||
| dependency extraction to a subset of these. |  | ||||||
|  |  | ||||||
| To restrict which Gradle subprojects contribute to the report, specify which projects to include via a regular expression. |  | ||||||
| You can provide this value via the `DEPENDENCY_GRAPH_INCLUDE_PROJECTS` environment variable or system property. |  | ||||||
|  |  | ||||||
| To restrict which Gradle configurations contribute to the report, you can filter configurations by name using a regular expression. |  | ||||||
| You can provide this value via the `DEPENDENCY_GRAPH_INCLUDE_CONFIGURATIONS` environment variable or system property. |  | ||||||
|  |  | ||||||
| For example, if you want to exclude dependencies in the `buildSrc` project, and only report on dependencies from the `runtimeClasspath` configuration, |  | ||||||
| you would use the following configuration: |  | ||||||
|  |  | ||||||
| ```yaml |  | ||||||
| jobs: |  | ||||||
|   build: |  | ||||||
|     runs-on: ubuntu-latest |  | ||||||
|     steps: |  | ||||||
|     - uses: actions/checkout@v4 |  | ||||||
|     - name: Setup Gradle to generate and submit dependency graphs |  | ||||||
|       uses: gradle/gradle-build-action@v2 |  | ||||||
|       with: |  | ||||||
|         dependency-graph: generate-and-submit |  | ||||||
|     - name: Run a build, generating the dependency graph from any resolved 'runtimeClasspath' configurations |  | ||||||
|       run: ./gradlew build |  | ||||||
|       env: |  | ||||||
|         DEPENDENCY_GRAPH_INCLUDE_PROJECTS: "^:(?!buildSrc).*" |  | ||||||
|         DEPENDENCY_GRAPH_INCLUDE_CONFIGURATIONS: runtimeClasspath |  | ||||||
| ``` |  | ||||||
|  |  | ||||||
| ### Use a dedicated workflow for dependency graph generation |  | ||||||
|  |  | ||||||
| Instead of generating a dependency graph from your existing CI workflow, it's possible to create a separate dedicated workflow (or Job) that is intended for generating a dependency graph. |  | ||||||
| Such a workflow will still need to execute Gradle, but can do so in a way that is targeted at resolving the specific dependencies required. |  | ||||||
|  |  | ||||||
| For example, the following workflow will report those dependencies that are resolved in order to build the `distributionZip` for the `my-app` project. Test dependencies and other dependencies not required by the `distributionZip` will not be included. |  | ||||||
|  |  | ||||||
| ```yaml |  | ||||||
| jobs: |  | ||||||
|   build: |  | ||||||
|     runs-on: ubuntu-latest |  | ||||||
|     steps: |  | ||||||
|     - uses: actions/checkout@v4 |  | ||||||
|     - name: Setup Gradle to generate and submit dependency graphs |  | ||||||
|       uses: gradle/gradle-build-action@v2 |  | ||||||
|       with: |  | ||||||
|         dependency-graph: generate-and-submit |  | ||||||
|     - name: Build the distribution Zip for `my-app` |  | ||||||
|       run: ./gradlew :my-app:distributionZip |  | ||||||
| ``` |  | ||||||
|  |  | ||||||
| Note that the above example will also include any `buildSrc` dependencies, dependencies resolved when configuring your Gradle build or dependencies resolved while applying plugin. All of these dependencies are resolved in the process of running the `distributionZip` task, and thus will form part of the generated dependency graph. |  | ||||||
|  |  | ||||||
| If this isn't desirable, you will still need to use the filtering mechanism described above. |  | ||||||
|  |  | ||||||
| ## Dependency Graphs for pull request workflows |  | ||||||
|  |  | ||||||
| This `contents: write` permission is not available for any workflow that is triggered by a pull request submitted from a forked repository, since it would permit a malicious pull request to make repository changes.  |  | ||||||
|  |  | ||||||
| Because of this restriction, it is not possible to `generate-and-submit` a dependency graph generated for a pull-request that comes from a repository fork. In order to do so, 2 workflows will be required: |  | ||||||
| 1. The first workflow runs directly against the pull request sources and will generate the dependency graph snapshot. |  | ||||||
| 2. The second workflow is triggered on `workflow_run` of the first workflow, and will submit the previously saved dependency snapshots. |  | ||||||
|  |  | ||||||
| Note: when `download-and-submit` is used in a workflow triggered via [workflow_run](https://docs.github.com/en/actions/using-workflows/events-that-trigger-workflows#workflow_run), the action will download snapshots saved in the triggering workflow. |  | ||||||
|  |  | ||||||
| ***Main workflow file*** |  | ||||||
| ```yaml |  | ||||||
| name: run-build-and-generate-dependency-snapshot |  | ||||||
|  |  | ||||||
| on: |  | ||||||
|   pull_request: |  | ||||||
|  |  | ||||||
| jobs: |  | ||||||
|   build: |  | ||||||
|     runs-on: ubuntu-latest |  | ||||||
|     steps: |  | ||||||
|     - uses: actions/checkout@v4 |  | ||||||
|     - name: Setup Gradle to generate and submit dependency graphs |  | ||||||
|       uses: gradle/gradle-build-action@v2 |  | ||||||
|       with: |  | ||||||
|         dependency-graph: generate # Only generate in this job |  | ||||||
|     - name: Run a build, generating the dependency graph snapshot which will be submitted |  | ||||||
|       run: ./gradlew build |  | ||||||
| ``` |  | ||||||
|  |  | ||||||
| ***Dependent workflow file*** |  | ||||||
| ```yaml |  | ||||||
| name: submit-dependency-snapshot |  | ||||||
|  |  | ||||||
| on: |  | ||||||
|   workflow_run: |  | ||||||
|     workflows: ['run-build-and-generate-dependency-snapshot'] |  | ||||||
|     types: [completed] |  | ||||||
|  |  | ||||||
| jobs: |  | ||||||
|   submit-dependency-graph: |  | ||||||
|     runs-on: ubuntu-latest |  | ||||||
|     steps: |  | ||||||
|     - name: Retrieve dependency graph artifact and submit |  | ||||||
|       uses: gradle/gradle-build-action@v2 |  | ||||||
|       with: |  | ||||||
|         dependency-graph: download-and-submit |  | ||||||
| ``` |  | ||||||
|  |  | ||||||
| ### Integrating `dependency-review-action` for pull request workflows |  | ||||||
|  |  | ||||||
| The GitHub [dependency-review-action](https://github.com/actions/dependency-review-action) helps you  |  | ||||||
| understand dependency changes (and the security impact of these changes) for a pull request. |  | ||||||
|  |  | ||||||
| To integrate the `dependency-review-action` into the pull request workflows above, a separate workflow should be added. |  | ||||||
| This workflow will be triggered directly on `pull_request`, but will need to wait until the dependency graph results are |  | ||||||
| submitted before the dependency review can complete. How long to wait is controlled by the `retry-on-snapshot-warnings` input parameters. |  | ||||||
|  |  | ||||||
| Here's an example of a separate "Dependency Review" workflow that will wait for 10 minutes for the PR check workflow to complete. |  | ||||||
|  |  | ||||||
| ```yaml |  | ||||||
| name: dependency-review |  | ||||||
| on: |  | ||||||
|   pull_request: |  | ||||||
|  |  | ||||||
| permissions: |  | ||||||
|   contents: read |  | ||||||
|   pull-requests: write |  | ||||||
|  |  | ||||||
| jobs: |  | ||||||
|   dependency-review: |  | ||||||
|     runs-on: ubuntu-latest |  | ||||||
|     steps: |  | ||||||
|     - name: 'Dependency Review' |  | ||||||
|       uses: actions/dependency-review-action@v3 |  | ||||||
|       with: |  | ||||||
|         retry-on-snapshot-warnings: true |  | ||||||
|         retry-on-snapshot-warnings-timeout: 600 |  | ||||||
| ``` |  | ||||||
|  |  | ||||||
| The `retry-on-snapshot-warnings-timeout` (in seconds) needs to be long enough to allow the entire `run-build-and-generate-dependency-snapshot` and `submit-dependency-snapshot` workflows (above) to complete. |  | ||||||
|  |  | ||||||
| ## Gradle version compatibility |  | ||||||
|  |  | ||||||
| The GitHub Dependency Graph plugin should be compatible with all versions of Gradle >= 5.0, and has been tested against  |  | ||||||
| Gradle versions "5.6.4", "6.9.4", "7.0.2", "7.6.2", "8.0.2" and the current Gradle release. |  | ||||||
|  |  | ||||||
| The plugin is compatible with running Gradle with the configuration-cache enabled. However, this support is |  | ||||||
| limited to Gradle "8.1.0" and later: |  | ||||||
| - With Gradle "8.0", the build should run successfully, but an empty dependency graph will be generated. |  | ||||||
| - With Gradle <= "7.6.4", the plugin will cause the build to fail with configuration-cache enabled. |  | ||||||
|  |  | ||||||
| To use this plugin with versions of Gradle older than "8.1.0", you'll need to invoke Gradle with the |  | ||||||
| configuration-cache disabled. |  | ||||||
|  |  | ||||||
| ## Reducing storage costs for saved dependency graph artifacts |  | ||||||
|  |  | ||||||
| When `generate` or `generate-and-submit` is used with the action, the dependency graph that is generated is stored as a workflow artifact.  |  | ||||||
| By default, these artifacts are retained for a period of 30 days (or as configured for the repository). |  | ||||||
| To reduce storage costs for these artifacts, you can set the `artifact-retention-days` value to a lower number. |  | ||||||
|  |  | ||||||
| ```yaml |  | ||||||
|     steps: |  | ||||||
|     - name: Generate dependency graph, but only retain artifact for one day |  | ||||||
|       uses: gradle/gradle-build-action@v2 |  | ||||||
|       with: |  | ||||||
|         dependency-graph: generate |  | ||||||
|         artifact-retention-days: 1 |  | ||||||
| ``` |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
| # Gradle Enterprise plugin injection |  | ||||||
|  |  | ||||||
| The `gradle-build-action` provides support for injecting and configuring the Gradle Enterprise Gradle plugin into any Gradle build, without any modification to the project sources. |  | ||||||
| This is achieved via an init-script installed into Gradle User Home, which is enabled and parameterized via environment variables. | This is achieved via an init-script installed into Gradle User Home, which is enabled and parameterized via environment variables. | ||||||
|  |  | ||||||
| The same auto-injection behavior is available for the Common Custom User Data Gradle plugin, which enriches any build scans published with additional useful information. | The same auto-injection behavior is available for the Common Custom User Data Gradle plugin, which enriches any build scans published with additional useful information. | ||||||
|  |  | ||||||
| ## Enabling Gradle Enterprise injection | ## Enabling Develocity injection | ||||||
|  |  | ||||||
| In order to enable Gradle Enterprise for your build, you must provide the required configuration via environment variables.  | In order to enable Develocity injection for your build, you must provide the required configuration via environment variables.  | ||||||
|  |  | ||||||
| Here's a minimal example: | Here's a minimal example: | ||||||
|  |  | ||||||
| ```yaml | ```yaml | ||||||
| name: Run build with Gradle Enterprise injection | name: Run build with Develocity injection | ||||||
|    |    | ||||||
| env: | env: | ||||||
|   GRADLE_ENTERPRISE_INJECTION_ENABLED: true |   DEVELOCITY_INJECTION_ENABLED: true | ||||||
|   GRADLE_ENTERPRISE_URL: https://ge.gradle.org |   DEVELOCITY_URL: https://develocity.your-server.com | ||||||
|   GRADLE_ENTERPRISE_PLUGIN_VERSION: 3.15.1 |   DEVELOCITY_PLUGIN_VERSION: 3.16.2 | ||||||
|   GRADLE_ENTERPRISE_ACCESS_KEY: ${{ secrets.GE_ACCESS_KEY }} # Required to publish scans to ge.gradle.org |  | ||||||
|  |  | ||||||
| jobs: | jobs: | ||||||
|   build: |   build: | ||||||
| @@ -879,36 +517,53 @@ jobs: | |||||||
|     - uses: actions/checkout@v4 |     - uses: actions/checkout@v4 | ||||||
|     - name: Setup Gradle |     - name: Setup Gradle | ||||||
|       uses: gradle/gradle-build-action@v2 |       uses: gradle/gradle-build-action@v2 | ||||||
|     - name: Run a Gradle build with Gradle Enterprise injection enabled |     - name: Run a Gradle build with Develocity injection enabled | ||||||
|       run: ./gradlew build |       run: ./gradlew build | ||||||
| ``` | ``` | ||||||
|  |  | ||||||
| This configuration will automatically apply `v3.15.1` of the [Gradle Enterprise Gradle plugin](https://docs.gradle.com/enterprise/gradle-plugin/), and publish build scans to https://ge.gradle.org. | This configuration will automatically apply `v3.16.2` of the [Develocity Gradle plugin](https://docs.gradle.com/enterprise/gradle-plugin/), and publish build scans to https://develocity.your-server.com. | ||||||
|  |  | ||||||
| Note that the `ge.gradle.org` server requires authentication in order to publish scans. The provided `GRADLE_ENTERPRISE_ACCESS_KEY` isn't required by the Gradle Enterprise injection script,  | This example assumes that the `develocity.your-server.com` server allows anonymous publishing of build scans. | ||||||
| but will be used by the GE plugin in order to authenticate with the server. | In the likely scenario that your Develocity server requires authentication, you will also need to configure an addition environment variable | ||||||
|  | with a valid [Develocity access key](https://docs.gradle.com/enterprise/gradle-plugin/#via_environment_variable). | ||||||
|  |  | ||||||
| ## Configuring Gradle Enterprise injection | ## Configuring Develocity injection | ||||||
|  |  | ||||||
| The `init-script` supports a number of additional configuration parameters that you may fine useful. All configuration options (required and optional) are detailed below: | The `init-script` supports a number of additional configuration parameters that you may fine useful. All configuration options (required and optional) are detailed below: | ||||||
|  |  | ||||||
| | Variable                          | Required | Description | | | Variable                          | Required | Description | | ||||||
| | --- | --- | --- | | |-----------------------------------| --- | --- | | ||||||
| | GRADLE_ENTERPRISE_INJECTION_ENABLED | :white_check_mark: | enables Gradle Enterprise injection | | | DEVELOCITY_INJECTION_ENABLED      | :white_check_mark: | enables Develocity injection | | ||||||
| | GRADLE_ENTERPRISE_URL | :white_check_mark: | the URL of the Gradle Enterprise server | | | DEVELOCITY_URL                    | :white_check_mark: | the URL of the Develocity server | | ||||||
| | GRADLE_ENTERPRISE_ALLOW_UNTRUSTED_SERVER | | allow communication with an untrusted server; set to _true_ if your Gradle Enterprise instance is using a self-signed certificate | | | DEVELOCITY_ALLOW_UNTRUSTED_SERVER | | allow communication with an untrusted server; set to _true_ if your Develocity instance is using a self-signed certificate | | ||||||
| | GRADLE_ENTERPRISE_ENFORCE_URL | | enforce the configured Gradle Enterprise URL over a URL configured in the project's build; set to _true_ to enforce publication of build scans to the configured Gradle Enterprise URL | | | DEVELOCITY_ENFORCE_URL            | | enforce the configured Develocity URL over a URL configured in the project's build; set to _true_ to enforce publication of build scans to the configured Develocity URL | | ||||||
| | GRADLE_ENTERPRISE_PLUGIN_VERSION | :white_check_mark: | the version of the [Gradle Enterprise Gradle plugin](https://docs.gradle.com/enterprise/gradle-plugin/) to apply | | | DEVELOCITY_PLUGIN_VERSION         | :white_check_mark: | the version of the [Develocity Gradle plugin](https://docs.gradle.com/enterprise/gradle-plugin/) to apply | | ||||||
| | GRADLE_ENTERPRISE_CCUD_PLUGIN_VERSION |  | the version of the [Common Custom User Data Gradle plugin](https://github.com/gradle/common-custom-user-data-gradle-plugin) to apply, if any | | | DEVELOCITY_CCUD_PLUGIN_VERSION    |  | the version of the [Common Custom User Data Gradle plugin](https://github.com/gradle/common-custom-user-data-gradle-plugin) to apply, if any | | ||||||
| | GRADLE_ENTERPRISE_PLUGIN_REPOSITORY_URL |  | the URL of the repository to use when resolving the GE and CCUD plugins; the Gradle Plugin Portal is used by default | | | GRADLE_PLUGIN_REPOSITORY_URL      |  | the URL of the repository to use when resolving the Develocity and CCUD plugins; the Gradle Plugin Portal is used by default | | ||||||
|  |  | ||||||
| ## Publishing to scans.gradle.com | ## Publishing to scans.gradle.com | ||||||
|  |  | ||||||
| Gradle Enterprise injection is designed to enable publishing of build scans to a Gradle Enterprise instance, | Develocity injection is designed to enable publishing of build scans to a Develocity instance, | ||||||
| and is not suitable for publishing to the public Build Scans instance (https://scans.gradle.com). | but is also useful for publishing to the public Build Scans instance (https://scans.gradle.com). | ||||||
|  |  | ||||||
| In order to publish Build Scans to scans.gradle.com, you need to: | To publish to https://scans.gradle.com, you must specify in your workflow that you accept the [Gradle Terms of Service](https://gradle.com/terms-of-service). | ||||||
| - Apply the Gradle Enterprise plugin to your build configuration ([see docs](https://docs.gradle.com/enterprise/get-started/#applying_the_plugin)) |  | ||||||
| - Programmatically accept the Terms of Service for scans.gradle.com ([see docs](https://docs.gradle.com/enterprise/gradle-plugin/#connecting_to_scans_gradle_com)) | ```yaml | ||||||
| - Execute the build with `--scan` or configure your build with `publishAlways()` ([see docs](https://docs.gradle.com/enterprise/get-started/#always_publishing_a_build_scan)) | name: Run build and publish Build Scan | ||||||
|  |  | ||||||
|  | jobs: | ||||||
|  |   build: | ||||||
|  |     runs-on: ubuntu-latest | ||||||
|  |     steps: | ||||||
|  |     - uses: actions/checkout@v4 | ||||||
|  |     - name: Setup Gradle to publish build scans | ||||||
|  |       uses: gradle/gradle-build-action@v2 | ||||||
|  |       with: | ||||||
|  |         build-scan-publish: true | ||||||
|  |         build-scan-terms-of-service-url: "https://gradle.com/terms-of-service" | ||||||
|  |         build-scan-terms-of-service-agree: "yes" | ||||||
|  |  | ||||||
|  |     - name: Run a Gradle build - a build scan will be published automatically | ||||||
|  |       run: ./gradlew build | ||||||
|  | ``` | ||||||
|  |  | ||||||
|   | |||||||
							
								
								
									
										107
									
								
								action.yml
									
									
									
									
									
								
							
							
						
						
									
										107
									
								
								action.yml
									
									
									
									
									
								
							| @@ -8,14 +8,6 @@ inputs: | |||||||
|     description: Gradle version to use. If specified, this Gradle version will be downloaded, added to the PATH and used for invoking Gradle. |     description: Gradle version to use. If specified, this Gradle version will be downloaded, added to the PATH and used for invoking Gradle. | ||||||
|     required: false |     required: false | ||||||
|  |  | ||||||
|   gradle-executable: |  | ||||||
|     description: Path to the Gradle executable. If specified, this executable will be added to the PATH and used for invoking Gradle. |  | ||||||
|     required: false |  | ||||||
|  |  | ||||||
|   build-root-directory: |  | ||||||
|     description: Path to the root directory of the build. Default is the root of the GitHub workspace. |  | ||||||
|     required: false |  | ||||||
|  |  | ||||||
|   cache-disabled: |   cache-disabled: | ||||||
|     description: When 'true', all caching is disabled. No entries will be written to or read from the cache. |     description: When 'true', all caching is disabled. No entries will be written to or read from the cache. | ||||||
|     required: false |     required: false | ||||||
| @@ -40,6 +32,13 @@ inputs: | |||||||
|     required: false |     required: false | ||||||
|     default: false |     default: false | ||||||
|  |  | ||||||
|  |   cache-encryption-key: | ||||||
|  |     description: | | ||||||
|  |       A base64 encoded AES key used to encrypt the configuration-cache data. The key is exported as 'GRADLE_ENCRYPTION_KEY' for later steps.  | ||||||
|  |       A suitable key can be generated with `openssl rand -base64 16`. | ||||||
|  |       Configuration-cache data will not be saved/restored without an encryption key being provided. | ||||||
|  |     required: false | ||||||
|  |  | ||||||
|   gradle-home-cache-includes: |   gradle-home-cache-includes: | ||||||
|     description: Paths within Gradle User Home to cache. |     description: Paths within Gradle User Home to cache. | ||||||
|     required: false |     required: false | ||||||
| @@ -59,23 +58,61 @@ inputs: | |||||||
|     required: false |     required: false | ||||||
|     default: false |     default: false | ||||||
|  |  | ||||||
|  |   add-job-summary: | ||||||
|  |     description: Specifies when a Job Summary should be inluded in the action results. Valid values are 'never', 'always' (default), and 'on-failure'. | ||||||
|  |     required: false | ||||||
|  |     default: 'always' | ||||||
|  |  | ||||||
|  |   add-job-summary-as-pr-comment: | ||||||
|  |     description: Specifies when each Job Summary should be added as a PR comment. Valid values are 'never' (default), 'always', and 'on-failure'. No action will be taken if the workflow was not triggered from a pull request. | ||||||
|  |     required: false | ||||||
|  |     default: 'never' | ||||||
|  |  | ||||||
|  |   dependency-graph: | ||||||
|  |     description: Specifies if a GitHub dependency snapshot should be generated for each Gradle build, and if so, how. Valid values are 'disabled' (default), 'generate', 'generate-and-submit', 'generate-and-upload', 'download-and-submit' and 'clear'. | ||||||
|  |     required: false | ||||||
|  |     default: 'disabled' | ||||||
|  |  | ||||||
|  |   dependency-graph-continue-on-failure: | ||||||
|  |     description: When 'false' a failure to generate or submit a dependency graph will fail the Step or Job. When 'true' a warning will be emitted but no failure will result. | ||||||
|  |     required: false | ||||||
|  |     default: true | ||||||
|  |  | ||||||
|  |   artifact-retention-days: | ||||||
|  |     description: Specifies the number of days to retain any artifacts generated by the action. If not set, the default retention settings for the repository will apply. | ||||||
|  |     required: false | ||||||
|  |  | ||||||
|  |   build-scan-publish: | ||||||
|  |     description: | | ||||||
|  |       Set to 'true' to automatically publish build results as a Build Scan on scans.gradle.com. | ||||||
|  |       For publication to succeed without user input, you must also provide values for `build-scan-terms-of-service-url` and 'build-scan-terms-of-service-agree'. | ||||||
|  |     required: false | ||||||
|  |     default: false | ||||||
|  |  | ||||||
|  |   build-scan-terms-of-service-url: | ||||||
|  |     description: The URL to the Build Scan® terms of service. This input must be set to 'https://gradle.com/terms-of-service'. | ||||||
|  |     required: false | ||||||
|  |    | ||||||
|  |   build-scan-terms-of-service-agree: | ||||||
|  |     description: Indicate that you agree to the Build Scan® terms of service. This input value must be "yes". | ||||||
|  |     required: false | ||||||
|  |  | ||||||
|  |   # DEPRECATED ACTION INPUTS | ||||||
|   arguments: |   arguments: | ||||||
|     description: Gradle command line arguments (supports multi-line input) |     description: Gradle command line arguments (supports multi-line input) | ||||||
|     required: false |     required: false | ||||||
|  |     deprecation-message: Using the action to execute Gradle directly is deprecated in favor of using the action to setup Gradle, and executing Gradle in a subsequent Step. See https://github.com/gradle/gradle-build-action?tab=readme-ov-file#use-the-action-to-setup-gradle.  | ||||||
|  |  | ||||||
|  |   build-root-directory: | ||||||
|  |     description: Path to the root directory of the build. Default is the root of the GitHub workspace. | ||||||
|  |     required: false | ||||||
|  |     deprecation-message: Using the action to execute Gradle directly is deprecated in favor of using the action to setup Gradle, and executing Gradle in a subsequent Step. See https://github.com/gradle/gradle-build-action?tab=readme-ov-file#use-the-action-to-setup-gradle.  | ||||||
|  |  | ||||||
|   generate-job-summary: |   generate-job-summary: | ||||||
|     description: When 'false', no Job Summary will be generated for the Job. |     description: When 'false', no Job Summary will be generated for the Job. | ||||||
|     required: false |     required: false | ||||||
|     default: true |     default: true | ||||||
|  |     deprecation-message: Superceded by the new 'add-job-summary' and 'add-job-summary-as-pr-comment' parameters. | ||||||
|   dependency-graph: |  | ||||||
|     description: Specifies if a GitHub dependency snapshot should be generated for each Gradle build, and if so, how. Valid values are 'disabled' (default), 'generate', 'generate-and-submit' and 'download-and-submit'. |  | ||||||
|     required: false |  | ||||||
|     default: 'disabled' |  | ||||||
|  |  | ||||||
|   artifact-retention-days: |  | ||||||
|     description: Specifies the number of days to retain any artifacts generated by the action. If not set, the default retention settings for the repository will apply. |  | ||||||
|     required: false |  | ||||||
|  |  | ||||||
|   # EXPERIMENTAL & INTERNAL ACTION INPUTS |   # EXPERIMENTAL & INTERNAL ACTION INPUTS | ||||||
|   # The following action properties allow fine-grained tweaking of the action caching behaviour. |   # The following action properties allow fine-grained tweaking of the action caching behaviour. | ||||||
| @@ -85,6 +122,7 @@ inputs: | |||||||
|     description: When 'true', the action will not attempt to restore the Gradle User Home entries from other Jobs. |     description: When 'true', the action will not attempt to restore the Gradle User Home entries from other Jobs. | ||||||
|     required: false |     required: false | ||||||
|     default: false |     default: false | ||||||
|  |      | ||||||
|   workflow-job-context: |   workflow-job-context: | ||||||
|     description: Used to uniquely identify the current job invocation. Defaults to the matrix values for this job; this should not be overridden by users (INTERNAL). |     description: Used to uniquely identify the current job invocation. Defaults to the matrix values for this job; this should not be overridden by users (INTERNAL). | ||||||
|     required: false |     required: false | ||||||
| @@ -98,15 +136,44 @@ inputs: | |||||||
| outputs: | outputs: | ||||||
|   build-scan-url: |   build-scan-url: | ||||||
|     description: Link to the Build Scan® generated by a Gradle build. Note that this output applies to a Step executing Gradle, not to the `gradle-build-action` Step itself. |     description: Link to the Build Scan® generated by a Gradle build. Note that this output applies to a Step executing Gradle, not to the `gradle-build-action` Step itself. | ||||||
|  |     value: ${{ steps.setup-gradle.outputs.build-scan-url }} | ||||||
|   dependency-graph-file: |   dependency-graph-file: | ||||||
|     description: Path to the GitHub Dependency Graph snapshot file generated by a Gradle build. Note that this output applies to a Step executing Gradle, not to the `gradle-build-action` Step itself. |     description: Path to the GitHub Dependency Graph snapshot file generated by a Gradle build. Note that this output applies to a Step executing Gradle, not to the `gradle-build-action` Step itself. | ||||||
|  |     value: ${{ steps.setup-gradle.outputs.dependency-graph-file }} | ||||||
|   gradle-version: |   gradle-version: | ||||||
|     description: Version of Gradle that was setup by the action |     description: Version of Gradle that was setup by the action | ||||||
|  |     value: ${{ steps.setup-gradle.outputs.gradle-version }} | ||||||
|  |  | ||||||
| runs: | runs: | ||||||
|   using: 'node16' |   using: "composite" | ||||||
|   main: 'dist/main/index.js' |   steps: | ||||||
|   post: 'dist/post/index.js' |     - name: Setup Gradle | ||||||
|  |       id: setup-gradle | ||||||
|  |       uses: gradle/actions/setup-gradle@v3.0.0-rc.2 | ||||||
|  |       with: | ||||||
|  |         gradle-version: ${{ inputs.gradle-version }} | ||||||
|  |         cache-disabled: ${{ inputs.cache-disabled }} | ||||||
|  |         cache-read-only: ${{ inputs.cache-read-only }} | ||||||
|  |         cache-write-only: ${{ inputs.cache-write-only }} | ||||||
|  |         cache-overwrite-existing: ${{ inputs.cache-overwrite-existing }} | ||||||
|  |         cache-encryption-key: ${{ inputs.cache-encryption-key }} | ||||||
|  |         gradle-home-cache-includes: ${{ inputs.gradle-home-cache-includes }} | ||||||
|  |         gradle-home-cache-excludes: ${{ inputs.gradle-home-cache-excludes }} | ||||||
|  |         gradle-home-cache-cleanup: ${{ inputs.gradle-home-cache-cleanup }} | ||||||
|  |         add-job-summary: ${{ inputs.add-job-summary }} | ||||||
|  |         add-job-summary-as-pr-comment: ${{ inputs.add-job-summary-as-pr-comment }} | ||||||
|  |         dependency-graph: ${{ inputs.dependency-graph }} | ||||||
|  |         dependency-graph-continue-on-failure: ${{ inputs.dependency-graph-continue-on-failure }} | ||||||
|  |         artifact-retention-days: ${{ inputs.artifact-retention-days }} | ||||||
|  |         build-scan-publish: ${{ inputs.build-scan-publish }} | ||||||
|  |         build-scan-terms-of-service-url: ${{ inputs.build-scan-terms-of-service-url }} | ||||||
|  |         build-scan-terms-of-service-agree: ${{ inputs.build-scan-terms-of-service-agree }} | ||||||
|  |         arguments: ${{ inputs.arguments }} | ||||||
|  |         build-root-directory: ${{ inputs.build-root-directory }} | ||||||
|  |         generate-job-summary: ${{ inputs.generate-job-summary }} | ||||||
|  |         gradle-home-cache-strict-match: ${{ inputs.gradle-home-cache-strict-match }} | ||||||
|  |         workflow-job-context: ${{ inputs.workflow-job-context }} | ||||||
|  |         github-token: ${{ inputs.github-token }} | ||||||
|  |  | ||||||
| branding: | branding: | ||||||
|   icon: 'box' |   icon: 'box' | ||||||
|   | |||||||
| @@ -1,24 +0,0 @@ | |||||||
| name: 'Clear dependency graph for a correlator' |  | ||||||
|  |  | ||||||
| inputs: |  | ||||||
|   job-correlator: |  | ||||||
|     required: true |  | ||||||
|  |  | ||||||
| runs: |  | ||||||
|   using: "composite" |  | ||||||
|   steps: |  | ||||||
|   - name: Set current timestamp as env variable |  | ||||||
|     shell: bash |  | ||||||
|     run: echo "NOW=$(date -Iseconds)" >> $GITHUB_ENV |  | ||||||
|   - name: Submit empty dependency graph |  | ||||||
|     shell: bash |  | ||||||
|     run: | |  | ||||||
|       curl -L \ |  | ||||||
|       -X POST \ |  | ||||||
|       -H "Accept: application/vnd.github+json" \ |  | ||||||
|       -H "Authorization: Bearer ${{ github.token }}" \ |  | ||||||
|       -H "X-GitHub-Api-Version: 2022-11-28" \ |  | ||||||
|       https://api.github.com/repos/${{ github.repository }}/dependency-graph/snapshots \ |  | ||||||
|       -d '{ "version" : 0, "job" : { "id" : "${{ github.run_id }}", "correlator" : "${{ inputs.job-correlator }} " }, "sha" : "${{ github.sha }}", "ref" : "${{ github.ref }}",  "detector" : { "name" : "GitHub Dependency Graph Gradle Plugin", "version" : "0.0.3", "url" : "https://github.com/gradle/github-dependency-graph-gradle-plugin" }, "manifests" : {}, "scanned" : "${{ env.NOW }}" }' |  | ||||||
|   - run: echo "::notice ::Cleared dependency graph for job correlator '${{ inputs.job-correlator }}'" |  | ||||||
|     shell: bash |  | ||||||
							
								
								
									
										95095
									
								
								dist/main/index.js
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										95095
									
								
								dist/main/index.js
									
									
									
									
										vendored
									
									
								
							
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							
							
								
								
									
										1
									
								
								dist/main/index.js.map
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										1
									
								
								dist/main/index.js.map
									
									
									
									
										vendored
									
									
								
							
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							
							
								
								
									
										94727
									
								
								dist/post/index.js
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										94727
									
								
								dist/post/index.js
									
									
									
									
										vendored
									
									
								
							
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							
							
								
								
									
										1
									
								
								dist/post/index.js.map
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										1
									
								
								dist/post/index.js.map
									
									
									
									
										vendored
									
									
								
							
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							| @@ -1,10 +0,0 @@ | |||||||
| module.exports = { |  | ||||||
|   clearMocks: true, |  | ||||||
|   moduleFileExtensions: ['js', 'ts', 'json'], |  | ||||||
|   testEnvironment: 'node', |  | ||||||
|   testMatch: ['**/*.test.ts'], |  | ||||||
|   transform: { |  | ||||||
|     '^.+\\.ts$': 'ts-jest' |  | ||||||
|   }, |  | ||||||
|   verbose: true |  | ||||||
| } |  | ||||||
							
								
								
									
										14282
									
								
								package-lock.json
									
									
									
										generated
									
									
									
								
							
							
						
						
									
										14282
									
								
								package-lock.json
									
									
									
										generated
									
									
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							
							
								
								
									
										62
									
								
								package.json
									
									
									
									
									
								
							
							
						
						
									
										62
									
								
								package.json
									
									
									
									
									
								
							| @@ -1,62 +0,0 @@ | |||||||
| { |  | ||||||
|   "name": "gradle-build-action", |  | ||||||
|   "version": "1.0.0", |  | ||||||
|   "private": true, |  | ||||||
|   "description": "Execute Gradle Build", |  | ||||||
|   "scripts": { |  | ||||||
|     "postinstall": "patch-package", |  | ||||||
|     "format": "prettier --write **/*.ts", |  | ||||||
|     "format-check": "prettier --check **/*.ts", |  | ||||||
|     "lint": "eslint src/**/*.ts", |  | ||||||
|  |  | ||||||
|     "compile-main": "ncc build src/main.ts --out dist/main --source-map --no-source-map-register", |  | ||||||
|     "compile-post": "ncc build src/post.ts --out dist/post --source-map --no-source-map-register", |  | ||||||
|     "compile": "npm run compile-main && npm run compile-post", |  | ||||||
|  |  | ||||||
|     "test": "jest", |  | ||||||
|     "check": "npm run format && npm run lint", |  | ||||||
|     "build": "npm run check && npm run compile", |  | ||||||
|     "all": "npm run build && npm test" |  | ||||||
|   }, |  | ||||||
|   "repository": { |  | ||||||
|     "type": "git", |  | ||||||
|     "url": "git+https://github.com/gradle/gradle-build-action.git" |  | ||||||
|   }, |  | ||||||
|   "keywords": [ |  | ||||||
|     "github", |  | ||||||
|     "actions", |  | ||||||
|     "github-actions", |  | ||||||
|     "gradle" |  | ||||||
|   ], |  | ||||||
|   "license": "MIT", |  | ||||||
|   "dependencies": { |  | ||||||
|     "@actions/artifact": "1.1.2", |  | ||||||
|     "@actions/cache": "3.2.2", |  | ||||||
|     "@actions/core": "1.10.1", |  | ||||||
|     "@actions/exec": "1.1.1", |  | ||||||
|     "@actions/github": "5.1.1", |  | ||||||
|     "@actions/glob": "0.4.0", |  | ||||||
|     "@actions/http-client": "2.2.0", |  | ||||||
|     "@actions/tool-cache": "2.0.1", |  | ||||||
|     "@octokit/rest": "19.0.13", |  | ||||||
|     "@octokit/webhooks-types": "7.3.1", |  | ||||||
|     "string-argv": "0.3.2" |  | ||||||
|   }, |  | ||||||
|   "devDependencies": { |  | ||||||
|     "@types/node": "16.18.38", |  | ||||||
|     "@types/jest": "29.5.11", |  | ||||||
|     "@types/unzipper": "0.10.9", |  | ||||||
|     "@typescript-eslint/parser": "6.14.0", |  | ||||||
|     "@vercel/ncc": "0.38.1", |  | ||||||
|     "eslint": "8.55.0", |  | ||||||
|     "eslint-plugin-github": "4.10.1", |  | ||||||
|     "eslint-plugin-jest": "27.6.0", |  | ||||||
|     "eslint-plugin-prettier": "5.0.1", |  | ||||||
|     "jest": "29.7.0", |  | ||||||
|     "js-yaml": "4.1.0", |  | ||||||
|     "patch-package": "8.0.0", |  | ||||||
|     "prettier": "3.1.1", |  | ||||||
|     "ts-jest": "29.1.1", |  | ||||||
|     "typescript": "5.3.3" |  | ||||||
|   } |  | ||||||
| } |  | ||||||
| @@ -1,113 +0,0 @@ | |||||||
| diff --git a/node_modules/@actions/cache/lib/cache.d.ts b/node_modules/@actions/cache/lib/cache.d.ts |  | ||||||
| index 4658366..b796e58 100644 |  | ||||||
| --- a/node_modules/@actions/cache/lib/cache.d.ts |  | ||||||
| +++ b/node_modules/@actions/cache/lib/cache.d.ts |  | ||||||
| @@ -21,7 +21,7 @@ export declare function isFeatureAvailable(): boolean; |  | ||||||
|   * @param enableCrossOsArchive an optional boolean enabled to restore on windows any cache created on any platform |  | ||||||
|   * @returns string returns the key for the cache hit, otherwise returns undefined |  | ||||||
|   */ |  | ||||||
| -export declare function restoreCache(paths: string[], primaryKey: string, restoreKeys?: string[], options?: DownloadOptions, enableCrossOsArchive?: boolean): Promise<string | undefined>; |  | ||||||
| +export declare function restoreCache(paths: string[], primaryKey: string, restoreKeys?: string[], options?: DownloadOptions, enableCrossOsArchive?: boolean): Promise<CacheEntry | undefined>; |  | ||||||
|  /** |  | ||||||
|   * Saves a list of files with the specified key |  | ||||||
|   * |  | ||||||
| @@ -31,4 +31,12 @@ export declare function restoreCache(paths: string[], primaryKey: string, restor |  | ||||||
|   * @param options cache upload options |  | ||||||
|   * @returns number returns cacheId if the cache was saved successfully and throws an error if save fails |  | ||||||
|   */ |  | ||||||
| -export declare function saveCache(paths: string[], key: string, options?: UploadOptions, enableCrossOsArchive?: boolean): Promise<number>; |  | ||||||
| +export declare function saveCache(paths: string[], key: string, options?: UploadOptions, enableCrossOsArchive?: boolean): Promise<CacheEntry>; |  | ||||||
| + |  | ||||||
| +// PATCHED: Add `CacheEntry` as return type for save/restore functions |  | ||||||
| +// This allows us to track and report on cache entry sizes. |  | ||||||
| +export declare class CacheEntry { |  | ||||||
| +    key: string; |  | ||||||
| +    size?: number; |  | ||||||
| +    constructor(key: string, size?: number); |  | ||||||
| +} |  | ||||||
| diff --git a/node_modules/@actions/cache/lib/cache.js b/node_modules/@actions/cache/lib/cache.js |  | ||||||
| index 9d636aa..a176bd7 100644 |  | ||||||
| --- a/node_modules/@actions/cache/lib/cache.js |  | ||||||
| +++ b/node_modules/@actions/cache/lib/cache.js |  | ||||||
| @@ -127,18 +127,21 @@ function restoreCache(paths, primaryKey, restoreKeys, options, enableCrossOsArch |  | ||||||
|              core.info(`Cache Size: ~${Math.round(archiveFileSize / (1024 * 1024))} MB (${archiveFileSize} B)`); |  | ||||||
|              yield (0, tar_1.extractTar)(archivePath, compressionMethod); |  | ||||||
|              core.info('Cache restored successfully'); |  | ||||||
| -            return cacheEntry.cacheKey; |  | ||||||
| -        } |  | ||||||
| -        catch (error) { |  | ||||||
| -            const typedError = error; |  | ||||||
| -            if (typedError.name === ValidationError.name) { |  | ||||||
| -                throw error; |  | ||||||
| -            } |  | ||||||
| -            else { |  | ||||||
| -                // Supress all non-validation cache related errors because caching should be optional |  | ||||||
| -                core.warning(`Failed to restore: ${error.message}`); |  | ||||||
| -            } |  | ||||||
| + |  | ||||||
| +            // PATCHED - Return more inforamtion about restored entry |  | ||||||
| +            return new CacheEntry(cacheEntry.cacheKey, archiveFileSize);; |  | ||||||
|          } |  | ||||||
| +        // PATCHED - propagate errors |  | ||||||
| +        // catch (error) { |  | ||||||
| +        //     const typedError = error; |  | ||||||
| +        //     if (typedError.name === ValidationError.name) { |  | ||||||
| +        //         throw error; |  | ||||||
| +        //     } |  | ||||||
| +        //     else { |  | ||||||
| +        //         // Supress all non-validation cache related errors because caching should be optional |  | ||||||
| +        //         core.warning(`Failed to restore: ${error.message}`); |  | ||||||
| +        //     } |  | ||||||
| +        // } |  | ||||||
|          finally { |  | ||||||
|              // Try to delete the archive to save space |  | ||||||
|              try { |  | ||||||
| @@ -206,19 +209,23 @@ function saveCache(paths, key, options, enableCrossOsArchive = false) { |  | ||||||
|              } |  | ||||||
|              core.debug(`Saving Cache (ID: ${cacheId})`); |  | ||||||
|              yield cacheHttpClient.saveCache(cacheId, archivePath, options); |  | ||||||
| + |  | ||||||
| +            // PATCHED - Return more inforamtion about saved entry |  | ||||||
| +            return new CacheEntry(key, archiveFileSize); |  | ||||||
|          } |  | ||||||
| -        catch (error) { |  | ||||||
| -            const typedError = error; |  | ||||||
| -            if (typedError.name === ValidationError.name) { |  | ||||||
| -                throw error; |  | ||||||
| -            } |  | ||||||
| -            else if (typedError.name === ReserveCacheError.name) { |  | ||||||
| -                core.info(`Failed to save: ${typedError.message}`); |  | ||||||
| -            } |  | ||||||
| -            else { |  | ||||||
| -                core.warning(`Failed to save: ${typedError.message}`); |  | ||||||
| -            } |  | ||||||
| -        } |  | ||||||
| +        // PATCHED - propagate errors |  | ||||||
| +        // catch (error) { |  | ||||||
| +        //     const typedError = error; |  | ||||||
| +        //     if (typedError.name === ValidationError.name) { |  | ||||||
| +        //         throw error; |  | ||||||
| +        //     } |  | ||||||
| +        //     else if (typedError.name === ReserveCacheError.name) { |  | ||||||
| +        //         core.info(`Failed to save: ${typedError.message}`); |  | ||||||
| +        //     } |  | ||||||
| +        //     else { |  | ||||||
| +        //         core.warning(`Failed to save: ${typedError.message}`); |  | ||||||
| +        //     } |  | ||||||
| +        // } |  | ||||||
|          finally { |  | ||||||
|              // Try to delete the archive to save space |  | ||||||
|              try { |  | ||||||
| @@ -232,4 +239,11 @@ function saveCache(paths, key, options, enableCrossOsArchive = false) { |  | ||||||
|      }); |  | ||||||
|  } |  | ||||||
|  exports.saveCache = saveCache; |  | ||||||
| +class CacheEntry { |  | ||||||
| +    constructor(key, size) { |  | ||||||
| +        this.key = key; |  | ||||||
| +        this.size = size; |  | ||||||
| +    } |  | ||||||
| +} |  | ||||||
| +exports.CacheEntry = CacheEntry; |  | ||||||
|  //# sourceMappingURL=cache.js.map |  | ||||||
| \ No newline at end of file |  | ||||||
| @@ -1,48 +0,0 @@ | |||||||
| diff --git a/node_modules/@azure/logger/dist-esm/src/debug.js b/node_modules/@azure/logger/dist-esm/src/debug.js |  | ||||||
| index d202779..30e8313 100644 |  | ||||||
| --- a/node_modules/@azure/logger/dist-esm/src/debug.js |  | ||||||
| +++ b/node_modules/@azure/logger/dist-esm/src/debug.js |  | ||||||
| @@ -7,7 +7,7 @@ let enabledNamespaces = []; |  | ||||||
|  let skippedNamespaces = []; |  | ||||||
|  const debuggers = []; |  | ||||||
|  if (debugEnvVariable) { |  | ||||||
| -    enable(debugEnvVariable); |  | ||||||
| +    // enable(debugEnvVariable); |  | ||||||
|  } |  | ||||||
|  const debugObj = Object.assign((namespace) => { |  | ||||||
|      return createDebugger(namespace); |  | ||||||
| diff --git a/node_modules/@azure/logger/dist-esm/src/index.js b/node_modules/@azure/logger/dist-esm/src/index.js |  | ||||||
| index cc25720..2925db5 100644 |  | ||||||
| --- a/node_modules/@azure/logger/dist-esm/src/index.js |  | ||||||
| +++ b/node_modules/@azure/logger/dist-esm/src/index.js |  | ||||||
| @@ -20,7 +20,7 @@ if (logLevelFromEnv) { |  | ||||||
|          setLogLevel(logLevelFromEnv); |  | ||||||
|      } |  | ||||||
|      else { |  | ||||||
| -        console.error(`AZURE_LOG_LEVEL set to unknown log level '${logLevelFromEnv}'; logging is not enabled. Acceptable values: ${AZURE_LOG_LEVELS.join(", ")}.`); |  | ||||||
| +        console.error(`AZURE_LOG_LEVEL set to unknown log level; logging is not enabled. Acceptable values: ${AZURE_LOG_LEVELS.join(", ")}.`); |  | ||||||
|      } |  | ||||||
|  } |  | ||||||
|  /** |  | ||||||
| diff --git a/node_modules/@azure/logger/dist/index.js b/node_modules/@azure/logger/dist/index.js |  | ||||||
| index 81e97c3..a415e2f 100644 |  | ||||||
| --- a/node_modules/@azure/logger/dist/index.js |  | ||||||
| +++ b/node_modules/@azure/logger/dist/index.js |  | ||||||
| @@ -21,7 +21,7 @@ let enabledNamespaces = []; |  | ||||||
|  let skippedNamespaces = []; |  | ||||||
|  const debuggers = []; |  | ||||||
|  if (debugEnvVariable) { |  | ||||||
| -    enable(debugEnvVariable); |  | ||||||
| +    // enable(debugEnvVariable); |  | ||||||
|  } |  | ||||||
|  const debugObj = Object.assign((namespace) => { |  | ||||||
|      return createDebugger(namespace); |  | ||||||
| @@ -125,7 +125,7 @@ if (logLevelFromEnv) { |  | ||||||
|          setLogLevel(logLevelFromEnv); |  | ||||||
|      } |  | ||||||
|      else { |  | ||||||
| -        console.error(`AZURE_LOG_LEVEL set to unknown log level '${logLevelFromEnv}'; logging is not enabled. Acceptable values: ${AZURE_LOG_LEVELS.join(", ")}.`); |  | ||||||
| +        console.error(`AZURE_LOG_LEVEL set to unknown log level; logging is not enabled. Acceptable values: ${AZURE_LOG_LEVELS.join(", ")}.`); |  | ||||||
|      } |  | ||||||
|  } |  | ||||||
|  /** |  | ||||||
| @@ -1,27 +0,0 @@ | |||||||
| import * as fs from 'fs' |  | ||||||
| import * as path from 'path' |  | ||||||
|  |  | ||||||
| export interface BuildResult { |  | ||||||
|     get rootProjectName(): string |  | ||||||
|     get rootProjectDir(): string |  | ||||||
|     get requestedTasks(): string |  | ||||||
|     get gradleVersion(): string |  | ||||||
|     get gradleHomeDir(): string |  | ||||||
|     get buildFailed(): boolean |  | ||||||
|     get buildScanUri(): string |  | ||||||
|     get buildScanFailed(): boolean |  | ||||||
| } |  | ||||||
|  |  | ||||||
| export function loadBuildResults(): BuildResult[] { |  | ||||||
|     const buildResultsDir = path.resolve(process.env['RUNNER_TEMP']!, '.build-results') |  | ||||||
|     if (!fs.existsSync(buildResultsDir)) { |  | ||||||
|         return [] |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     return fs.readdirSync(buildResultsDir).map(file => { |  | ||||||
|         // Every file in the .build-results dir should be a BuildResults JSON |  | ||||||
|         const filePath = path.join(buildResultsDir, file) |  | ||||||
|         const content = fs.readFileSync(filePath, 'utf8') |  | ||||||
|         return JSON.parse(content) as BuildResult |  | ||||||
|     }) |  | ||||||
| } |  | ||||||
| @@ -1,243 +0,0 @@ | |||||||
| import * as core from '@actions/core' |  | ||||||
| import * as exec from '@actions/exec' |  | ||||||
| import path from 'path' |  | ||||||
| import fs from 'fs' |  | ||||||
| import * as params from './input-params' |  | ||||||
| import {CacheListener} from './cache-reporting' |  | ||||||
| import {saveCache, restoreCache, cacheDebug, isCacheDebuggingEnabled, tryDelete, generateCacheKey} from './cache-utils' |  | ||||||
| import {GradleHomeEntryExtractor} from './cache-extract-entries' |  | ||||||
|  |  | ||||||
| const RESTORED_CACHE_KEY_KEY = 'restored-cache-key' |  | ||||||
|  |  | ||||||
| export const META_FILE_DIR = '.gradle-build-action' |  | ||||||
|  |  | ||||||
| export class GradleStateCache { |  | ||||||
|     private cacheName: string |  | ||||||
|     private cacheDescription: string |  | ||||||
|  |  | ||||||
|     protected readonly gradleUserHome: string |  | ||||||
|  |  | ||||||
|     constructor(gradleUserHome: string) { |  | ||||||
|         this.gradleUserHome = gradleUserHome |  | ||||||
|         this.cacheName = 'gradle' |  | ||||||
|         this.cacheDescription = 'Gradle User Home' |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     init(): void { |  | ||||||
|         const actionCacheDir = path.resolve(this.gradleUserHome, '.gradle-build-action') |  | ||||||
|         fs.mkdirSync(actionCacheDir, {recursive: true}) |  | ||||||
|  |  | ||||||
|         const initScriptsDir = path.resolve(this.gradleUserHome, 'init.d') |  | ||||||
|         fs.mkdirSync(initScriptsDir, {recursive: true}) |  | ||||||
|  |  | ||||||
|         this.initializeGradleUserHome(this.gradleUserHome, initScriptsDir) |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     cacheOutputExists(): boolean { |  | ||||||
|         const cachesDir = path.resolve(this.gradleUserHome, 'caches') |  | ||||||
|         if (fs.existsSync(cachesDir)) { |  | ||||||
|             cacheDebug(`Cache output exists at ${cachesDir}`) |  | ||||||
|             return true |  | ||||||
|         } |  | ||||||
|         return false |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Restores the cache entry, finding the closest match to the currently running job. |  | ||||||
|      */ |  | ||||||
|     async restore(listener: CacheListener): Promise<void> { |  | ||||||
|         const entryListener = listener.entry(this.cacheDescription) |  | ||||||
|  |  | ||||||
|         const cacheKey = generateCacheKey(this.cacheName) |  | ||||||
|  |  | ||||||
|         cacheDebug( |  | ||||||
|             `Requesting ${this.cacheDescription} with |  | ||||||
|     key:${cacheKey.key} |  | ||||||
|     restoreKeys:[${cacheKey.restoreKeys}]` |  | ||||||
|         ) |  | ||||||
|  |  | ||||||
|         const cacheResult = await restoreCache(this.getCachePath(), cacheKey.key, cacheKey.restoreKeys, entryListener) |  | ||||||
|         if (!cacheResult) { |  | ||||||
|             core.info(`${this.cacheDescription} cache not found. Will initialize empty.`) |  | ||||||
|             return |  | ||||||
|         } |  | ||||||
|  |  | ||||||
|         core.saveState(RESTORED_CACHE_KEY_KEY, cacheResult.key) |  | ||||||
|  |  | ||||||
|         core.info(`Restored ${this.cacheDescription} from cache key: ${cacheResult.key}`) |  | ||||||
|  |  | ||||||
|         try { |  | ||||||
|             await this.afterRestore(listener) |  | ||||||
|         } catch (error) { |  | ||||||
|             core.warning(`Restore ${this.cacheDescription} failed in 'afterRestore': ${error}`) |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Restore any extracted cache entries after the main Gradle User Home entry is restored. |  | ||||||
|      */ |  | ||||||
|     async afterRestore(listener: CacheListener): Promise<void> { |  | ||||||
|         await this.debugReportGradleUserHomeSize('as restored from cache') |  | ||||||
|         await new GradleHomeEntryExtractor(this.gradleUserHome).restore(listener) |  | ||||||
|         // await new ConfigurationCacheEntryExtractor(this.gradleUserHome).restore(listener) |  | ||||||
|         await this.debugReportGradleUserHomeSize('after restoring common artifacts') |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Saves the cache entry based on the current cache key unless the cache was restored with the exact key, |  | ||||||
|      * in which case we cannot overwrite it. |  | ||||||
|      * |  | ||||||
|      * If the cache entry was restored with a partial match on a restore key, then |  | ||||||
|      * it is saved with the exact key. |  | ||||||
|      */ |  | ||||||
|     async save(listener: CacheListener): Promise<void> { |  | ||||||
|         const cacheKey = generateCacheKey(this.cacheName).key |  | ||||||
|         const restoredCacheKey = core.getState(RESTORED_CACHE_KEY_KEY) |  | ||||||
|         const gradleHomeEntryListener = listener.entry(this.cacheDescription) |  | ||||||
|  |  | ||||||
|         if (restoredCacheKey && cacheKey === restoredCacheKey) { |  | ||||||
|             core.info(`Cache hit occurred on the cache key ${cacheKey}, not saving cache.`) |  | ||||||
|  |  | ||||||
|             for (const entryListener of listener.cacheEntries) { |  | ||||||
|                 if (entryListener === gradleHomeEntryListener) { |  | ||||||
|                     entryListener.markNotSaved('cache key not changed') |  | ||||||
|                 } else { |  | ||||||
|                     entryListener.markNotSaved(`referencing '${this.cacheDescription}' cache entry not saved`) |  | ||||||
|                 } |  | ||||||
|             } |  | ||||||
|             return |  | ||||||
|         } |  | ||||||
|  |  | ||||||
|         try { |  | ||||||
|             await this.beforeSave(listener) |  | ||||||
|         } catch (error) { |  | ||||||
|             core.warning(`Save ${this.cacheDescription} failed in 'beforeSave': ${error}`) |  | ||||||
|             return |  | ||||||
|         } |  | ||||||
|  |  | ||||||
|         core.info(`Caching ${this.cacheDescription} with cache key: ${cacheKey}`) |  | ||||||
|         const cachePath = this.getCachePath() |  | ||||||
|         await saveCache(cachePath, cacheKey, gradleHomeEntryListener) |  | ||||||
|  |  | ||||||
|         return |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Extract and save any defined extracted cache entries prior to the main Gradle User Home entry being saved. |  | ||||||
|      */ |  | ||||||
|     async beforeSave(listener: CacheListener): Promise<void> { |  | ||||||
|         await this.debugReportGradleUserHomeSize('before saving common artifacts') |  | ||||||
|         this.deleteExcludedPaths() |  | ||||||
|         await Promise.all([ |  | ||||||
|             new GradleHomeEntryExtractor(this.gradleUserHome).extract(listener) |  | ||||||
|             // new ConfigurationCacheEntryExtractor(this.gradleUserHome).extract(listener) |  | ||||||
|         ]) |  | ||||||
|         await this.debugReportGradleUserHomeSize( |  | ||||||
|             "after extracting common artifacts (only 'caches' and 'notifications' will be stored)" |  | ||||||
|         ) |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Delete any file paths that are excluded by the `gradle-home-cache-excludes` parameter. |  | ||||||
|      */ |  | ||||||
|     private deleteExcludedPaths(): void { |  | ||||||
|         const rawPaths: string[] = params.getCacheExcludes() |  | ||||||
|         const resolvedPaths = rawPaths.map(x => path.resolve(this.gradleUserHome, x)) |  | ||||||
|  |  | ||||||
|         for (const p of resolvedPaths) { |  | ||||||
|             cacheDebug(`Deleting excluded path: ${p}`) |  | ||||||
|             tryDelete(p) |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Determines the paths within Gradle User Home to cache. |  | ||||||
|      * By default, this is the 'caches' and 'notifications' directories, |  | ||||||
|      * but this can be overridden by the `gradle-home-cache-includes` parameter. |  | ||||||
|      */ |  | ||||||
|     protected getCachePath(): string[] { |  | ||||||
|         const rawPaths: string[] = params.getCacheIncludes() |  | ||||||
|         rawPaths.push(META_FILE_DIR) |  | ||||||
|         const resolvedPaths = rawPaths.map(x => this.resolveCachePath(x)) |  | ||||||
|         cacheDebug(`Using cache paths: ${resolvedPaths}`) |  | ||||||
|         return resolvedPaths |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     private resolveCachePath(rawPath: string): string { |  | ||||||
|         if (rawPath.startsWith('!')) { |  | ||||||
|             const resolved = this.resolveCachePath(rawPath.substring(1)) |  | ||||||
|             return `!${resolved}` |  | ||||||
|         } |  | ||||||
|         return path.resolve(this.gradleUserHome, rawPath) |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     private initializeGradleUserHome(gradleUserHome: string, initScriptsDir: string): void { |  | ||||||
|         // Ensure that pre-installed java versions are detected. Only add property if it isn't already defined. |  | ||||||
|         const gradleProperties = path.resolve(gradleUserHome, 'gradle.properties') |  | ||||||
|         const existingGradleProperties = fs.existsSync(gradleProperties) |  | ||||||
|             ? fs.readFileSync(gradleProperties, 'utf8') |  | ||||||
|             : '' |  | ||||||
|         if (!existingGradleProperties.includes('org.gradle.java.installations.fromEnv=')) { |  | ||||||
|             fs.appendFileSync( |  | ||||||
|                 gradleProperties, |  | ||||||
|                 ` |  | ||||||
| # Auto-detect pre-installed JDKs |  | ||||||
| org.gradle.java.installations.fromEnv=JAVA_HOME_8_X64,JAVA_HOME_11_X64,JAVA_HOME_17_X64 |  | ||||||
| ` |  | ||||||
|             ) |  | ||||||
|         } |  | ||||||
|  |  | ||||||
|         // Copy init scripts from src/resources |  | ||||||
|         const initScriptFilenames = [ |  | ||||||
|             'gradle-build-action.build-result-capture.init.gradle', |  | ||||||
|             'gradle-build-action.build-result-capture-service.plugin.groovy', |  | ||||||
|             'gradle-build-action.github-dependency-graph.init.gradle', |  | ||||||
|             'gradle-build-action.github-dependency-graph-gradle-plugin-apply.groovy', |  | ||||||
|             'gradle-build-action.inject-gradle-enterprise.init.gradle' |  | ||||||
|         ] |  | ||||||
|         for (const initScriptFilename of initScriptFilenames) { |  | ||||||
|             const initScriptContent = this.readInitScriptAsString(initScriptFilename) |  | ||||||
|             const initScriptPath = path.resolve(initScriptsDir, initScriptFilename) |  | ||||||
|             fs.writeFileSync(initScriptPath, initScriptContent) |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     private readInitScriptAsString(resource: string): string { |  | ||||||
|         // Resolving relative to __dirname will allow node to find the resource at runtime |  | ||||||
|         const absolutePath = path.resolve(__dirname, '..', '..', 'src', 'resources', 'init-scripts', resource) |  | ||||||
|         return fs.readFileSync(absolutePath, 'utf8') |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * When cache debugging is enabled, this method will give a detailed report |  | ||||||
|      * of the Gradle User Home contents. |  | ||||||
|      */ |  | ||||||
|     private async debugReportGradleUserHomeSize(label: string): Promise<void> { |  | ||||||
|         if (!isCacheDebuggingEnabled()) { |  | ||||||
|             return |  | ||||||
|         } |  | ||||||
|         if (!fs.existsSync(this.gradleUserHome)) { |  | ||||||
|             return |  | ||||||
|         } |  | ||||||
|         const result = await exec.getExecOutput('du', ['-h', '-c', '-t', '5M'], { |  | ||||||
|             cwd: this.gradleUserHome, |  | ||||||
|             silent: true, |  | ||||||
|             ignoreReturnCode: true |  | ||||||
|         }) |  | ||||||
|  |  | ||||||
|         core.info(`Gradle User Home (directories >5M): ${label}`) |  | ||||||
|  |  | ||||||
|         core.info( |  | ||||||
|             result.stdout |  | ||||||
|                 .trimEnd() |  | ||||||
|                 .replace(/\t/g, '    ') |  | ||||||
|                 .split('\n') |  | ||||||
|                 .map(it => { |  | ||||||
|                     return `  ${it}` |  | ||||||
|                 }) |  | ||||||
|                 .join('\n') |  | ||||||
|         ) |  | ||||||
|  |  | ||||||
|         core.info('-----------------------') |  | ||||||
|     } |  | ||||||
| } |  | ||||||
| @@ -1,69 +0,0 @@ | |||||||
| import * as core from '@actions/core' |  | ||||||
| import * as exec from '@actions/exec' |  | ||||||
| import * as glob from '@actions/glob' |  | ||||||
| import fs from 'fs' |  | ||||||
| import path from 'path' |  | ||||||
|  |  | ||||||
| export class CacheCleaner { |  | ||||||
|     private readonly gradleUserHome: string |  | ||||||
|     private readonly tmpDir: string |  | ||||||
|  |  | ||||||
|     constructor(gradleUserHome: string, tmpDir: string) { |  | ||||||
|         this.gradleUserHome = gradleUserHome |  | ||||||
|         this.tmpDir = tmpDir |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     async prepare(): Promise<void> { |  | ||||||
|         // Reset the file-access journal so that files appear not to have been used recently |  | ||||||
|         fs.rmSync(path.resolve(this.gradleUserHome, 'caches/journal-1'), {recursive: true, force: true}) |  | ||||||
|         fs.mkdirSync(path.resolve(this.gradleUserHome, 'caches/journal-1'), {recursive: true}) |  | ||||||
|         fs.writeFileSync( |  | ||||||
|             path.resolve(this.gradleUserHome, 'caches/journal-1/file-access.properties'), |  | ||||||
|             'inceptionTimestamp=0' |  | ||||||
|         ) |  | ||||||
|  |  | ||||||
|         // Set the modification time of all files to the past: this timestamp is used when there is no matching entry in the journal |  | ||||||
|         await this.ageAllFiles() |  | ||||||
|  |  | ||||||
|         // Touch all 'gc' files so that cache cleanup won't run immediately. |  | ||||||
|         await this.touchAllFiles('gc.properties') |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     async forceCleanup(): Promise<void> { |  | ||||||
|         // Age all 'gc' files so that cache cleanup will run immediately. |  | ||||||
|         await this.ageAllFiles('gc.properties') |  | ||||||
|  |  | ||||||
|         // Run a dummy Gradle build to trigger cache cleanup |  | ||||||
|         const cleanupProjectDir = path.resolve(this.tmpDir, 'dummy-cleanup-project') |  | ||||||
|         fs.mkdirSync(cleanupProjectDir, {recursive: true}) |  | ||||||
|         fs.writeFileSync( |  | ||||||
|             path.resolve(cleanupProjectDir, 'settings.gradle'), |  | ||||||
|             'rootProject.name = "dummy-cleanup-project"' |  | ||||||
|         ) |  | ||||||
|         fs.writeFileSync(path.resolve(cleanupProjectDir, 'build.gradle'), 'task("noop") {}') |  | ||||||
|  |  | ||||||
|         const gradleCommand = `gradle -g ${this.gradleUserHome} --no-daemon --build-cache --no-scan --quiet -DGITHUB_DEPENDENCY_GRAPH_ENABLED=false noop` |  | ||||||
|         await exec.exec(gradleCommand, [], { |  | ||||||
|             cwd: cleanupProjectDir |  | ||||||
|         }) |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     private async ageAllFiles(fileName = '*'): Promise<void> { |  | ||||||
|         core.debug(`Aging all files in Gradle User Home with name ${fileName}`) |  | ||||||
|         await this.setUtimes(`${this.gradleUserHome}/**/${fileName}`, new Date(0)) |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     private async touchAllFiles(fileName = '*'): Promise<void> { |  | ||||||
|         core.debug(`Touching all files in Gradle User Home with name ${fileName}`) |  | ||||||
|         await this.setUtimes(`${this.gradleUserHome}/**/${fileName}`, new Date()) |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     private async setUtimes(pattern: string, timestamp: Date): Promise<void> { |  | ||||||
|         const globber = await glob.create(pattern, { |  | ||||||
|             implicitDescendants: false |  | ||||||
|         }) |  | ||||||
|         for await (const file of globber.globGenerator()) { |  | ||||||
|             fs.utimesSync(file, timestamp, timestamp) |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| } |  | ||||||
| @@ -1,387 +0,0 @@ | |||||||
| import path from 'path' |  | ||||||
| import fs from 'fs' |  | ||||||
| import * as core from '@actions/core' |  | ||||||
| import * as glob from '@actions/glob' |  | ||||||
|  |  | ||||||
| import * as params from './input-params' |  | ||||||
|  |  | ||||||
| import {META_FILE_DIR} from './cache-base' |  | ||||||
| import {CacheEntryListener, CacheListener} from './cache-reporting' |  | ||||||
| import {cacheDebug, getCacheKeyPrefix, hashFileNames, restoreCache, saveCache, tryDelete} from './cache-utils' |  | ||||||
| import {loadBuildResults} from './build-results' |  | ||||||
|  |  | ||||||
| const SKIP_RESTORE_VAR = 'GRADLE_BUILD_ACTION_SKIP_RESTORE' |  | ||||||
|  |  | ||||||
| /** |  | ||||||
|  * Represents the result of attempting to load or store an extracted cache entry. |  | ||||||
|  * An undefined cacheKey indicates that the operation did not succeed. |  | ||||||
|  * The collected results are then used to populate the `cache-metadata.json` file for later use. |  | ||||||
|  */ |  | ||||||
| class ExtractedCacheEntry { |  | ||||||
|     artifactType: string |  | ||||||
|     pattern: string |  | ||||||
|     cacheKey: string | undefined |  | ||||||
|  |  | ||||||
|     constructor(artifactType: string, pattern: string, cacheKey: string | undefined) { |  | ||||||
|         this.artifactType = artifactType |  | ||||||
|         this.pattern = pattern |  | ||||||
|         this.cacheKey = cacheKey |  | ||||||
|     } |  | ||||||
| } |  | ||||||
|  |  | ||||||
| /** |  | ||||||
|  * Representation of all of the extracted cache entries for this Gradle User Home. |  | ||||||
|  * This object is persisted to JSON file in the Gradle User Home directory for storing, |  | ||||||
|  * and subsequently used to restore the Gradle User Home. |  | ||||||
|  */ |  | ||||||
| class ExtractedCacheEntryMetadata { |  | ||||||
|     entries: ExtractedCacheEntry[] = [] |  | ||||||
| } |  | ||||||
|  |  | ||||||
| /** |  | ||||||
|  * The specification for a type of extracted cache entry. |  | ||||||
|  */ |  | ||||||
| class ExtractedCacheEntryDefinition { |  | ||||||
|     artifactType: string |  | ||||||
|     pattern: string |  | ||||||
|     bundle: boolean |  | ||||||
|     uniqueFileNames = true |  | ||||||
|  |  | ||||||
|     constructor(artifactType: string, pattern: string, bundle: boolean) { |  | ||||||
|         this.artifactType = artifactType |  | ||||||
|         this.pattern = pattern |  | ||||||
|         this.bundle = bundle |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     withNonUniqueFileNames(): ExtractedCacheEntryDefinition { |  | ||||||
|         this.uniqueFileNames = false |  | ||||||
|         return this |  | ||||||
|     } |  | ||||||
| } |  | ||||||
|  |  | ||||||
| /** |  | ||||||
|  * Caches and restores the entire Gradle User Home directory, extracting entries containing common artifacts |  | ||||||
|  * for more efficient storage. |  | ||||||
|  */ |  | ||||||
| abstract class AbstractEntryExtractor { |  | ||||||
|     protected readonly gradleUserHome: string |  | ||||||
|     private extractorName: string |  | ||||||
|  |  | ||||||
|     constructor(gradleUserHome: string, extractorName: string) { |  | ||||||
|         this.gradleUserHome = gradleUserHome |  | ||||||
|         this.extractorName = extractorName |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Restores any artifacts that were cached separately, based on the information in the `cache-metadata.json` file. |  | ||||||
|      * Each extracted cache entry is restored in parallel, except when debugging is enabled. |  | ||||||
|      */ |  | ||||||
|     async restore(listener: CacheListener): Promise<void> { |  | ||||||
|         const previouslyExtractedCacheEntries = this.loadExtractedCacheEntries() |  | ||||||
|  |  | ||||||
|         const processes: Promise<ExtractedCacheEntry>[] = [] |  | ||||||
|  |  | ||||||
|         for (const cacheEntry of previouslyExtractedCacheEntries) { |  | ||||||
|             const artifactType = cacheEntry.artifactType |  | ||||||
|             const entryListener = listener.entry(cacheEntry.pattern) |  | ||||||
|  |  | ||||||
|             // Handle case where the extracted-cache-entry definitions have been changed |  | ||||||
|             const skipRestore = process.env[SKIP_RESTORE_VAR] || '' |  | ||||||
|             if (skipRestore.includes(artifactType)) { |  | ||||||
|                 core.info(`Not restoring extracted cache entry for ${artifactType}`) |  | ||||||
|                 entryListener.markRequested('SKIP_RESTORE') |  | ||||||
|             } else { |  | ||||||
|                 processes.push( |  | ||||||
|                     this.awaitForDebugging( |  | ||||||
|                         this.restoreExtractedCacheEntry( |  | ||||||
|                             artifactType, |  | ||||||
|                             cacheEntry.cacheKey!, |  | ||||||
|                             cacheEntry.pattern, |  | ||||||
|                             entryListener |  | ||||||
|                         ) |  | ||||||
|                     ) |  | ||||||
|                 ) |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
|  |  | ||||||
|         this.saveMetadataForCacheResults(await Promise.all(processes)) |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     private async restoreExtractedCacheEntry( |  | ||||||
|         artifactType: string, |  | ||||||
|         cacheKey: string, |  | ||||||
|         pattern: string, |  | ||||||
|         listener: CacheEntryListener |  | ||||||
|     ): Promise<ExtractedCacheEntry> { |  | ||||||
|         const restoredEntry = await restoreCache([pattern], cacheKey, [], listener) |  | ||||||
|         if (restoredEntry) { |  | ||||||
|             core.info(`Restored ${artifactType} with key ${cacheKey} to ${pattern}`) |  | ||||||
|             return new ExtractedCacheEntry(artifactType, pattern, cacheKey) |  | ||||||
|         } else { |  | ||||||
|             core.info(`Did not restore ${artifactType} with key ${cacheKey} to ${pattern}`) |  | ||||||
|             return new ExtractedCacheEntry(artifactType, pattern, undefined) |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Saves any artifacts that are configured to be cached separately, based on the extracted cache entry definitions. |  | ||||||
|      * Each entry is extracted and saved in parallel, except when debugging is enabled. |  | ||||||
|      */ |  | ||||||
|     async extract(listener: CacheListener): Promise<void> { |  | ||||||
|         // Load the cache entry definitions (from config) and the previously restored entries (from persisted metadata file) |  | ||||||
|         const cacheEntryDefinitions = this.getExtractedCacheEntryDefinitions() |  | ||||||
|         cacheDebug( |  | ||||||
|             `Extracting cache entries for ${this.extractorName}: ${JSON.stringify(cacheEntryDefinitions, null, 2)}` |  | ||||||
|         ) |  | ||||||
|  |  | ||||||
|         const previouslyRestoredEntries = this.loadExtractedCacheEntries() |  | ||||||
|         const cacheActions: Promise<ExtractedCacheEntry>[] = [] |  | ||||||
|  |  | ||||||
|         // For each cache entry definition, determine if it has already been restored, and if not, extract it |  | ||||||
|         for (const cacheEntryDefinition of cacheEntryDefinitions) { |  | ||||||
|             const artifactType = cacheEntryDefinition.artifactType |  | ||||||
|             const pattern = cacheEntryDefinition.pattern |  | ||||||
|  |  | ||||||
|             // Find all matching files for this cache entry definition |  | ||||||
|             const globber = await glob.create(pattern, { |  | ||||||
|                 implicitDescendants: false |  | ||||||
|             }) |  | ||||||
|             const matchingFiles = await globber.glob() |  | ||||||
|  |  | ||||||
|             if (matchingFiles.length === 0) { |  | ||||||
|                 cacheDebug(`No files found to cache for ${artifactType}`) |  | ||||||
|                 continue |  | ||||||
|             } |  | ||||||
|  |  | ||||||
|             if (cacheEntryDefinition.bundle) { |  | ||||||
|                 // For an extracted "bundle", use the defined pattern and cache all matching files in a single entry. |  | ||||||
|                 cacheActions.push( |  | ||||||
|                     this.awaitForDebugging( |  | ||||||
|                         this.saveExtractedCacheEntry( |  | ||||||
|                             matchingFiles, |  | ||||||
|                             artifactType, |  | ||||||
|                             pattern, |  | ||||||
|                             cacheEntryDefinition.uniqueFileNames, |  | ||||||
|                             previouslyRestoredEntries, |  | ||||||
|                             listener.entry(pattern) |  | ||||||
|                         ) |  | ||||||
|                     ) |  | ||||||
|                 ) |  | ||||||
|             } else { |  | ||||||
|                 // Otherwise cache each matching file in a separate entry, using the complete file path as the cache pattern. |  | ||||||
|                 for (const cacheFile of matchingFiles) { |  | ||||||
|                     cacheActions.push( |  | ||||||
|                         this.awaitForDebugging( |  | ||||||
|                             this.saveExtractedCacheEntry( |  | ||||||
|                                 [cacheFile], |  | ||||||
|                                 artifactType, |  | ||||||
|                                 cacheFile, |  | ||||||
|                                 cacheEntryDefinition.uniqueFileNames, |  | ||||||
|                                 previouslyRestoredEntries, |  | ||||||
|                                 listener.entry(cacheFile) |  | ||||||
|                             ) |  | ||||||
|                         ) |  | ||||||
|                     ) |  | ||||||
|                 } |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
|  |  | ||||||
|         this.saveMetadataForCacheResults(await Promise.all(cacheActions)) |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     private async saveExtractedCacheEntry( |  | ||||||
|         matchingFiles: string[], |  | ||||||
|         artifactType: string, |  | ||||||
|         pattern: string, |  | ||||||
|         uniqueFileNames: boolean, |  | ||||||
|         previouslyRestoredEntries: ExtractedCacheEntry[], |  | ||||||
|         entryListener: CacheEntryListener |  | ||||||
|     ): Promise<ExtractedCacheEntry> { |  | ||||||
|         const cacheKey = uniqueFileNames |  | ||||||
|             ? this.createCacheKeyFromFileNames(artifactType, matchingFiles) |  | ||||||
|             : await this.createCacheKeyFromFileContents(artifactType, pattern) |  | ||||||
|         const previouslyRestoredKey = previouslyRestoredEntries.find( |  | ||||||
|             x => x.artifactType === artifactType && x.pattern === pattern |  | ||||||
|         )?.cacheKey |  | ||||||
|  |  | ||||||
|         if (previouslyRestoredKey === cacheKey) { |  | ||||||
|             cacheDebug(`No change to previously restored ${artifactType}. Not saving.`) |  | ||||||
|             entryListener.markNotSaved('contents unchanged') |  | ||||||
|         } else { |  | ||||||
|             core.info(`Caching ${artifactType} with path '${pattern}' and cache key: ${cacheKey}`) |  | ||||||
|             await saveCache([pattern], cacheKey, entryListener) |  | ||||||
|         } |  | ||||||
|  |  | ||||||
|         for (const file of matchingFiles) { |  | ||||||
|             tryDelete(file) |  | ||||||
|         } |  | ||||||
|  |  | ||||||
|         return new ExtractedCacheEntry(artifactType, pattern, cacheKey) |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     protected createCacheKeyFromFileNames(artifactType: string, files: string[]): string { |  | ||||||
|         const cacheKeyPrefix = getCacheKeyPrefix() |  | ||||||
|         const relativeFiles = files.map(x => path.relative(this.gradleUserHome, x)) |  | ||||||
|         const key = hashFileNames(relativeFiles) |  | ||||||
|  |  | ||||||
|         cacheDebug(`Generating cache key for ${artifactType} from file names: ${relativeFiles}`) |  | ||||||
|  |  | ||||||
|         return `${cacheKeyPrefix}${artifactType}-${key}` |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     protected async createCacheKeyFromFileContents(artifactType: string, pattern: string): Promise<string> { |  | ||||||
|         const cacheKeyPrefix = getCacheKeyPrefix() |  | ||||||
|         const key = await glob.hashFiles(pattern) |  | ||||||
|  |  | ||||||
|         cacheDebug(`Generating cache key for ${artifactType} from files matching: ${pattern}`) |  | ||||||
|  |  | ||||||
|         return `${cacheKeyPrefix}${artifactType}-${key}` |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     // Run actions sequentially if debugging is enabled |  | ||||||
|     private async awaitForDebugging(p: Promise<ExtractedCacheEntry>): Promise<ExtractedCacheEntry> { |  | ||||||
|         if (params.isCacheDebuggingEnabled()) { |  | ||||||
|             await p |  | ||||||
|         } |  | ||||||
|         return p |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Load information about the extracted cache entries previously restored/saved. This is loaded from the 'cache-metadata.json' file. |  | ||||||
|      */ |  | ||||||
|     protected loadExtractedCacheEntries(): ExtractedCacheEntry[] { |  | ||||||
|         const cacheMetadataFile = this.getCacheMetadataFile() |  | ||||||
|         if (!fs.existsSync(cacheMetadataFile)) { |  | ||||||
|             return [] |  | ||||||
|         } |  | ||||||
|  |  | ||||||
|         const filedata = fs.readFileSync(cacheMetadataFile, 'utf-8') |  | ||||||
|         cacheDebug(`Loaded cache metadata: ${filedata}`) |  | ||||||
|         const extractedCacheEntryMetadata = JSON.parse(filedata) as ExtractedCacheEntryMetadata |  | ||||||
|         return extractedCacheEntryMetadata.entries |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Saves information about the extracted cache entries into the 'cache-metadata.json' file. |  | ||||||
|      */ |  | ||||||
|     private saveMetadataForCacheResults(results: ExtractedCacheEntry[]): void { |  | ||||||
|         const extractedCacheEntryMetadata = new ExtractedCacheEntryMetadata() |  | ||||||
|         extractedCacheEntryMetadata.entries = results.filter(x => x.cacheKey !== undefined) |  | ||||||
|  |  | ||||||
|         const filedata = JSON.stringify(extractedCacheEntryMetadata) |  | ||||||
|         cacheDebug(`Saving cache metadata: ${filedata}`) |  | ||||||
|  |  | ||||||
|         fs.writeFileSync(this.getCacheMetadataFile(), filedata, 'utf-8') |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     private getCacheMetadataFile(): string { |  | ||||||
|         const actionMetadataDirectory = path.resolve(this.gradleUserHome, META_FILE_DIR) |  | ||||||
|         fs.mkdirSync(actionMetadataDirectory, {recursive: true}) |  | ||||||
|  |  | ||||||
|         return path.resolve(actionMetadataDirectory, `${this.extractorName}-entry-metadata.json`) |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     protected abstract getExtractedCacheEntryDefinitions(): ExtractedCacheEntryDefinition[] |  | ||||||
| } |  | ||||||
|  |  | ||||||
| export class GradleHomeEntryExtractor extends AbstractEntryExtractor { |  | ||||||
|     constructor(gradleUserHome: string) { |  | ||||||
|         super(gradleUserHome, 'gradle-home') |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     async extract(listener: CacheListener): Promise<void> { |  | ||||||
|         await this.deleteWrapperZips() |  | ||||||
|         return super.extract(listener) |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Delete any downloaded wrapper zip files that are not needed after extraction. |  | ||||||
|      * These files are cleaned up by Gradle >= 7.5, but for older versions we remove them manually. |  | ||||||
|      */ |  | ||||||
|     private async deleteWrapperZips(): Promise<void> { |  | ||||||
|         const wrapperZips = path.resolve(this.gradleUserHome, 'wrapper/dists/*/*/*.zip') |  | ||||||
|         const globber = await glob.create(wrapperZips, { |  | ||||||
|             implicitDescendants: false |  | ||||||
|         }) |  | ||||||
|  |  | ||||||
|         for (const wrapperZip of await globber.glob()) { |  | ||||||
|             cacheDebug(`Deleting wrapper zip: ${wrapperZip}`) |  | ||||||
|             await tryDelete(wrapperZip) |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Return the extracted cache entry definitions, which determine which artifacts will be cached |  | ||||||
|      * separately from the rest of the Gradle User Home cache entry. |  | ||||||
|      */ |  | ||||||
|     protected getExtractedCacheEntryDefinitions(): ExtractedCacheEntryDefinition[] { |  | ||||||
|         const entryDefinition = ( |  | ||||||
|             artifactType: string, |  | ||||||
|             patterns: string[], |  | ||||||
|             bundle: boolean |  | ||||||
|         ): ExtractedCacheEntryDefinition => { |  | ||||||
|             const resolvedPatterns = patterns |  | ||||||
|                 .map(x => { |  | ||||||
|                     const isDir = x.endsWith('/') |  | ||||||
|                     const resolved = path.resolve(this.gradleUserHome, x) |  | ||||||
|                     return isDir ? `${resolved}/` : resolved // Restore trailing '/' removed by path.resolve() |  | ||||||
|                 }) |  | ||||||
|                 .join('\n') |  | ||||||
|             return new ExtractedCacheEntryDefinition(artifactType, resolvedPatterns, bundle) |  | ||||||
|         } |  | ||||||
|  |  | ||||||
|         return [ |  | ||||||
|             entryDefinition('generated-gradle-jars', ['caches/*/generated-gradle-jars/*.jar'], false), |  | ||||||
|             entryDefinition('wrapper-zips', ['wrapper/dists/*/*/'], false), // Each wrapper directory cached separately |  | ||||||
|             entryDefinition('java-toolchains', ['jdks/*/'], false), // Each extracted JDK cached separately |  | ||||||
|             entryDefinition('dependencies', ['caches/modules-*/files-*/*/*/*/*'], true), |  | ||||||
|             entryDefinition('instrumented-jars', ['caches/jars-*/*'], true), |  | ||||||
|             entryDefinition('kotlin-dsl', ['caches/*/kotlin-dsl/*/*'], true) |  | ||||||
|         ] |  | ||||||
|     } |  | ||||||
| } |  | ||||||
|  |  | ||||||
| export class ConfigurationCacheEntryExtractor extends AbstractEntryExtractor { |  | ||||||
|     constructor(gradleUserHome: string) { |  | ||||||
|         super(gradleUserHome, 'configuration-cache') |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Handle the case where Gradle User Home has not been fully restored, so that the configuration-cache |  | ||||||
|      * entry is not reusable. |  | ||||||
|      */ |  | ||||||
|     async restore(listener: CacheListener): Promise<void> { |  | ||||||
|         if (listener.fullyRestored) { |  | ||||||
|             return super.restore(listener) |  | ||||||
|         } |  | ||||||
|  |  | ||||||
|         core.info('Not restoring configuration-cache state, as Gradle User Home was not fully restored') |  | ||||||
|         for (const cacheEntry of this.loadExtractedCacheEntries()) { |  | ||||||
|             listener.entry(cacheEntry.pattern).markRequested('NOT_RESTORED') |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Extract cache entries for the configuration cache in each project. |  | ||||||
|      */ |  | ||||||
|     protected getExtractedCacheEntryDefinitions(): ExtractedCacheEntryDefinition[] { |  | ||||||
|         return this.getProjectRoots().map(projectRoot => { |  | ||||||
|             const configCachePath = path.resolve(projectRoot, '.gradle/configuration-cache') |  | ||||||
|             return new ExtractedCacheEntryDefinition( |  | ||||||
|                 'configuration-cache', |  | ||||||
|                 configCachePath, |  | ||||||
|                 true |  | ||||||
|             ).withNonUniqueFileNames() |  | ||||||
|         }) |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * For every Gradle invocation, we record the project root directory. This method returns the entire |  | ||||||
|      * set of project roots, to allow saving of configuration-cache entries for each. |  | ||||||
|      */ |  | ||||||
|     private getProjectRoots(): string[] { |  | ||||||
|         const buildResults = loadBuildResults() |  | ||||||
|         const projectRootDirs = buildResults.map(x => x.rootProjectDir) |  | ||||||
|         return [...new Set(projectRootDirs)] // Remove duplicates |  | ||||||
|     } |  | ||||||
| } |  | ||||||
| @@ -1,229 +0,0 @@ | |||||||
| import * as core from '@actions/core' |  | ||||||
| import * as cache from '@actions/cache' |  | ||||||
|  |  | ||||||
| /** |  | ||||||
|  * Collects information on what entries were saved and restored during the action. |  | ||||||
|  * This information is used to generate a summary of the cache usage. |  | ||||||
|  */ |  | ||||||
| export class CacheListener { |  | ||||||
|     cacheEntries: CacheEntryListener[] = [] |  | ||||||
|     cacheReadOnly = false |  | ||||||
|     cacheWriteOnly = false |  | ||||||
|     cacheDisabled = false |  | ||||||
|     cacheDisabledReason = 'disabled' |  | ||||||
|  |  | ||||||
|     get fullyRestored(): boolean { |  | ||||||
|         return this.cacheEntries.every(x => !x.wasRequestedButNotRestored()) |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     get cacheStatus(): string { |  | ||||||
|         if (!cache.isFeatureAvailable()) return 'not available' |  | ||||||
|         if (this.cacheDisabled) return this.cacheDisabledReason |  | ||||||
|         if (this.cacheWriteOnly) return 'write-only' |  | ||||||
|         if (this.cacheReadOnly) return 'read-only' |  | ||||||
|         return 'enabled' |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     entry(name: string): CacheEntryListener { |  | ||||||
|         for (const entry of this.cacheEntries) { |  | ||||||
|             if (entry.entryName === name) { |  | ||||||
|                 return entry |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
|  |  | ||||||
|         const newEntry = new CacheEntryListener(name) |  | ||||||
|         this.cacheEntries.push(newEntry) |  | ||||||
|         return newEntry |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     stringify(): string { |  | ||||||
|         return JSON.stringify(this) |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     static rehydrate(stringRep: string): CacheListener { |  | ||||||
|         if (stringRep === '') { |  | ||||||
|             return new CacheListener() |  | ||||||
|         } |  | ||||||
|         const rehydrated: CacheListener = Object.assign(new CacheListener(), JSON.parse(stringRep)) |  | ||||||
|         const entries = rehydrated.cacheEntries |  | ||||||
|         for (let index = 0; index < entries.length; index++) { |  | ||||||
|             const rawEntry = entries[index] |  | ||||||
|             entries[index] = Object.assign(new CacheEntryListener(rawEntry.entryName), rawEntry) |  | ||||||
|         } |  | ||||||
|         return rehydrated |  | ||||||
|     } |  | ||||||
| } |  | ||||||
|  |  | ||||||
| /** |  | ||||||
|  * Collects information on the state of a single cache entry. |  | ||||||
|  */ |  | ||||||
| export class CacheEntryListener { |  | ||||||
|     entryName: string |  | ||||||
|     requestedKey: string | undefined |  | ||||||
|     requestedRestoreKeys: string[] | undefined |  | ||||||
|     restoredKey: string | undefined |  | ||||||
|     restoredSize: number | undefined |  | ||||||
|     notRestored: string | undefined |  | ||||||
|  |  | ||||||
|     savedKey: string | undefined |  | ||||||
|     savedSize: number | undefined |  | ||||||
|     notSaved: string | undefined |  | ||||||
|  |  | ||||||
|     constructor(entryName: string) { |  | ||||||
|         this.entryName = entryName |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     wasRequestedButNotRestored(): boolean { |  | ||||||
|         return this.requestedKey !== undefined && this.restoredKey === undefined |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     markRequested(key: string, restoreKeys: string[] = []): CacheEntryListener { |  | ||||||
|         this.requestedKey = key |  | ||||||
|         this.requestedRestoreKeys = restoreKeys |  | ||||||
|         return this |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     markRestored(key: string, size: number | undefined): CacheEntryListener { |  | ||||||
|         this.restoredKey = key |  | ||||||
|         this.restoredSize = size |  | ||||||
|         return this |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     markNotRestored(message: string): CacheEntryListener { |  | ||||||
|         this.notRestored = message |  | ||||||
|         return this |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     markSaved(key: string, size: number | undefined): CacheEntryListener { |  | ||||||
|         this.savedKey = key |  | ||||||
|         this.savedSize = size |  | ||||||
|         return this |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     markAlreadyExists(key: string): CacheEntryListener { |  | ||||||
|         this.savedKey = key |  | ||||||
|         this.savedSize = 0 |  | ||||||
|         return this |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     markNotSaved(message: string): CacheEntryListener { |  | ||||||
|         this.notSaved = message |  | ||||||
|         return this |  | ||||||
|     } |  | ||||||
| } |  | ||||||
|  |  | ||||||
| export function writeCachingReport(listener: CacheListener): void { |  | ||||||
|     const entries = listener.cacheEntries |  | ||||||
|  |  | ||||||
|     core.summary.addRaw( |  | ||||||
|         `\n<details><summary><h4>Caching for gradle-build-action was ${listener.cacheStatus} - expand for details</h4></summary>\n` |  | ||||||
|     ) |  | ||||||
|  |  | ||||||
|     core.summary.addTable([ |  | ||||||
|         [ |  | ||||||
|             {data: '', header: true}, |  | ||||||
|             {data: 'Count', header: true}, |  | ||||||
|             {data: 'Total Size (Mb)', header: true} |  | ||||||
|         ], |  | ||||||
|         ['Entries Restored', `${getCount(entries, e => e.restoredSize)}`, `${getSize(entries, e => e.restoredSize)}`], |  | ||||||
|         ['Entries Saved', `${getCount(entries, e => e.savedSize)}`, `${getSize(entries, e => e.savedSize)}`] |  | ||||||
|     ]) |  | ||||||
|  |  | ||||||
|     core.summary.addHeading('Cache Entry Details', 5) |  | ||||||
|  |  | ||||||
|     const entryDetails = renderEntryDetails(listener) |  | ||||||
|     core.summary.addRaw(`<pre> |  | ||||||
| ${entryDetails} |  | ||||||
| </pre> |  | ||||||
| </details> |  | ||||||
| `) |  | ||||||
| } |  | ||||||
|  |  | ||||||
| export function logCachingReport(listener: CacheListener): void { |  | ||||||
|     const entries = listener.cacheEntries |  | ||||||
|  |  | ||||||
|     core.startGroup(`Caching for gradle-build-action was ${listener.cacheStatus} - expand for details`) |  | ||||||
|  |  | ||||||
|     core.info( |  | ||||||
|         `Entries Restored: ${getCount(entries, e => e.restoredSize)} (${getSize(entries, e => e.restoredSize)} Mb)` |  | ||||||
|     ) |  | ||||||
|     core.info(`Entries Saved   : ${getCount(entries, e => e.savedSize)} (${getSize(entries, e => e.savedSize)} Mb)`) |  | ||||||
|  |  | ||||||
|     core.info(`Cache Entry Details`) |  | ||||||
|     core.info(renderEntryDetails(listener)) |  | ||||||
|  |  | ||||||
|     core.endGroup() |  | ||||||
| } |  | ||||||
|  |  | ||||||
| function renderEntryDetails(listener: CacheListener): string { |  | ||||||
|     return listener.cacheEntries |  | ||||||
|         .map( |  | ||||||
|             entry => `Entry: ${entry.entryName} |  | ||||||
|     Requested Key : ${entry.requestedKey ?? ''} |  | ||||||
|     Restored  Key : ${entry.restoredKey ?? ''} |  | ||||||
|               Size: ${formatSize(entry.restoredSize)} |  | ||||||
|               ${getRestoredMessage(entry, listener.cacheWriteOnly)} |  | ||||||
|     Saved     Key : ${entry.savedKey ?? ''} |  | ||||||
|               Size: ${formatSize(entry.savedSize)} |  | ||||||
|               ${getSavedMessage(entry, listener.cacheReadOnly)} |  | ||||||
| ` |  | ||||||
|         ) |  | ||||||
|         .join('---\n') |  | ||||||
| } |  | ||||||
|  |  | ||||||
| function getRestoredMessage(entry: CacheEntryListener, cacheWriteOnly: boolean): string { |  | ||||||
|     if (entry.notRestored) { |  | ||||||
|         return `(Entry not restored: ${entry.notRestored})` |  | ||||||
|     } |  | ||||||
|     if (cacheWriteOnly) { |  | ||||||
|         return '(Entry not restored: cache is write-only)' |  | ||||||
|     } |  | ||||||
|     if (entry.requestedKey === undefined) { |  | ||||||
|         return '(Entry not restored: not requested)' |  | ||||||
|     } |  | ||||||
|     if (entry.restoredKey === undefined) { |  | ||||||
|         return '(Entry not restored: no match found)' |  | ||||||
|     } |  | ||||||
|     if (entry.restoredKey === entry.requestedKey) { |  | ||||||
|         return '(Entry restored: exact match found)' |  | ||||||
|     } |  | ||||||
|     return '(Entry restored: partial match found)' |  | ||||||
| } |  | ||||||
|  |  | ||||||
| function getSavedMessage(entry: CacheEntryListener, cacheReadOnly: boolean): string { |  | ||||||
|     if (entry.notSaved) { |  | ||||||
|         return `(Entry not saved: ${entry.notSaved})` |  | ||||||
|     } |  | ||||||
|     if (entry.savedKey === undefined) { |  | ||||||
|         if (cacheReadOnly) { |  | ||||||
|             return '(Entry not saved: cache is read-only)' |  | ||||||
|         } |  | ||||||
|         return '(Entry not saved: reason unknown)' |  | ||||||
|     } |  | ||||||
|     if (entry.savedSize === 0) { |  | ||||||
|         return '(Entry not saved: entry with key already exists)' |  | ||||||
|     } |  | ||||||
|     return '(Entry saved)' |  | ||||||
| } |  | ||||||
|  |  | ||||||
| function getCount( |  | ||||||
|     cacheEntries: CacheEntryListener[], |  | ||||||
|     predicate: (value: CacheEntryListener) => number | undefined |  | ||||||
| ): number { |  | ||||||
|     return cacheEntries.filter(e => predicate(e)).length |  | ||||||
| } |  | ||||||
|  |  | ||||||
| function getSize( |  | ||||||
|     cacheEntries: CacheEntryListener[], |  | ||||||
|     predicate: (value: CacheEntryListener) => number | undefined |  | ||||||
| ): number { |  | ||||||
|     const bytes = cacheEntries.map(e => predicate(e) ?? 0).reduce((p, v) => p + v, 0) |  | ||||||
|     return Math.round(bytes / (1024 * 1024)) |  | ||||||
| } |  | ||||||
|  |  | ||||||
| function formatSize(bytes: number | undefined): string { |  | ||||||
|     if (bytes === undefined || bytes === 0) { |  | ||||||
|         return '' |  | ||||||
|     } |  | ||||||
|     return `${Math.round(bytes / (1024 * 1024))} MB (${bytes} B)` |  | ||||||
| } |  | ||||||
| @@ -1,269 +0,0 @@ | |||||||
| import * as core from '@actions/core' |  | ||||||
| import * as cache from '@actions/cache' |  | ||||||
| import * as github from '@actions/github' |  | ||||||
| import * as exec from '@actions/exec' |  | ||||||
|  |  | ||||||
| import * as crypto from 'crypto' |  | ||||||
| import * as path from 'path' |  | ||||||
| import * as fs from 'fs' |  | ||||||
|  |  | ||||||
| import * as params from './input-params' |  | ||||||
|  |  | ||||||
| import {CacheEntryListener} from './cache-reporting' |  | ||||||
|  |  | ||||||
| const CACHE_PROTOCOL_VERSION = 'v8-' |  | ||||||
|  |  | ||||||
| const CACHE_KEY_PREFIX_VAR = 'GRADLE_BUILD_ACTION_CACHE_KEY_PREFIX' |  | ||||||
| const CACHE_KEY_OS_VAR = 'GRADLE_BUILD_ACTION_CACHE_KEY_ENVIRONMENT' |  | ||||||
| const CACHE_KEY_JOB_VAR = 'GRADLE_BUILD_ACTION_CACHE_KEY_JOB' |  | ||||||
| const CACHE_KEY_JOB_INSTANCE_VAR = 'GRADLE_BUILD_ACTION_CACHE_KEY_JOB_INSTANCE' |  | ||||||
| const CACHE_KEY_JOB_EXECUTION_VAR = 'GRADLE_BUILD_ACTION_CACHE_KEY_JOB_EXECUTION' |  | ||||||
|  |  | ||||||
| const SEGMENT_DOWNLOAD_TIMEOUT_VAR = 'SEGMENT_DOWNLOAD_TIMEOUT_MINS' |  | ||||||
| const SEGMENT_DOWNLOAD_TIMEOUT_DEFAULT = 10 * 60 * 1000 // 10 minutes |  | ||||||
|  |  | ||||||
| export function isCacheDisabled(): boolean { |  | ||||||
|     if (!cache.isFeatureAvailable()) { |  | ||||||
|         return true |  | ||||||
|     } |  | ||||||
|     return params.isCacheDisabled() |  | ||||||
| } |  | ||||||
|  |  | ||||||
| export function isCacheReadOnly(): boolean { |  | ||||||
|     return !isCacheWriteOnly() && params.isCacheReadOnly() |  | ||||||
| } |  | ||||||
|  |  | ||||||
| export function isCacheWriteOnly(): boolean { |  | ||||||
|     return params.isCacheWriteOnly() |  | ||||||
| } |  | ||||||
|  |  | ||||||
| export function isCacheOverwriteExisting(): boolean { |  | ||||||
|     return params.isCacheOverwriteExisting() |  | ||||||
| } |  | ||||||
|  |  | ||||||
| export function isCacheDebuggingEnabled(): boolean { |  | ||||||
|     return params.isCacheDebuggingEnabled() |  | ||||||
| } |  | ||||||
|  |  | ||||||
| export function isCacheCleanupEnabled(): boolean { |  | ||||||
|     return params.isCacheCleanupEnabled() |  | ||||||
| } |  | ||||||
|  |  | ||||||
| /** |  | ||||||
|  * Represents a key used to restore a cache entry. |  | ||||||
|  * The Github Actions cache will first try for an exact match on the key. |  | ||||||
|  * If that fails, it will try for a prefix match on any of the restoreKeys. |  | ||||||
|  */ |  | ||||||
| export class CacheKey { |  | ||||||
|     key: string |  | ||||||
|     restoreKeys: string[] |  | ||||||
|  |  | ||||||
|     constructor(key: string, restoreKeys: string[]) { |  | ||||||
|         this.key = key |  | ||||||
|         this.restoreKeys = restoreKeys |  | ||||||
|     } |  | ||||||
| } |  | ||||||
|  |  | ||||||
| /** |  | ||||||
|  * Generates a cache key specific to the current job execution. |  | ||||||
|  * The key is constructed from the following inputs (with some user overrides): |  | ||||||
|  * - The cache protocol version |  | ||||||
|  * - The name of the cache |  | ||||||
|  * - The runner operating system |  | ||||||
|  * - The name of the workflow and Job being executed |  | ||||||
|  * - The matrix values for the Job being executed (job context) |  | ||||||
|  * - The SHA of the commit being executed |  | ||||||
|  * |  | ||||||
|  * Caches are restored by trying to match the these key prefixes in order: |  | ||||||
|  * - The full key with SHA |  | ||||||
|  * - A previous key for this Job + matrix |  | ||||||
|  * - Any previous key for this Job (any matrix) |  | ||||||
|  * - Any previous key for this cache on the current OS |  | ||||||
|  */ |  | ||||||
| export function generateCacheKey(cacheName: string): CacheKey { |  | ||||||
|     const cacheKeyBase = `${getCacheKeyPrefix()}${CACHE_PROTOCOL_VERSION}${cacheName}` |  | ||||||
|  |  | ||||||
|     // At the most general level, share caches for all executions on the same OS |  | ||||||
|     const cacheKeyForEnvironment = `${cacheKeyBase}|${getCacheKeyEnvironment()}` |  | ||||||
|  |  | ||||||
|     // Prefer caches that run this job |  | ||||||
|     const cacheKeyForJob = `${cacheKeyForEnvironment}|${getCacheKeyJob()}` |  | ||||||
|  |  | ||||||
|     // Prefer (even more) jobs that run this job with the same context (matrix) |  | ||||||
|     const cacheKeyForJobContext = `${cacheKeyForJob}[${getCacheKeyJobInstance()}]` |  | ||||||
|  |  | ||||||
|     // Exact match on Git SHA |  | ||||||
|     const cacheKey = `${cacheKeyForJobContext}-${getCacheKeyJobExecution()}` |  | ||||||
|  |  | ||||||
|     if (params.isCacheStrictMatch()) { |  | ||||||
|         return new CacheKey(cacheKey, [cacheKeyForJobContext]) |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     return new CacheKey(cacheKey, [cacheKeyForJobContext, cacheKeyForJob, cacheKeyForEnvironment]) |  | ||||||
| } |  | ||||||
|  |  | ||||||
| export function getCacheKeyPrefix(): string { |  | ||||||
|     // Prefix can be used to force change all cache keys (defaults to cache protocol version) |  | ||||||
|     return process.env[CACHE_KEY_PREFIX_VAR] || '' |  | ||||||
| } |  | ||||||
|  |  | ||||||
| function getCacheKeyEnvironment(): string { |  | ||||||
|     const runnerOs = process.env['RUNNER_OS'] || '' |  | ||||||
|     return process.env[CACHE_KEY_OS_VAR] || runnerOs |  | ||||||
| } |  | ||||||
|  |  | ||||||
| function getCacheKeyJob(): string { |  | ||||||
|     return process.env[CACHE_KEY_JOB_VAR] || getCacheKeyForJob(github.context.workflow, github.context.job) |  | ||||||
| } |  | ||||||
|  |  | ||||||
| export function getCacheKeyForJob(workflowName: string, jobId: string): string { |  | ||||||
|     const sanitizedWorkflow = workflowName.replace(/,/g, '').toLowerCase() |  | ||||||
|     return `${sanitizedWorkflow}-${jobId}` |  | ||||||
| } |  | ||||||
|  |  | ||||||
| function getCacheKeyJobInstance(): string { |  | ||||||
|     const override = process.env[CACHE_KEY_JOB_INSTANCE_VAR] |  | ||||||
|     if (override) { |  | ||||||
|         return override |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     // By default, we hash the full `matrix` data for the run, to uniquely identify this job invocation |  | ||||||
|     // The only way we can obtain the `matrix` data is via the `workflow-job-context` parameter in action.yml. |  | ||||||
|     const workflowJobContext = params.getJobMatrix() |  | ||||||
|     return hashStrings([workflowJobContext]) |  | ||||||
| } |  | ||||||
|  |  | ||||||
| export function getUniqueLabelForJobInstance(): string { |  | ||||||
|     return getUniqueLabelForJobInstanceValues(github.context.workflow, github.context.job, params.getJobMatrix()) |  | ||||||
| } |  | ||||||
|  |  | ||||||
| export function getUniqueLabelForJobInstanceValues(workflow: string, jobId: string, matrixJson: string): string { |  | ||||||
|     const matrix = JSON.parse(matrixJson) |  | ||||||
|     const matrixString = Object.values(matrix).join('-') |  | ||||||
|     const label = matrixString ? `${workflow}-${jobId}-${matrixString}` : `${workflow}-${jobId}` |  | ||||||
|     return sanitize(label) |  | ||||||
| } |  | ||||||
|  |  | ||||||
| function sanitize(value: string): string { |  | ||||||
|     return value.replace(/[^a-zA-Z0-9_-]/g, '').toLowerCase() |  | ||||||
| } |  | ||||||
|  |  | ||||||
| function getCacheKeyJobExecution(): string { |  | ||||||
|     // Used to associate a cache key with a particular execution (default is bound to the git commit sha) |  | ||||||
|     return process.env[CACHE_KEY_JOB_EXECUTION_VAR] || github.context.sha |  | ||||||
| } |  | ||||||
|  |  | ||||||
| export function hashFileNames(fileNames: string[]): string { |  | ||||||
|     return hashStrings(fileNames.map(x => x.replace(new RegExp(`\\${path.sep}`, 'g'), '/'))) |  | ||||||
| } |  | ||||||
|  |  | ||||||
| export function hashStrings(values: string[]): string { |  | ||||||
|     const hash = crypto.createHash('md5') |  | ||||||
|     for (const value of values) { |  | ||||||
|         hash.update(value) |  | ||||||
|     } |  | ||||||
|     return hash.digest('hex') |  | ||||||
| } |  | ||||||
|  |  | ||||||
| export async function restoreCache( |  | ||||||
|     cachePath: string[], |  | ||||||
|     cacheKey: string, |  | ||||||
|     cacheRestoreKeys: string[], |  | ||||||
|     listener: CacheEntryListener |  | ||||||
| ): Promise<cache.CacheEntry | undefined> { |  | ||||||
|     listener.markRequested(cacheKey, cacheRestoreKeys) |  | ||||||
|     try { |  | ||||||
|         // Only override the read timeout if the SEGMENT_DOWNLOAD_TIMEOUT_MINS env var has NOT been set |  | ||||||
|         const cacheRestoreOptions = process.env[SEGMENT_DOWNLOAD_TIMEOUT_VAR] |  | ||||||
|             ? {} |  | ||||||
|             : {segmentTimeoutInMs: SEGMENT_DOWNLOAD_TIMEOUT_DEFAULT} |  | ||||||
|         const restoredEntry = await cache.restoreCache(cachePath, cacheKey, cacheRestoreKeys, cacheRestoreOptions) |  | ||||||
|         if (restoredEntry !== undefined) { |  | ||||||
|             listener.markRestored(restoredEntry.key, restoredEntry.size) |  | ||||||
|         } |  | ||||||
|         return restoredEntry |  | ||||||
|     } catch (error) { |  | ||||||
|         listener.markNotRestored((error as Error).message) |  | ||||||
|         handleCacheFailure(error, `Failed to restore ${cacheKey}`) |  | ||||||
|         return undefined |  | ||||||
|     } |  | ||||||
| } |  | ||||||
|  |  | ||||||
| export async function saveCache(cachePath: string[], cacheKey: string, listener: CacheEntryListener): Promise<void> { |  | ||||||
|     try { |  | ||||||
|         const savedEntry = await cache.saveCache(cachePath, cacheKey) |  | ||||||
|         listener.markSaved(savedEntry.key, savedEntry.size) |  | ||||||
|     } catch (error) { |  | ||||||
|         if (error instanceof cache.ReserveCacheError) { |  | ||||||
|             listener.markAlreadyExists(cacheKey) |  | ||||||
|         } else { |  | ||||||
|             listener.markNotSaved((error as Error).message) |  | ||||||
|         } |  | ||||||
|         handleCacheFailure(error, `Failed to save cache entry with path '${cachePath}' and key: ${cacheKey}`) |  | ||||||
|     } |  | ||||||
| } |  | ||||||
|  |  | ||||||
| export function cacheDebug(message: string): void { |  | ||||||
|     if (isCacheDebuggingEnabled()) { |  | ||||||
|         core.info(message) |  | ||||||
|     } else { |  | ||||||
|         core.debug(message) |  | ||||||
|     } |  | ||||||
| } |  | ||||||
|  |  | ||||||
| export function handleCacheFailure(error: unknown, message: string): void { |  | ||||||
|     if (error instanceof cache.ValidationError) { |  | ||||||
|         // Fail on cache validation errors |  | ||||||
|         throw error |  | ||||||
|     } |  | ||||||
|     if (error instanceof cache.ReserveCacheError) { |  | ||||||
|         // Reserve cache errors are expected if the artifact has been previously cached |  | ||||||
|         core.info(`${message}: ${error}`) |  | ||||||
|     } else { |  | ||||||
|         // Warn on all other errors |  | ||||||
|         core.warning(`${message}: ${error}`) |  | ||||||
|         if (error instanceof Error && error.stack) { |  | ||||||
|             cacheDebug(error.stack) |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| } |  | ||||||
|  |  | ||||||
| /** |  | ||||||
|  * Attempt to delete a file or directory, waiting to allow locks to be released |  | ||||||
|  */ |  | ||||||
| export async function tryDelete(file: string): Promise<void> { |  | ||||||
|     const maxAttempts = 5 |  | ||||||
|     for (let attempt = 1; attempt <= maxAttempts; attempt++) { |  | ||||||
|         if (!fs.existsSync(file)) { |  | ||||||
|             return |  | ||||||
|         } |  | ||||||
|         try { |  | ||||||
|             const stat = fs.lstatSync(file) |  | ||||||
|             if (stat.isDirectory()) { |  | ||||||
|                 fs.rmSync(file, {recursive: true}) |  | ||||||
|             } else { |  | ||||||
|                 fs.unlinkSync(file) |  | ||||||
|             } |  | ||||||
|             return |  | ||||||
|         } catch (error) { |  | ||||||
|             if (attempt === maxAttempts) { |  | ||||||
|                 core.warning(`Failed to delete ${file}, which will impact caching.  |  | ||||||
| It is likely locked by another process. Output of 'jps -ml': |  | ||||||
| ${await getJavaProcesses()}`) |  | ||||||
|                 throw error |  | ||||||
|             } else { |  | ||||||
|                 cacheDebug(`Attempt to delete ${file} failed. Will try again.`) |  | ||||||
|                 await delay(1000) |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| } |  | ||||||
|  |  | ||||||
| async function delay(ms: number): Promise<void> { |  | ||||||
|     return new Promise(resolve => setTimeout(resolve, ms)) |  | ||||||
| } |  | ||||||
|  |  | ||||||
| async function getJavaProcesses(): Promise<string> { |  | ||||||
|     const jpsOutput = await exec.getExecOutput('jps', ['-lm']) |  | ||||||
|     return jpsOutput.stdout |  | ||||||
| } |  | ||||||
							
								
								
									
										103
									
								
								src/caches.ts
									
									
									
									
									
								
							
							
						
						
									
										103
									
								
								src/caches.ts
									
									
									
									
									
								
							| @@ -1,103 +0,0 @@ | |||||||
| import * as core from '@actions/core' |  | ||||||
| import { |  | ||||||
|     isCacheCleanupEnabled, |  | ||||||
|     isCacheDisabled, |  | ||||||
|     isCacheReadOnly, |  | ||||||
|     isCacheWriteOnly, |  | ||||||
|     isCacheOverwriteExisting |  | ||||||
| } from './cache-utils' |  | ||||||
| import {CacheListener} from './cache-reporting' |  | ||||||
| import {DaemonController} from './daemon-controller' |  | ||||||
| import {GradleStateCache} from './cache-base' |  | ||||||
| import {CacheCleaner} from './cache-cleaner' |  | ||||||
|  |  | ||||||
| const CACHE_RESTORED_VAR = 'GRADLE_BUILD_ACTION_CACHE_RESTORED' |  | ||||||
|  |  | ||||||
| export async function restore(gradleUserHome: string, cacheListener: CacheListener): Promise<void> { |  | ||||||
|     // Bypass restore cache on all but first action step in workflow. |  | ||||||
|     if (process.env[CACHE_RESTORED_VAR]) { |  | ||||||
|         core.info('Cache only restored on first action step.') |  | ||||||
|         return |  | ||||||
|     } |  | ||||||
|     core.exportVariable(CACHE_RESTORED_VAR, true) |  | ||||||
|  |  | ||||||
|     const gradleStateCache = new GradleStateCache(gradleUserHome) |  | ||||||
|  |  | ||||||
|     if (isCacheDisabled()) { |  | ||||||
|         core.info('Cache is disabled: will not restore state from previous builds.') |  | ||||||
|         // Initialize the Gradle User Home even when caching is disabled. |  | ||||||
|         gradleStateCache.init() |  | ||||||
|         cacheListener.cacheDisabled = true |  | ||||||
|         return |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     if (gradleStateCache.cacheOutputExists()) { |  | ||||||
|         if (!isCacheOverwriteExisting()) { |  | ||||||
|             core.info('Gradle User Home already exists: will not restore from cache.') |  | ||||||
|             // Initialize pre-existing Gradle User Home. |  | ||||||
|             gradleStateCache.init() |  | ||||||
|             cacheListener.cacheDisabled = true |  | ||||||
|             cacheListener.cacheDisabledReason = 'disabled due to pre-existing Gradle User Home' |  | ||||||
|             return |  | ||||||
|         } |  | ||||||
|         core.info('Gradle User Home already exists: will overwrite with cached contents.') |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     gradleStateCache.init() |  | ||||||
|     // Mark the state as restored so that post-action will perform save. |  | ||||||
|     core.saveState(CACHE_RESTORED_VAR, true) |  | ||||||
|  |  | ||||||
|     if (isCacheWriteOnly()) { |  | ||||||
|         core.info('Cache is write-only: will not restore from cache.') |  | ||||||
|         cacheListener.cacheWriteOnly = true |  | ||||||
|         return |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     await core.group('Restore Gradle state from cache', async () => { |  | ||||||
|         await gradleStateCache.restore(cacheListener) |  | ||||||
|     }) |  | ||||||
|  |  | ||||||
|     if (isCacheCleanupEnabled() && !isCacheReadOnly()) { |  | ||||||
|         core.info('Preparing cache for cleanup.') |  | ||||||
|         const cacheCleaner = new CacheCleaner(gradleUserHome, process.env['RUNNER_TEMP']!) |  | ||||||
|         await cacheCleaner.prepare() |  | ||||||
|     } |  | ||||||
| } |  | ||||||
|  |  | ||||||
| export async function save( |  | ||||||
|     gradleUserHome: string, |  | ||||||
|     cacheListener: CacheListener, |  | ||||||
|     daemonController: DaemonController |  | ||||||
| ): Promise<void> { |  | ||||||
|     if (isCacheDisabled()) { |  | ||||||
|         core.info('Cache is disabled: will not save state for later builds.') |  | ||||||
|         return |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     if (!core.getState(CACHE_RESTORED_VAR)) { |  | ||||||
|         core.info('Cache will not be saved: not restored in main action step.') |  | ||||||
|         return |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     if (isCacheReadOnly()) { |  | ||||||
|         core.info('Cache is read-only: will not save state for use in subsequent builds.') |  | ||||||
|         cacheListener.cacheReadOnly = true |  | ||||||
|         return |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     await daemonController.stopAllDaemons() |  | ||||||
|  |  | ||||||
|     if (isCacheCleanupEnabled()) { |  | ||||||
|         core.info('Forcing cache cleanup.') |  | ||||||
|         const cacheCleaner = new CacheCleaner(gradleUserHome, process.env['RUNNER_TEMP']!) |  | ||||||
|         try { |  | ||||||
|             await cacheCleaner.forceCleanup() |  | ||||||
|         } catch (e) { |  | ||||||
|             core.warning(`Cache cleanup failed. Will continue. ${String(e)}`) |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     await core.group('Caching Gradle state', async () => { |  | ||||||
|         return new GradleStateCache(gradleUserHome).save(cacheListener) |  | ||||||
|     }) |  | ||||||
| } |  | ||||||
| @@ -1,36 +0,0 @@ | |||||||
| import * as core from '@actions/core' |  | ||||||
| import * as exec from '@actions/exec' |  | ||||||
| import * as fs from 'fs' |  | ||||||
| import * as path from 'path' |  | ||||||
| import {BuildResult} from './build-results' |  | ||||||
|  |  | ||||||
| export class DaemonController { |  | ||||||
|     private readonly gradleHomes |  | ||||||
|  |  | ||||||
|     constructor(buildResults: BuildResult[]) { |  | ||||||
|         const allHomes = buildResults.map(buildResult => buildResult.gradleHomeDir) |  | ||||||
|         this.gradleHomes = Array.from(new Set(allHomes)) |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     async stopAllDaemons(): Promise<void> { |  | ||||||
|         core.info('Stopping all Gradle daemons before saving Gradle User Home state') |  | ||||||
|  |  | ||||||
|         const executions: Promise<number>[] = [] |  | ||||||
|         const args = ['--stop'] |  | ||||||
|  |  | ||||||
|         for (const gradleHome of this.gradleHomes) { |  | ||||||
|             const executable = path.resolve(gradleHome, 'bin', 'gradle') |  | ||||||
|             if (!fs.existsSync(executable)) { |  | ||||||
|                 core.warning(`Gradle executable not found at ${executable}. Could not stop Gradle daemons.`) |  | ||||||
|                 continue |  | ||||||
|             } |  | ||||||
|             core.info(`Stopping Gradle daemons for ${gradleHome}`) |  | ||||||
|             executions.push( |  | ||||||
|                 exec.exec(executable, args, { |  | ||||||
|                     ignoreReturnCode: true |  | ||||||
|                 }) |  | ||||||
|             ) |  | ||||||
|         } |  | ||||||
|         await Promise.all(executions) |  | ||||||
|     } |  | ||||||
| } |  | ||||||
| @@ -1,228 +0,0 @@ | |||||||
| import * as core from '@actions/core' |  | ||||||
| import * as artifact from '@actions/artifact' |  | ||||||
| import * as github from '@actions/github' |  | ||||||
| import * as glob from '@actions/glob' |  | ||||||
| import * as toolCache from '@actions/tool-cache' |  | ||||||
| import {GitHub} from '@actions/github/lib/utils' |  | ||||||
| import {RequestError} from '@octokit/request-error' |  | ||||||
| import type {PullRequestEvent} from '@octokit/webhooks-types' |  | ||||||
|  |  | ||||||
| import * as path from 'path' |  | ||||||
| import fs from 'fs' |  | ||||||
|  |  | ||||||
| import * as layout from './repository-layout' |  | ||||||
| import {DependencyGraphOption, getJobMatrix, getArtifactRetentionDays} from './input-params' |  | ||||||
|  |  | ||||||
| const DEPENDENCY_GRAPH_ARTIFACT = 'dependency-graph' |  | ||||||
|  |  | ||||||
| export async function setup(option: DependencyGraphOption): Promise<void> { |  | ||||||
|     if (option === DependencyGraphOption.Disabled) { |  | ||||||
|         return |  | ||||||
|     } |  | ||||||
|     // Download and submit early, for compatability with dependency review. |  | ||||||
|     if (option === DependencyGraphOption.DownloadAndSubmit) { |  | ||||||
|         await downloadAndSubmitDependencyGraphs() |  | ||||||
|         return |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     core.info('Enabling dependency graph generation') |  | ||||||
|     core.exportVariable('GITHUB_DEPENDENCY_GRAPH_ENABLED', 'true') |  | ||||||
|     core.exportVariable('GITHUB_DEPENDENCY_GRAPH_JOB_CORRELATOR', getJobCorrelator()) |  | ||||||
|     core.exportVariable('GITHUB_DEPENDENCY_GRAPH_JOB_ID', github.context.runId) |  | ||||||
|     core.exportVariable('GITHUB_DEPENDENCY_GRAPH_REF', github.context.ref) |  | ||||||
|     core.exportVariable('GITHUB_DEPENDENCY_GRAPH_SHA', getShaFromContext()) |  | ||||||
|     core.exportVariable('GITHUB_DEPENDENCY_GRAPH_WORKSPACE', layout.workspaceDirectory()) |  | ||||||
|     core.exportVariable( |  | ||||||
|         'DEPENDENCY_GRAPH_REPORT_DIR', |  | ||||||
|         path.resolve(layout.workspaceDirectory(), 'dependency-graph-reports') |  | ||||||
|     ) |  | ||||||
| } |  | ||||||
|  |  | ||||||
| export async function complete(option: DependencyGraphOption): Promise<void> { |  | ||||||
|     switch (option) { |  | ||||||
|         case DependencyGraphOption.Disabled: |  | ||||||
|         case DependencyGraphOption.DownloadAndSubmit: // Performed in setup |  | ||||||
|             return |  | ||||||
|         case DependencyGraphOption.Generate: |  | ||||||
|             await uploadDependencyGraphs() |  | ||||||
|             return |  | ||||||
|         case DependencyGraphOption.GenerateAndSubmit: |  | ||||||
|             await submitDependencyGraphs(await uploadDependencyGraphs()) |  | ||||||
|             return |  | ||||||
|     } |  | ||||||
| } |  | ||||||
|  |  | ||||||
| async function uploadDependencyGraphs(): Promise<string[]> { |  | ||||||
|     const workspaceDirectory = layout.workspaceDirectory() |  | ||||||
|     const graphFiles = await findDependencyGraphFiles(workspaceDirectory) |  | ||||||
|  |  | ||||||
|     const relativeGraphFiles = graphFiles.map(x => getRelativePathFromWorkspace(x)) |  | ||||||
|     core.info(`Uploading dependency graph files: ${relativeGraphFiles}`) |  | ||||||
|  |  | ||||||
|     const artifactClient = artifact.create() |  | ||||||
|     artifactClient.uploadArtifact(DEPENDENCY_GRAPH_ARTIFACT, graphFiles, workspaceDirectory, { |  | ||||||
|         retentionDays: getArtifactRetentionDays() |  | ||||||
|     }) |  | ||||||
|  |  | ||||||
|     return graphFiles |  | ||||||
| } |  | ||||||
|  |  | ||||||
| async function downloadAndSubmitDependencyGraphs(): Promise<void> { |  | ||||||
|     const workspaceDirectory = layout.workspaceDirectory() |  | ||||||
|     submitDependencyGraphs(await retrieveDependencyGraphs(workspaceDirectory)) |  | ||||||
| } |  | ||||||
|  |  | ||||||
| async function submitDependencyGraphs(dependencyGraphFiles: string[]): Promise<void> { |  | ||||||
|     for (const jsonFile of dependencyGraphFiles) { |  | ||||||
|         try { |  | ||||||
|             await submitDependencyGraphFile(jsonFile) |  | ||||||
|         } catch (error) { |  | ||||||
|             if (error instanceof RequestError) { |  | ||||||
|                 core.warning(buildWarningMessage(jsonFile, error)) |  | ||||||
|             } else { |  | ||||||
|                 throw error |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| } |  | ||||||
|  |  | ||||||
| function buildWarningMessage(jsonFile: string, error: RequestError): string { |  | ||||||
|     const relativeJsonFile = getRelativePathFromWorkspace(jsonFile) |  | ||||||
|     const mainWarning = `Failed to submit dependency graph ${relativeJsonFile}.\n${String(error)}` |  | ||||||
|     if (error.message === 'Resource not accessible by integration') { |  | ||||||
|         return `${mainWarning} |  | ||||||
| Please ensure that the 'contents: write' permission is available for the workflow job. |  | ||||||
| Note that this permission is never available for a 'pull_request' trigger from a repository fork. |  | ||||||
|         ` |  | ||||||
|     } |  | ||||||
|     return mainWarning |  | ||||||
| } |  | ||||||
|  |  | ||||||
| async function submitDependencyGraphFile(jsonFile: string): Promise<void> { |  | ||||||
|     const octokit = getOctokit() |  | ||||||
|     const jsonContent = fs.readFileSync(jsonFile, 'utf8') |  | ||||||
|  |  | ||||||
|     const jsonObject = JSON.parse(jsonContent) |  | ||||||
|     jsonObject.owner = github.context.repo.owner |  | ||||||
|     jsonObject.repo = github.context.repo.repo |  | ||||||
|     const response = await octokit.request('POST /repos/{owner}/{repo}/dependency-graph/snapshots', jsonObject) |  | ||||||
|  |  | ||||||
|     const relativeJsonFile = getRelativePathFromWorkspace(jsonFile) |  | ||||||
|     core.notice(`Submitted ${relativeJsonFile}: ${response.data.message}`) |  | ||||||
| } |  | ||||||
|  |  | ||||||
| async function retrieveDependencyGraphs(workspaceDirectory: string): Promise<string[]> { |  | ||||||
|     if (github.context.payload.workflow_run) { |  | ||||||
|         return await retrieveDependencyGraphsForWorkflowRun(github.context.payload.workflow_run.id, workspaceDirectory) |  | ||||||
|     } |  | ||||||
|     return retrieveDependencyGraphsForCurrentWorkflow(workspaceDirectory) |  | ||||||
| } |  | ||||||
|  |  | ||||||
| async function retrieveDependencyGraphsForWorkflowRun(runId: number, workspaceDirectory: string): Promise<string[]> { |  | ||||||
|     const octokit = getOctokit() |  | ||||||
|  |  | ||||||
|     // Find the workflow run artifacts named "dependency-graph" |  | ||||||
|     const artifacts = await octokit.rest.actions.listWorkflowRunArtifacts({ |  | ||||||
|         owner: github.context.repo.owner, |  | ||||||
|         repo: github.context.repo.repo, |  | ||||||
|         run_id: runId |  | ||||||
|     }) |  | ||||||
|  |  | ||||||
|     const matchArtifact = artifacts.data.artifacts.find(candidate => { |  | ||||||
|         return candidate.name === DEPENDENCY_GRAPH_ARTIFACT |  | ||||||
|     }) |  | ||||||
|  |  | ||||||
|     if (matchArtifact === undefined) { |  | ||||||
|         throw new Error(`Dependency graph artifact not found. Has it been generated by workflow run '${runId}'?`) |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     // Download the dependency-graph artifact |  | ||||||
|     const download = await octokit.rest.actions.downloadArtifact({ |  | ||||||
|         owner: github.context.repo.owner, |  | ||||||
|         repo: github.context.repo.repo, |  | ||||||
|         artifact_id: matchArtifact.id, |  | ||||||
|         archive_format: 'zip' |  | ||||||
|     }) |  | ||||||
|  |  | ||||||
|     const downloadBuffer = download.data as ArrayBuffer |  | ||||||
|     const downloadZip = path.resolve(workspaceDirectory, 'dependency-graph.zip') |  | ||||||
|     fs.writeFileSync(downloadZip, Buffer.from(downloadBuffer)) |  | ||||||
|  |  | ||||||
|     // Expance the dependency-graph zip and locate each dependency-graph JSON file |  | ||||||
|     const extractDir = path.resolve(workspaceDirectory, 'dependency-graph') |  | ||||||
|     const extracted = await toolCache.extractZip(downloadZip, extractDir) |  | ||||||
|     core.info(`Extracted dependency graph artifacts to ${extracted}: ${fs.readdirSync(extracted)}`) |  | ||||||
|  |  | ||||||
|     return findDependencyGraphFiles(extracted) |  | ||||||
| } |  | ||||||
|  |  | ||||||
| async function retrieveDependencyGraphsForCurrentWorkflow(workspaceDirectory: string): Promise<string[]> { |  | ||||||
|     const artifactClient = artifact.create() |  | ||||||
|     const downloadPath = path.resolve(workspaceDirectory, 'dependency-graph') |  | ||||||
|     await artifactClient.downloadArtifact(DEPENDENCY_GRAPH_ARTIFACT, downloadPath) |  | ||||||
|     return await findDependencyGraphFiles(downloadPath) |  | ||||||
| } |  | ||||||
|  |  | ||||||
| async function findDependencyGraphFiles(dir: string): Promise<string[]> { |  | ||||||
|     const globber = await glob.create(`${dir}/dependency-graph-reports/*.json`) |  | ||||||
|     const graphFiles = globber.glob() |  | ||||||
|     return graphFiles |  | ||||||
| } |  | ||||||
|  |  | ||||||
| function getOctokit(): InstanceType<typeof GitHub> { |  | ||||||
|     return github.getOctokit(getGithubToken()) |  | ||||||
| } |  | ||||||
|  |  | ||||||
| function getGithubToken(): string { |  | ||||||
|     return core.getInput('github-token', {required: true}) |  | ||||||
| } |  | ||||||
|  |  | ||||||
| function getRelativePathFromWorkspace(file: string): string { |  | ||||||
|     const workspaceDirectory = layout.workspaceDirectory() |  | ||||||
|     return path.relative(workspaceDirectory, file) |  | ||||||
| } |  | ||||||
|  |  | ||||||
| function getShaFromContext(): string { |  | ||||||
|     const context = github.context |  | ||||||
|     const pullRequestEvents = [ |  | ||||||
|         'pull_request', |  | ||||||
|         'pull_request_comment', |  | ||||||
|         'pull_request_review', |  | ||||||
|         'pull_request_review_comment' |  | ||||||
|         // Note that pull_request_target is omitted here. |  | ||||||
|         // That event runs in the context of the base commit of the PR, |  | ||||||
|         // so the snapshot should not be associated with the head commit. |  | ||||||
|     ] |  | ||||||
|     if (pullRequestEvents.includes(context.eventName)) { |  | ||||||
|         const pr = (context.payload as PullRequestEvent).pull_request |  | ||||||
|         return pr.head.sha |  | ||||||
|     } else { |  | ||||||
|         return context.sha |  | ||||||
|     } |  | ||||||
| } |  | ||||||
|  |  | ||||||
| function getJobCorrelator(): string { |  | ||||||
|     return constructJobCorrelator(github.context.workflow, github.context.job, getJobMatrix()) |  | ||||||
| } |  | ||||||
|  |  | ||||||
| export function constructJobCorrelator(workflow: string, jobId: string, matrixJson: string): string { |  | ||||||
|     const matrixString = describeMatrix(matrixJson) |  | ||||||
|     const label = matrixString ? `${workflow}-${jobId}-${matrixString}` : `${workflow}-${jobId}` |  | ||||||
|     return sanitize(label) |  | ||||||
| } |  | ||||||
|  |  | ||||||
| function describeMatrix(matrixJson: string): string { |  | ||||||
|     core.debug(`Got matrix json: ${matrixJson}`) |  | ||||||
|     const matrix = JSON.parse(matrixJson) |  | ||||||
|     if (matrix) { |  | ||||||
|         return Object.values(matrix).join('-') |  | ||||||
|     } |  | ||||||
|     return '' |  | ||||||
| } |  | ||||||
|  |  | ||||||
| function sanitize(value: string): string { |  | ||||||
|     return value |  | ||||||
|         .replace(/[^a-zA-Z0-9_-\s]/g, '') |  | ||||||
|         .replace(/\s+/g, '_') |  | ||||||
|         .toLowerCase() |  | ||||||
| } |  | ||||||
| @@ -1,17 +0,0 @@ | |||||||
| import * as core from '@actions/core' |  | ||||||
| import * as exec from '@actions/exec' |  | ||||||
| import * as gradlew from './gradlew' |  | ||||||
|  |  | ||||||
| export async function executeGradleBuild(executable: string | undefined, root: string, args: string[]): Promise<void> { |  | ||||||
|     // Use the provided executable, or look for a Gradle wrapper script to run |  | ||||||
|     const toExecute = executable ?? gradlew.gradleWrapperScript(root) |  | ||||||
|  |  | ||||||
|     const status: number = await exec.exec(toExecute, args, { |  | ||||||
|         cwd: root, |  | ||||||
|         ignoreReturnCode: true |  | ||||||
|     }) |  | ||||||
|  |  | ||||||
|     if (status !== 0) { |  | ||||||
|         core.setFailed(`Gradle build failed: see console output for details`) |  | ||||||
|     } |  | ||||||
| } |  | ||||||
| @@ -1,42 +0,0 @@ | |||||||
| import * as path from 'path' |  | ||||||
| import fs from 'fs' |  | ||||||
|  |  | ||||||
| const IS_WINDOWS = process.platform === 'win32' |  | ||||||
|  |  | ||||||
| export function wrapperScriptFilename(): string { |  | ||||||
|     return IS_WINDOWS ? './gradlew.bat' : './gradlew' |  | ||||||
| } |  | ||||||
|  |  | ||||||
| export function installScriptFilename(): string { |  | ||||||
|     return IS_WINDOWS ? 'gradle.bat' : 'gradle' |  | ||||||
| } |  | ||||||
|  |  | ||||||
| export function gradleWrapperScript(buildRootDirectory: string): string { |  | ||||||
|     validateGradleWrapper(buildRootDirectory) |  | ||||||
|     return wrapperScriptFilename() |  | ||||||
| } |  | ||||||
|  |  | ||||||
| function validateGradleWrapper(buildRootDirectory: string): void { |  | ||||||
|     const wrapperScript = path.resolve(buildRootDirectory, wrapperScriptFilename()) |  | ||||||
|     verifyExists(wrapperScript, 'Gradle Wrapper script') |  | ||||||
|     verifyIsExecutableScript(wrapperScript) |  | ||||||
|  |  | ||||||
|     const wrapperProperties = path.resolve(buildRootDirectory, 'gradle/wrapper/gradle-wrapper.properties') |  | ||||||
|     verifyExists(wrapperProperties, 'Gradle wrapper properties file') |  | ||||||
| } |  | ||||||
|  |  | ||||||
| function verifyExists(file: string, description: string): void { |  | ||||||
|     if (!fs.existsSync(file)) { |  | ||||||
|         throw new Error( |  | ||||||
|             `Cannot locate ${description} at '${file}'. Specify 'gradle-version' or 'gradle-executable' for projects without Gradle wrapper configured.` |  | ||||||
|         ) |  | ||||||
|     } |  | ||||||
| } |  | ||||||
|  |  | ||||||
| function verifyIsExecutableScript(toExecute: string): void { |  | ||||||
|     try { |  | ||||||
|         fs.accessSync(toExecute, fs.constants.X_OK) |  | ||||||
|     } catch (err) { |  | ||||||
|         throw new Error(`Gradle script '${toExecute}' is not executable.`) |  | ||||||
|     } |  | ||||||
| } |  | ||||||
| @@ -1,126 +0,0 @@ | |||||||
| import * as core from '@actions/core' |  | ||||||
| import {parseArgsStringToArgv} from 'string-argv' |  | ||||||
|  |  | ||||||
| export function isCacheDisabled(): boolean { |  | ||||||
|     return getBooleanInput('cache-disabled') |  | ||||||
| } |  | ||||||
|  |  | ||||||
| export function isCacheReadOnly(): boolean { |  | ||||||
|     return getBooleanInput('cache-read-only') |  | ||||||
| } |  | ||||||
|  |  | ||||||
| export function isCacheWriteOnly(): boolean { |  | ||||||
|     return getBooleanInput('cache-write-only') |  | ||||||
| } |  | ||||||
|  |  | ||||||
| export function isCacheOverwriteExisting(): boolean { |  | ||||||
|     return getBooleanInput('cache-overwrite-existing') |  | ||||||
| } |  | ||||||
|  |  | ||||||
| export function isCacheStrictMatch(): boolean { |  | ||||||
|     return getBooleanInput('gradle-home-cache-strict-match') |  | ||||||
| } |  | ||||||
|  |  | ||||||
| export function isCacheDebuggingEnabled(): boolean { |  | ||||||
|     return process.env['GRADLE_BUILD_ACTION_CACHE_DEBUG_ENABLED'] ? true : false |  | ||||||
| } |  | ||||||
|  |  | ||||||
| export function isCacheCleanupEnabled(): boolean { |  | ||||||
|     return getBooleanInput('gradle-home-cache-cleanup') |  | ||||||
| } |  | ||||||
|  |  | ||||||
| export function getCacheIncludes(): string[] { |  | ||||||
|     return core.getMultilineInput('gradle-home-cache-includes') |  | ||||||
| } |  | ||||||
|  |  | ||||||
| export function getCacheExcludes(): string[] { |  | ||||||
|     return core.getMultilineInput('gradle-home-cache-excludes') |  | ||||||
| } |  | ||||||
|  |  | ||||||
| export function getBuildRootDirectory(): string { |  | ||||||
|     return core.getInput('build-root-directory') |  | ||||||
| } |  | ||||||
|  |  | ||||||
| export function getGradleVersion(): string { |  | ||||||
|     return core.getInput('gradle-version') |  | ||||||
| } |  | ||||||
|  |  | ||||||
| export function getGradleExecutable(): string { |  | ||||||
|     return core.getInput('gradle-executable') |  | ||||||
| } |  | ||||||
|  |  | ||||||
| export function getArguments(): string[] { |  | ||||||
|     const input = core.getInput('arguments') |  | ||||||
|     return parseArgsStringToArgv(input) |  | ||||||
| } |  | ||||||
|  |  | ||||||
| // Internal parameters |  | ||||||
| export function getJobMatrix(): string { |  | ||||||
|     return core.getInput('workflow-job-context') |  | ||||||
| } |  | ||||||
|  |  | ||||||
| export function getGithubToken(): string { |  | ||||||
|     return core.getInput('github-token', {required: true}) |  | ||||||
| } |  | ||||||
|  |  | ||||||
| export function isJobSummaryEnabled(): boolean { |  | ||||||
|     return getBooleanInput('generate-job-summary', true) |  | ||||||
| } |  | ||||||
|  |  | ||||||
| export function isDependencyGraphEnabled(): boolean { |  | ||||||
|     return getBooleanInput('generate-dependency-graph', true) |  | ||||||
| } |  | ||||||
|  |  | ||||||
| export function getDependencyGraphOption(): DependencyGraphOption { |  | ||||||
|     const val = core.getInput('dependency-graph') |  | ||||||
|     switch (val.toLowerCase().trim()) { |  | ||||||
|         case 'disabled': |  | ||||||
|             return DependencyGraphOption.Disabled |  | ||||||
|         case 'generate': |  | ||||||
|             return DependencyGraphOption.Generate |  | ||||||
|         case 'generate-and-submit': |  | ||||||
|             return DependencyGraphOption.GenerateAndSubmit |  | ||||||
|         case 'download-and-submit': |  | ||||||
|             return DependencyGraphOption.DownloadAndSubmit |  | ||||||
|     } |  | ||||||
|     throw TypeError( |  | ||||||
|         `The value '${val} is not valid for 'dependency-graph. Valid values are: [disabled, generate-and-upload, generate-and-submit, download-and-submit]. The default value is 'disabled'.` |  | ||||||
|     ) |  | ||||||
| } |  | ||||||
|  |  | ||||||
| export function getArtifactRetentionDays(): number { |  | ||||||
|     const val = core.getInput('artifact-retention-days') |  | ||||||
|     return parseNumericInput('artifact-retention-days', val, 0) |  | ||||||
|     // Zero indicates that the default repository settings should be used |  | ||||||
| } |  | ||||||
|  |  | ||||||
| export function parseNumericInput(paramName: string, paramValue: string, paramDefault: number): number { |  | ||||||
|     if (paramValue.length === 0) { |  | ||||||
|         return paramDefault |  | ||||||
|     } |  | ||||||
|     const numericValue = parseInt(paramValue) |  | ||||||
|     if (isNaN(numericValue)) { |  | ||||||
|         throw TypeError(`The value '${paramValue}' is not a valid numeric value for '${paramName}'.`) |  | ||||||
|     } |  | ||||||
|     return numericValue |  | ||||||
| } |  | ||||||
|  |  | ||||||
| function getBooleanInput(paramName: string, paramDefault = false): boolean { |  | ||||||
|     const paramValue = core.getInput(paramName) |  | ||||||
|     switch (paramValue.toLowerCase().trim()) { |  | ||||||
|         case '': |  | ||||||
|             return paramDefault |  | ||||||
|         case 'false': |  | ||||||
|             return false |  | ||||||
|         case 'true': |  | ||||||
|             return true |  | ||||||
|     } |  | ||||||
|     throw TypeError(`The value '${paramValue} is not valid for '${paramName}. Valid values are: [true, false]`) |  | ||||||
| } |  | ||||||
|  |  | ||||||
| export enum DependencyGraphOption { |  | ||||||
|     Disabled, |  | ||||||
|     Generate, |  | ||||||
|     GenerateAndSubmit, |  | ||||||
|     DownloadAndSubmit |  | ||||||
| } |  | ||||||
| @@ -1,94 +0,0 @@ | |||||||
| import * as core from '@actions/core' |  | ||||||
| import {BuildResult} from './build-results' |  | ||||||
| import {writeCachingReport, CacheListener, logCachingReport} from './cache-reporting' |  | ||||||
|  |  | ||||||
| export async function writeJobSummary(buildResults: BuildResult[], cacheListener: CacheListener): Promise<void> { |  | ||||||
|     core.info('Writing job summary') |  | ||||||
|  |  | ||||||
|     if (buildResults.length === 0) { |  | ||||||
|         core.debug('No Gradle build results found. Summary table will not be generated.') |  | ||||||
|     } else { |  | ||||||
|         writeSummaryTable(buildResults) |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     writeCachingReport(cacheListener) |  | ||||||
|  |  | ||||||
|     await core.summary.write() |  | ||||||
| } |  | ||||||
|  |  | ||||||
| export async function logJobSummary(buildResults: BuildResult[], cacheListener: CacheListener): Promise<void> { |  | ||||||
|     if (buildResults.length === 0) { |  | ||||||
|         core.debug('No Gradle build results found. Summary table will not be logged.') |  | ||||||
|     } else { |  | ||||||
|         logSummaryTable(buildResults) |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     logCachingReport(cacheListener) |  | ||||||
| } |  | ||||||
|  |  | ||||||
| function writeSummaryTable(results: BuildResult[]): void { |  | ||||||
|     core.summary.addHeading('Gradle Builds', 3) |  | ||||||
|  |  | ||||||
|     core.summary.addRaw(` |  | ||||||
| <table> |  | ||||||
|     <tr> |  | ||||||
|         <th>Root Project</th> |  | ||||||
|         <th>Requested Tasks</th> |  | ||||||
|         <th>Gradle Version</th> |  | ||||||
|         <th>Build Outcome</th> |  | ||||||
|         <th>Build Scan®</th> |  | ||||||
|     </tr>${results.map(result => renderBuildResultRow(result)).join('')} |  | ||||||
| </table> |  | ||||||
|     `) |  | ||||||
| } |  | ||||||
|  |  | ||||||
| function renderBuildResultRow(result: BuildResult): string { |  | ||||||
|     return ` |  | ||||||
|     <tr> |  | ||||||
|         <td>${result.rootProjectName}</td> |  | ||||||
|         <td>${result.requestedTasks}</td> |  | ||||||
|         <td align='center'>${result.gradleVersion}</td> |  | ||||||
|         <td align='center'>${renderOutcome(result)}</td> |  | ||||||
|         <td>${renderBuildScan(result)}</td> |  | ||||||
|     </tr>` |  | ||||||
| } |  | ||||||
|  |  | ||||||
| function renderOutcome(result: BuildResult): string { |  | ||||||
|     return result.buildFailed ? ':x:' : ':white_check_mark:' |  | ||||||
| } |  | ||||||
|  |  | ||||||
| function renderBuildScan(result: BuildResult): string { |  | ||||||
|     if (result.buildScanFailed) { |  | ||||||
|         return renderBuildScanBadge( |  | ||||||
|             'PUBLISH_FAILED', |  | ||||||
|             'orange', |  | ||||||
|             'https://docs.gradle.com/enterprise/gradle-plugin/#troubleshooting' |  | ||||||
|         ) |  | ||||||
|     } |  | ||||||
|     if (result.buildScanUri) { |  | ||||||
|         return renderBuildScanBadge('PUBLISHED', '06A0CE', result.buildScanUri) |  | ||||||
|     } |  | ||||||
|     return renderBuildScanBadge('NOT_PUBLISHED', 'lightgrey', 'https://scans.gradle.com') |  | ||||||
| } |  | ||||||
|  |  | ||||||
| function renderBuildScanBadge(outcomeText: string, outcomeColor: string, targetUrl: string): string { |  | ||||||
|     const badgeUrl = `https://img.shields.io/badge/Build%20Scan%C2%AE-${outcomeText}-${outcomeColor}?logo=Gradle` |  | ||||||
|     const badgeHtml = `<img src="${badgeUrl}" alt="Build Scan ${outcomeText}" />` |  | ||||||
|     return `<a href="${targetUrl}" rel="nofollow">${badgeHtml}</a>` |  | ||||||
| } |  | ||||||
|  |  | ||||||
| function logSummaryTable(results: BuildResult[]): void { |  | ||||||
|     core.info('============================') |  | ||||||
|     core.info('Gradle Builds') |  | ||||||
|     core.info('----------------------------') |  | ||||||
|     core.info('Root Project | Requested Tasks | Gradle Version | Build Outcome | Build Scan®') |  | ||||||
|     core.info('----------------------------') |  | ||||||
|     for (const result of results) { |  | ||||||
|         core.info( |  | ||||||
|             `${result.rootProjectName} | ${result.requestedTasks} | ${result.gradleVersion} | ${ |  | ||||||
|                 result.buildFailed ? 'FAILED' : 'SUCCESS' |  | ||||||
|             } | ${result.buildScanFailed ? 'Publish failed' : result.buildScanUri}` |  | ||||||
|         ) |  | ||||||
|     } |  | ||||||
|     core.info('============================') |  | ||||||
| } |  | ||||||
							
								
								
									
										34
									
								
								src/main.ts
									
									
									
									
									
								
							
							
						
						
									
										34
									
								
								src/main.ts
									
									
									
									
									
								
							| @@ -1,34 +0,0 @@ | |||||||
| import * as core from '@actions/core' |  | ||||||
|  |  | ||||||
| import * as setupGradle from './setup-gradle' |  | ||||||
| import * as execution from './execution' |  | ||||||
| import * as provisioner from './provision' |  | ||||||
| import * as layout from './repository-layout' |  | ||||||
| import * as params from './input-params' |  | ||||||
|  |  | ||||||
| /** |  | ||||||
|  * The main entry point for the action, called by Github Actions for the step. |  | ||||||
|  */ |  | ||||||
| export async function run(): Promise<void> { |  | ||||||
|     try { |  | ||||||
|         // Configure Gradle environment (Gradle User Home) |  | ||||||
|         await setupGradle.setup() |  | ||||||
|  |  | ||||||
|         // Download and install Gradle if required |  | ||||||
|         const executable = await provisioner.provisionGradle() |  | ||||||
|  |  | ||||||
|         // Only execute if arguments have been provided |  | ||||||
|         const args: string[] = params.getArguments() |  | ||||||
|         if (args.length > 0) { |  | ||||||
|             const buildRootDirectory = layout.buildRootDirectory() |  | ||||||
|             await execution.executeGradleBuild(executable, buildRootDirectory, args) |  | ||||||
|         } |  | ||||||
|     } catch (error) { |  | ||||||
|         core.setFailed(String(error)) |  | ||||||
|         if (error instanceof Error && error.stack) { |  | ||||||
|             core.info(error.stack) |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| } |  | ||||||
|  |  | ||||||
| run() |  | ||||||
							
								
								
									
										27
									
								
								src/post.ts
									
									
									
									
									
								
							
							
						
						
									
										27
									
								
								src/post.ts
									
									
									
									
									
								
							| @@ -1,27 +0,0 @@ | |||||||
| import * as core from '@actions/core' |  | ||||||
| import * as setupGradle from './setup-gradle' |  | ||||||
|  |  | ||||||
| // Catch and log any unhandled exceptions.  These exceptions can leak out of the uploadChunk method in |  | ||||||
| // @actions/toolkit when a failed upload closes the file descriptor causing any in-process reads to |  | ||||||
| // throw an uncaught exception.  Instead of failing this action, just warn. |  | ||||||
| process.on('uncaughtException', e => handleFailure(e)) |  | ||||||
|  |  | ||||||
| /** |  | ||||||
|  * The post-execution entry point for the action, called by Github Actions after completing all steps for the Job. |  | ||||||
|  */ |  | ||||||
| export async function run(): Promise<void> { |  | ||||||
|     try { |  | ||||||
|         await setupGradle.complete() |  | ||||||
|     } catch (error) { |  | ||||||
|         handleFailure(error) |  | ||||||
|     } |  | ||||||
| } |  | ||||||
|  |  | ||||||
| function handleFailure(error: unknown): void { |  | ||||||
|     core.warning(`Unhandled error in Gradle post-action - job will continue: ${error}`) |  | ||||||
|     if (error instanceof Error && error.stack) { |  | ||||||
|         core.info(error.stack) |  | ||||||
|     } |  | ||||||
| } |  | ||||||
|  |  | ||||||
| run() |  | ||||||
							
								
								
									
										187
									
								
								src/provision.ts
									
									
									
									
									
								
							
							
						
						
									
										187
									
								
								src/provision.ts
									
									
									
									
									
								
							| @@ -1,187 +0,0 @@ | |||||||
| import * as fs from 'fs' |  | ||||||
| import * as os from 'os' |  | ||||||
| import * as path from 'path' |  | ||||||
| import * as httpm from '@actions/http-client' |  | ||||||
| import * as core from '@actions/core' |  | ||||||
| import * as cache from '@actions/cache' |  | ||||||
| import * as toolCache from '@actions/tool-cache' |  | ||||||
|  |  | ||||||
| import * as gradlew from './gradlew' |  | ||||||
| import * as params from './input-params' |  | ||||||
| import * as layout from './repository-layout' |  | ||||||
| import {handleCacheFailure, isCacheDisabled, isCacheReadOnly} from './cache-utils' |  | ||||||
|  |  | ||||||
| const gradleVersionsBaseUrl = 'https://services.gradle.org/versions' |  | ||||||
|  |  | ||||||
| /** |  | ||||||
|  * Install any configured version of Gradle, adding the executable to the PATH. |  | ||||||
|  * @return Installed Gradle executable or undefined if no version configured. |  | ||||||
|  */ |  | ||||||
| export async function provisionGradle(): Promise<string | undefined> { |  | ||||||
|     const gradleVersion = params.getGradleVersion() |  | ||||||
|     if (gradleVersion !== '' && gradleVersion !== 'wrapper') { |  | ||||||
|         return addToPath(path.resolve(await installGradle(gradleVersion))) |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     const gradleExecutable = params.getGradleExecutable() |  | ||||||
|     if (gradleExecutable !== '') { |  | ||||||
|         const workspaceDirectory = layout.workspaceDirectory() |  | ||||||
|         return addToPath(path.resolve(workspaceDirectory, gradleExecutable)) |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     return undefined |  | ||||||
| } |  | ||||||
|  |  | ||||||
| async function addToPath(executable: string): Promise<string> { |  | ||||||
|     core.addPath(path.dirname(executable)) |  | ||||||
|     return executable |  | ||||||
| } |  | ||||||
|  |  | ||||||
| async function installGradle(version: string): Promise<string> { |  | ||||||
|     const versionInfo = await resolveGradleVersion(version) |  | ||||||
|     core.setOutput('gradle-version', versionInfo.version) |  | ||||||
|     return installGradleVersion(versionInfo) |  | ||||||
| } |  | ||||||
|  |  | ||||||
| async function resolveGradleVersion(version: string): Promise<GradleVersionInfo> { |  | ||||||
|     switch (version) { |  | ||||||
|         case 'current': |  | ||||||
|             return gradleCurrent() |  | ||||||
|         case 'rc': |  | ||||||
|             core.warning(`Specifying gradle-version 'rc' has been deprecated. Use 'release-candidate' instead.`) |  | ||||||
|             return gradleReleaseCandidate() |  | ||||||
|         case 'release-candidate': |  | ||||||
|             return gradleReleaseCandidate() |  | ||||||
|         case 'nightly': |  | ||||||
|             return gradleNightly() |  | ||||||
|         case 'release-nightly': |  | ||||||
|             return gradleReleaseNightly() |  | ||||||
|         default: |  | ||||||
|             return gradle(version) |  | ||||||
|     } |  | ||||||
| } |  | ||||||
|  |  | ||||||
| async function gradleCurrent(): Promise<GradleVersionInfo> { |  | ||||||
|     return await gradleVersionDeclaration(`${gradleVersionsBaseUrl}/current`) |  | ||||||
| } |  | ||||||
|  |  | ||||||
| async function gradleReleaseCandidate(): Promise<GradleVersionInfo> { |  | ||||||
|     const versionInfo = await gradleVersionDeclaration(`${gradleVersionsBaseUrl}/release-candidate`) |  | ||||||
|     if (versionInfo && versionInfo.version && versionInfo.downloadUrl) { |  | ||||||
|         return versionInfo |  | ||||||
|     } |  | ||||||
|     core.info('No current release-candidate found, will fallback to current') |  | ||||||
|     return gradleCurrent() |  | ||||||
| } |  | ||||||
|  |  | ||||||
| async function gradleNightly(): Promise<GradleVersionInfo> { |  | ||||||
|     return await gradleVersionDeclaration(`${gradleVersionsBaseUrl}/nightly`) |  | ||||||
| } |  | ||||||
|  |  | ||||||
| async function gradleReleaseNightly(): Promise<GradleVersionInfo> { |  | ||||||
|     return await gradleVersionDeclaration(`${gradleVersionsBaseUrl}/release-nightly`) |  | ||||||
| } |  | ||||||
|  |  | ||||||
| async function gradle(version: string): Promise<GradleVersionInfo> { |  | ||||||
|     const versionInfo = await findGradleVersionDeclaration(version) |  | ||||||
|     if (!versionInfo) { |  | ||||||
|         throw new Error(`Gradle version ${version} does not exists`) |  | ||||||
|     } |  | ||||||
|     return versionInfo |  | ||||||
| } |  | ||||||
|  |  | ||||||
| async function gradleVersionDeclaration(url: string): Promise<GradleVersionInfo> { |  | ||||||
|     return await httpGetGradleVersion(url) |  | ||||||
| } |  | ||||||
|  |  | ||||||
| async function findGradleVersionDeclaration(version: string): Promise<GradleVersionInfo | undefined> { |  | ||||||
|     const gradleVersions = await httpGetGradleVersions(`${gradleVersionsBaseUrl}/all`) |  | ||||||
|     return gradleVersions.find((entry: GradleVersionInfo) => { |  | ||||||
|         return entry.version === version |  | ||||||
|     }) |  | ||||||
| } |  | ||||||
|  |  | ||||||
| async function installGradleVersion(versionInfo: GradleVersionInfo): Promise<string> { |  | ||||||
|     return core.group(`Provision Gradle ${versionInfo.version}`, async () => { |  | ||||||
|         return locateGradleAndDownloadIfRequired(versionInfo) |  | ||||||
|     }) |  | ||||||
| } |  | ||||||
|  |  | ||||||
| async function locateGradleAndDownloadIfRequired(versionInfo: GradleVersionInfo): Promise<string> { |  | ||||||
|     const installsDir = path.join(os.homedir(), 'gradle-installations/installs') |  | ||||||
|     const installDir = path.join(installsDir, `gradle-${versionInfo.version}`) |  | ||||||
|     if (fs.existsSync(installDir)) { |  | ||||||
|         core.info(`Gradle installation already exists at ${installDir}`) |  | ||||||
|         return executableFrom(installDir) |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     const downloadPath = await downloadAndCacheGradleDistribution(versionInfo) |  | ||||||
|     await toolCache.extractZip(downloadPath, installsDir) |  | ||||||
|     core.info(`Extracted Gradle ${versionInfo.version} to ${installDir}`) |  | ||||||
|  |  | ||||||
|     const executable = executableFrom(installDir) |  | ||||||
|     fs.chmodSync(executable, '755') |  | ||||||
|     core.info(`Provisioned Gradle executable ${executable}`) |  | ||||||
|  |  | ||||||
|     return executable |  | ||||||
| } |  | ||||||
|  |  | ||||||
| async function downloadAndCacheGradleDistribution(versionInfo: GradleVersionInfo): Promise<string> { |  | ||||||
|     const downloadPath = path.join(os.homedir(), `gradle-installations/downloads/gradle-${versionInfo.version}-bin.zip`) |  | ||||||
|  |  | ||||||
|     if (isCacheDisabled()) { |  | ||||||
|         await downloadGradleDistribution(versionInfo, downloadPath) |  | ||||||
|         return downloadPath |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     const cacheKey = `gradle-${versionInfo.version}` |  | ||||||
|     try { |  | ||||||
|         const restoreKey = await cache.restoreCache([downloadPath], cacheKey) |  | ||||||
|         if (restoreKey) { |  | ||||||
|             core.info(`Restored Gradle distribution ${cacheKey} from cache to ${downloadPath}`) |  | ||||||
|             return downloadPath |  | ||||||
|         } |  | ||||||
|     } catch (error) { |  | ||||||
|         handleCacheFailure(error, `Restore Gradle distribution ${versionInfo.version} failed`) |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     core.info(`Gradle distribution ${versionInfo.version} not found in cache. Will download.`) |  | ||||||
|     await downloadGradleDistribution(versionInfo, downloadPath) |  | ||||||
|  |  | ||||||
|     if (!isCacheReadOnly()) { |  | ||||||
|         try { |  | ||||||
|             await cache.saveCache([downloadPath], cacheKey) |  | ||||||
|         } catch (error) { |  | ||||||
|             handleCacheFailure(error, `Save Gradle distribution ${versionInfo.version} failed`) |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
|     return downloadPath |  | ||||||
| } |  | ||||||
|  |  | ||||||
| async function downloadGradleDistribution(versionInfo: GradleVersionInfo, downloadPath: string): Promise<void> { |  | ||||||
|     await toolCache.downloadTool(versionInfo.downloadUrl, downloadPath) |  | ||||||
|     core.info(`Downloaded ${versionInfo.downloadUrl} to ${downloadPath} (size ${fs.statSync(downloadPath).size})`) |  | ||||||
| } |  | ||||||
|  |  | ||||||
| function executableFrom(installDir: string): string { |  | ||||||
|     return path.join(installDir, 'bin', `${gradlew.installScriptFilename()}`) |  | ||||||
| } |  | ||||||
|  |  | ||||||
| async function httpGetGradleVersion(url: string): Promise<GradleVersionInfo> { |  | ||||||
|     return JSON.parse(await httpGetString(url)) |  | ||||||
| } |  | ||||||
|  |  | ||||||
| async function httpGetGradleVersions(url: string): Promise<GradleVersionInfo[]> { |  | ||||||
|     return JSON.parse(await httpGetString(url)) |  | ||||||
| } |  | ||||||
|  |  | ||||||
| async function httpGetString(url: string): Promise<string> { |  | ||||||
|     const httpClient = new httpm.HttpClient('gradle/gradle-build-action') |  | ||||||
|     const response = await httpClient.get(url) |  | ||||||
|     return response.readBody() |  | ||||||
| } |  | ||||||
|  |  | ||||||
| interface GradleVersionInfo { |  | ||||||
|     version: string |  | ||||||
|     downloadUrl: string |  | ||||||
| } |  | ||||||
| @@ -1,16 +0,0 @@ | |||||||
| import * as params from './input-params' |  | ||||||
| import * as path from 'path' |  | ||||||
|  |  | ||||||
| export function workspaceDirectory(): string { |  | ||||||
|     return process.env[`GITHUB_WORKSPACE`] || '' |  | ||||||
| } |  | ||||||
|  |  | ||||||
| export function buildRootDirectory(): string { |  | ||||||
|     const baseDirectory = workspaceDirectory() |  | ||||||
|     const buildRootDirectoryInput = params.getBuildRootDirectory() |  | ||||||
|     const resolvedBuildRootDirectory = |  | ||||||
|         buildRootDirectoryInput === '' |  | ||||||
|             ? path.resolve(baseDirectory) |  | ||||||
|             : path.resolve(baseDirectory, buildRootDirectoryInput) |  | ||||||
|     return resolvedBuildRootDirectory |  | ||||||
| } |  | ||||||
| @@ -1,65 +0,0 @@ | |||||||
| import org.gradle.tooling.events.* |  | ||||||
| import org.gradle.tooling.events.task.* |  | ||||||
| import org.gradle.util.GradleVersion |  | ||||||
|  |  | ||||||
| // Can't use settingsEvaluated since this script is applied inside a settingsEvaluated handler |  | ||||||
| // But projectsEvaluated is good enough, since the build service won't catch configuration failures anyway |  | ||||||
| projectsEvaluated { |  | ||||||
|     def projectTracker = gradle.sharedServices.registerIfAbsent("gradle-build-action-buildResultsRecorder", BuildResultsRecorder, { spec -> |  | ||||||
|         spec.getParameters().getRootProjectName().set(gradle.rootProject.name) |  | ||||||
|         spec.getParameters().getRootProjectDir().set(gradle.rootProject.rootDir.absolutePath) |  | ||||||
|         spec.getParameters().getRequestedTasks().set(gradle.startParameter.taskNames.join(" ")) |  | ||||||
|         spec.getParameters().getGradleHomeDir().set(gradle.gradleHomeDir.absolutePath) |  | ||||||
|         spec.getParameters().getInvocationId().set(gradle.ext.invocationId) |  | ||||||
|     }) |  | ||||||
|  |  | ||||||
|     gradle.services.get(BuildEventsListenerRegistry).onTaskCompletion(projectTracker) |  | ||||||
| } |  | ||||||
|  |  | ||||||
| abstract class BuildResultsRecorder implements BuildService<BuildResultsRecorder.Params>, OperationCompletionListener, AutoCloseable { |  | ||||||
|     private boolean buildFailed = false |  | ||||||
|     interface Params extends BuildServiceParameters { |  | ||||||
|         Property<String> getRootProjectName() |  | ||||||
|         Property<String> getRootProjectDir() |  | ||||||
|         Property<String> getRequestedTasks() |  | ||||||
|         Property<String> getGradleHomeDir() |  | ||||||
|         Property<String> getInvocationId() |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     public void onFinish(FinishEvent finishEvent) { |  | ||||||
|         if (finishEvent instanceof TaskFinishEvent && finishEvent.result instanceof TaskFailureResult) { |  | ||||||
|             buildFailed = true |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     @Override |  | ||||||
|     public void close() { |  | ||||||
|         def buildResults = [ |  | ||||||
|             rootProjectName: getParameters().getRootProjectName().get(), |  | ||||||
|             rootProjectDir: getParameters().getRootProjectDir().get(), |  | ||||||
|             requestedTasks: getParameters().getRequestedTasks().get(), |  | ||||||
|             gradleVersion: GradleVersion.current().version, |  | ||||||
|             gradleHomeDir: getParameters().getGradleHomeDir().get(), |  | ||||||
|             buildFailed: buildFailed, |  | ||||||
|             buildScanUri: null, |  | ||||||
|             buildScanFailed: false |  | ||||||
|         ] |  | ||||||
|  |  | ||||||
|         def runnerTempDir = System.getenv("RUNNER_TEMP") |  | ||||||
|         def githubActionStep = System.getenv("GITHUB_ACTION") |  | ||||||
|         if (!runnerTempDir || !githubActionStep) { |  | ||||||
|             return |  | ||||||
|         } |  | ||||||
|  |  | ||||||
|         try { |  | ||||||
|             def buildResultsDir = new File(runnerTempDir, ".build-results") |  | ||||||
|             buildResultsDir.mkdirs() |  | ||||||
|             def buildResultsFile = new File(buildResultsDir, githubActionStep + getParameters().getInvocationId().get() + ".json") |  | ||||||
|             if (!buildResultsFile.exists()) { |  | ||||||
|                 buildResultsFile << groovy.json.JsonOutput.toJson(buildResults) |  | ||||||
|             } |  | ||||||
|         } catch (Exception e) { |  | ||||||
|             println "\ngradle-build-action failed to write build-results file. Will continue.\n> ${e.getLocalizedMessage()}" |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| } |  | ||||||
| @@ -1,143 +0,0 @@ | |||||||
| /* |  | ||||||
|  * Capture information for each executed Gradle build to display in the job summary. |  | ||||||
|  */ |  | ||||||
| import org.gradle.util.GradleVersion |  | ||||||
|  |  | ||||||
| // Only run against root build. Do not run against included builds. |  | ||||||
| def isTopLevelBuild = gradle.getParent() == null |  | ||||||
| if (isTopLevelBuild) { |  | ||||||
|     def version = GradleVersion.current().baseVersion |  | ||||||
|  |  | ||||||
|     def atLeastGradle3 = version >= GradleVersion.version("3.0") |  | ||||||
|     def atLeastGradle6 = version >= GradleVersion.version("6.0") |  | ||||||
|  |  | ||||||
|     def invocationId = "-${System.currentTimeMillis()}" |  | ||||||
|  |  | ||||||
|     if (atLeastGradle6) { |  | ||||||
|         def useBuildService = version >= GradleVersion.version("6.6") |  | ||||||
|         settingsEvaluated { settings -> |  | ||||||
|             // By default, use standard mechanisms to capture build results |  | ||||||
|             if (useBuildService) { |  | ||||||
|                 captureUsingBuildService(settings, invocationId) |  | ||||||
|             } else { |  | ||||||
|                 captureUsingBuildFinished(gradle, invocationId) |  | ||||||
|             } |  | ||||||
|              |  | ||||||
|             // The `buildScanPublished` hook allows the capture of the Build Scan URI. |  | ||||||
|             // Results captured this way will overwrite any results from the other mechanism. |  | ||||||
|             settings.pluginManager.withPlugin("com.gradle.enterprise") { |  | ||||||
|                 captureUsingBuildScanPublished(settings.extensions["gradleEnterprise"].buildScan, settings.rootProject, invocationId) |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
|     } else if (atLeastGradle3) { |  | ||||||
|         projectsEvaluated { gradle -> |  | ||||||
|             // By default, use 'buildFinished' to capture build results |  | ||||||
|             captureUsingBuildFinished(gradle, invocationId) |  | ||||||
|  |  | ||||||
|             // The `buildScanPublished` hook allows the capture of the Build Scan URI. |  | ||||||
|             // Results captured this way will overwrite any results from 'buildFinished'. |  | ||||||
|             gradle.rootProject.pluginManager.withPlugin("com.gradle.build-scan") { |  | ||||||
|                 captureUsingBuildScanPublished(gradle.rootProject.extensions["buildScan"], gradle.rootProject, invocationId) |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| } |  | ||||||
|  |  | ||||||
| def captureUsingBuildScanPublished(buildScanExtension, rootProject, invocationId) { |  | ||||||
|     buildScanExtension.with { |  | ||||||
|         def buildResults = new BuildResults(invocationId, gradle, rootProject) |  | ||||||
|  |  | ||||||
|         buildFinished { result -> |  | ||||||
|             buildResults.setBuildResult(result) |  | ||||||
|         } |  | ||||||
|  |  | ||||||
|         buildScanPublished { buildScan -> |  | ||||||
|             buildResults.setBuildScanUri(buildScan.buildScanUri.toASCIIString()) |  | ||||||
|             buildResults.writeToResultsFile(true) |  | ||||||
|  |  | ||||||
|             def githubOutput = System.getenv("GITHUB_OUTPUT") |  | ||||||
|             if (githubOutput) { |  | ||||||
|                 new File(githubOutput) << "build-scan-url=${buildScan.buildScanUri}\n" |  | ||||||
|             } else { |  | ||||||
|                 // Retained for compatibility with older GitHub Enterprise versions |  | ||||||
|                 println("::set-output name=build-scan-url::${buildScan.buildScanUri}") |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
|  |  | ||||||
|         onError { error -> |  | ||||||
|             buildResults.setBuildScanFailed() |  | ||||||
|             buildResults.writeToResultsFile(true) |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| } |  | ||||||
|  |  | ||||||
| def captureUsingBuildFinished(gradle, invocationId) { |  | ||||||
|     gradle.buildFinished { result -> |  | ||||||
|         println "Got buildFinished: ${result}" |  | ||||||
|         def buildResults = new BuildResults(invocationId, gradle, gradle.rootProject) |  | ||||||
|         buildResults.setBuildResult(result) |  | ||||||
|         buildResults.writeToResultsFile(false) |  | ||||||
|     } |  | ||||||
| } |  | ||||||
|  |  | ||||||
| def captureUsingBuildService(settings, invocationId) { |  | ||||||
|     gradle.ext.invocationId = invocationId |  | ||||||
|     apply from: 'gradle-build-action.build-result-capture-service.plugin.groovy' |  | ||||||
| } |  | ||||||
|  |  | ||||||
| class BuildResults { |  | ||||||
|     def invocationId |  | ||||||
|     def buildResults |  | ||||||
|  |  | ||||||
|     BuildResults(String invocationId, def gradle, def rootProject) { |  | ||||||
|         this.invocationId = invocationId |  | ||||||
|         buildResults = [ |  | ||||||
|             rootProjectName: rootProject.name, |  | ||||||
|             rootProjectDir: rootProject.projectDir.absolutePath, |  | ||||||
|             requestedTasks: gradle.startParameter.taskNames.join(" "), |  | ||||||
|             gradleVersion: GradleVersion.current().version, |  | ||||||
|             gradleHomeDir: gradle.gradleHomeDir.absolutePath, |  | ||||||
|             buildFailed: false, |  | ||||||
|             buildScanUri: null, |  | ||||||
|             buildScanFailed: false |  | ||||||
|         ] |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     def setBuildResult(def result) { |  | ||||||
|         buildResults['buildFailed'] = result.failure != null |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     def setBuildScanUri(def buildScanUrl) { |  | ||||||
|         buildResults['buildScanUri'] = buildScanUrl |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     def setBuildScanFailed() { |  | ||||||
|         buildResults['buildScanFailed'] = true |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     def writeToResultsFile(boolean overwrite) { |  | ||||||
|         def runnerTempDir = System.getenv("RUNNER_TEMP") |  | ||||||
|         def githubActionStep = System.getenv("GITHUB_ACTION") |  | ||||||
|         if (!runnerTempDir || !githubActionStep) { |  | ||||||
|             return |  | ||||||
|         } |  | ||||||
|  |  | ||||||
|         try { |  | ||||||
|             def buildResultsDir = new File(runnerTempDir, ".build-results") |  | ||||||
|             buildResultsDir.mkdirs() |  | ||||||
|             def buildResultsFile = new File(buildResultsDir, githubActionStep + invocationId + ".json") |  | ||||||
|  |  | ||||||
|             // Overwrite any contents written by buildFinished or build service, since this result is a superset. |  | ||||||
|             if (buildResultsFile.exists()) { |  | ||||||
|                 if (overwrite) { |  | ||||||
|                     buildResultsFile.text = groovy.json.JsonOutput.toJson(buildResults) |  | ||||||
|                 } |  | ||||||
|             } else { |  | ||||||
|                 buildResultsFile << groovy.json.JsonOutput.toJson(buildResults) |  | ||||||
|             } |  | ||||||
|  |  | ||||||
|         } catch (Exception e) { |  | ||||||
|             println "\ngradle-build-action failed to write build-results file. Will continue.\n> ${e.getLocalizedMessage()}" |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| } |  | ||||||
| @@ -1,15 +0,0 @@ | |||||||
| buildscript { |  | ||||||
|   def getInputParam = { String name -> |  | ||||||
|       def envVarName = name.toUpperCase().replace('.', '_').replace('-', '_') |  | ||||||
|       return System.getProperty(name) ?: System.getenv(envVarName) |  | ||||||
|   } |  | ||||||
|   def pluginRepositoryUrl = getInputParam('gradle.plugin-repository.url') ?: 'https://plugins.gradle.org/m2' |  | ||||||
|  |  | ||||||
|   repositories { |  | ||||||
|     maven { url pluginRepositoryUrl } |  | ||||||
|   } |  | ||||||
|   dependencies { |  | ||||||
|     classpath "org.gradle:github-dependency-graph-gradle-plugin:1.0.0" |  | ||||||
|   } |  | ||||||
| } |  | ||||||
| apply plugin: org.gradle.github.GitHubDependencyGraphPlugin |  | ||||||
| @@ -1,66 +0,0 @@ | |||||||
| import org.gradle.util.GradleVersion |  | ||||||
|  |  | ||||||
| // Only run when dependency graph is explicitly enabled |  | ||||||
| if (getVariable('GITHUB_DEPENDENCY_GRAPH_ENABLED') != "true") { |  | ||||||
|   return |  | ||||||
| } |  | ||||||
|  |  | ||||||
| // Do not run for unsupported versions of Gradle |  | ||||||
| if (GradleVersion.current().baseVersion < GradleVersion.version("5.0")) { |  | ||||||
|   println "::warning::Dependency Graph is not supported for Gradle versions < 5.0. No dependency snapshot will be generated." |  | ||||||
|   return |  | ||||||
| } |  | ||||||
|  |  | ||||||
| // Attempt to find a unique job correlator to use based on the environment variable |  | ||||||
| // This is only required for top-level builds |  | ||||||
| def isTopLevelBuild = gradle.getParent() == null |  | ||||||
| if (isTopLevelBuild) { |  | ||||||
|   def reportFile = getUniqueReportFile(getVariable('GITHUB_DEPENDENCY_GRAPH_JOB_CORRELATOR')) |  | ||||||
|  |  | ||||||
|   if (reportFile == null) { |  | ||||||
|     println "::warning::No dependency snapshot generated for step. Could not determine unique job correlator - specify GITHUB_DEPENDENCY_GRAPH_JOB_CORRELATOR var for this step." |  | ||||||
|     return |  | ||||||
|   } |  | ||||||
|  |  | ||||||
|   def githubOutput = System.getenv("GITHUB_OUTPUT") |  | ||||||
|   if (githubOutput) { |  | ||||||
|       new File(githubOutput) << "dependency-graph-file=${reportFile.absolutePath}\n" |  | ||||||
|   } |  | ||||||
|  |  | ||||||
|  |  | ||||||
|   println "Generating dependency graph into '${reportFile}'" |  | ||||||
| } |  | ||||||
|  |  | ||||||
| apply from: 'gradle-build-action.github-dependency-graph-gradle-plugin-apply.groovy' |  | ||||||
|  |  | ||||||
| /** |  | ||||||
|  * Using the supplied jobCorrelator value: |  | ||||||
|  * - Checks if report file already exists |  | ||||||
|  * - If so, tries to find a unique value that does not yet have a corresponding report file. |  | ||||||
|  * - When found, this value is set as a System property override. |  | ||||||
|  */ |  | ||||||
| File getUniqueReportFile(String jobCorrelator) { |  | ||||||
|     def reportDir = getVariable('DEPENDENCY_GRAPH_REPORT_DIR') |  | ||||||
|     def reportFile = new File(reportDir, jobCorrelator + ".json") |  | ||||||
|     if (!reportFile.exists()) return reportFile |  | ||||||
|  |  | ||||||
|     // Try at most 100 suffixes |  | ||||||
|     for (int i = 1; i < 100; i++) { |  | ||||||
|         def candidateCorrelator = jobCorrelator + "-" + i |  | ||||||
|         def candidateFile = new File(reportDir, candidateCorrelator + ".json") |  | ||||||
|         if (!candidateFile.exists()) { |  | ||||||
|            System.properties['GITHUB_DEPENDENCY_GRAPH_JOB_CORRELATOR'] = candidateCorrelator |  | ||||||
|            return candidateFile |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     // Could not determine unique job correlator |  | ||||||
|     return null |  | ||||||
| } |  | ||||||
|  |  | ||||||
| /** |  | ||||||
| * Return the environment variable value, or equivalent system property (if set) |  | ||||||
| */ |  | ||||||
| String getVariable(String name) { |  | ||||||
|   return System.properties[name] ?: System.getenv(name) |  | ||||||
| } |  | ||||||
| @@ -1,192 +0,0 @@ | |||||||
| import org.gradle.util.GradleVersion |  | ||||||
|  |  | ||||||
| // note that there is no mechanism to share code between the initscript{} block and the main script, so some logic is duplicated |  | ||||||
|  |  | ||||||
| // conditionally apply the GE / Build Scan plugin to the classpath so it can be applied to the build further down in this script |  | ||||||
| initscript { |  | ||||||
|     def isTopLevelBuild = !gradle.parent |  | ||||||
|     if (!isTopLevelBuild) { |  | ||||||
|         return |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     def getInputParam = { String name -> |  | ||||||
|         def envVarName = name.toUpperCase().replace('.', '_').replace('-', '_') |  | ||||||
|         return System.getProperty(name) ?: System.getenv(envVarName) |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     // finish early if injection is disabled |  | ||||||
|     def gradleInjectionEnabled = getInputParam("gradle-enterprise.injection-enabled") |  | ||||||
|     if (gradleInjectionEnabled != "true") { |  | ||||||
|         return |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     def pluginRepositoryUrl = getInputParam('gradle-enterprise.plugin-repository.url') |  | ||||||
|     def gePluginVersion = getInputParam('gradle-enterprise.plugin.version') |  | ||||||
|     def ccudPluginVersion = getInputParam('gradle-enterprise.ccud-plugin.version') |  | ||||||
|  |  | ||||||
|     def atLeastGradle5 = GradleVersion.current() >= GradleVersion.version('5.0') |  | ||||||
|     def atLeastGradle4 = GradleVersion.current() >= GradleVersion.version('4.0') |  | ||||||
|  |  | ||||||
|     if (gePluginVersion || ccudPluginVersion && atLeastGradle4) { |  | ||||||
|         pluginRepositoryUrl = pluginRepositoryUrl ?: 'https://plugins.gradle.org/m2' |  | ||||||
|         logger.quiet("Gradle Enterprise plugins resolution: $pluginRepositoryUrl") |  | ||||||
|  |  | ||||||
|         repositories { |  | ||||||
|             maven { url pluginRepositoryUrl } |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     dependencies { |  | ||||||
|         if (gePluginVersion) { |  | ||||||
|             classpath atLeastGradle5 ? |  | ||||||
|                 "com.gradle:gradle-enterprise-gradle-plugin:$gePluginVersion" : |  | ||||||
|                 "com.gradle:build-scan-plugin:1.16" |  | ||||||
|         } |  | ||||||
|  |  | ||||||
|         if (ccudPluginVersion && atLeastGradle4) { |  | ||||||
|             classpath "com.gradle:common-custom-user-data-gradle-plugin:$ccudPluginVersion" |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| } |  | ||||||
|  |  | ||||||
| def BUILD_SCAN_PLUGIN_ID = 'com.gradle.build-scan' |  | ||||||
| def BUILD_SCAN_PLUGIN_CLASS = 'com.gradle.scan.plugin.BuildScanPlugin' |  | ||||||
|  |  | ||||||
| def GRADLE_ENTERPRISE_PLUGIN_ID = 'com.gradle.enterprise' |  | ||||||
| def GRADLE_ENTERPRISE_PLUGIN_CLASS = 'com.gradle.enterprise.gradleplugin.GradleEnterprisePlugin' |  | ||||||
| def GRADLE_ENTERPRISE_EXTENSION_CLASS = 'com.gradle.enterprise.gradleplugin.GradleEnterpriseExtension' |  | ||||||
| def CI_AUTO_INJECTION_CUSTOM_VALUE_NAME = 'CI auto injection' |  | ||||||
| def CI_AUTO_INJECTION_CUSTOM_VALUE_VALUE = 'gradle-build-action' |  | ||||||
| def CCUD_PLUGIN_ID = 'com.gradle.common-custom-user-data-gradle-plugin' |  | ||||||
| def CCUD_PLUGIN_CLASS = 'com.gradle.CommonCustomUserDataGradlePlugin' |  | ||||||
|  |  | ||||||
| def isTopLevelBuild = !gradle.parent |  | ||||||
| if (!isTopLevelBuild) { |  | ||||||
|     return |  | ||||||
| } |  | ||||||
|  |  | ||||||
| def getInputParam = { String name -> |  | ||||||
|     def envVarName = name.toUpperCase().replace('.', '_').replace('-', '_') |  | ||||||
|     return System.getProperty(name) ?: System.getenv(envVarName) |  | ||||||
| } |  | ||||||
|  |  | ||||||
| // finish early if injection is disabled |  | ||||||
| def gradleInjectionEnabled = getInputParam("gradle-enterprise.injection-enabled") |  | ||||||
| if (gradleInjectionEnabled != "true") { |  | ||||||
|     return |  | ||||||
| } |  | ||||||
|  |  | ||||||
| def geUrl = getInputParam('gradle-enterprise.url') |  | ||||||
| def geAllowUntrustedServer = Boolean.parseBoolean(getInputParam('gradle-enterprise.allow-untrusted-server')) |  | ||||||
| def geEnforceUrl = Boolean.parseBoolean(getInputParam('gradle-enterprise.enforce-url')) |  | ||||||
| def buildScanUploadInBackground = Boolean.parseBoolean(getInputParam('gradle-enterprise.build-scan.upload-in-background')) |  | ||||||
| def gePluginVersion = getInputParam('gradle-enterprise.plugin.version') |  | ||||||
| def ccudPluginVersion = getInputParam('gradle-enterprise.ccud-plugin.version') |  | ||||||
|  |  | ||||||
| def atLeastGradle4 = GradleVersion.current() >= GradleVersion.version('4.0') |  | ||||||
|  |  | ||||||
| // finish early if configuration parameters passed in via system properties are not valid/supported |  | ||||||
| if (ccudPluginVersion && isNotAtLeast(ccudPluginVersion, '1.7')) { |  | ||||||
|     logger.warn("Common Custom User Data Gradle plugin must be at least 1.7. Configured version is $ccudPluginVersion.") |  | ||||||
|     return |  | ||||||
| } |  | ||||||
|  |  | ||||||
| // register buildScanPublished listener and optionally apply the GE / Build Scan plugin |  | ||||||
| if (GradleVersion.current() < GradleVersion.version('6.0')) { |  | ||||||
|     rootProject { |  | ||||||
|         buildscript.configurations.getByName("classpath").incoming.afterResolve { ResolvableDependencies incoming -> |  | ||||||
|             def resolutionResult = incoming.resolutionResult |  | ||||||
|  |  | ||||||
|             if (gePluginVersion) { |  | ||||||
|                 def scanPluginComponent = resolutionResult.allComponents.find { |  | ||||||
|                     it.moduleVersion.with { group == "com.gradle" && (name == "build-scan-plugin" || name == "gradle-enterprise-gradle-plugin") } |  | ||||||
|                 } |  | ||||||
|                 if (!scanPluginComponent) { |  | ||||||
|                     logger.quiet("Applying $BUILD_SCAN_PLUGIN_CLASS via init script") |  | ||||||
|                     logger.quiet("Connection to Gradle Enterprise: $geUrl, allowUntrustedServer: $geAllowUntrustedServer") |  | ||||||
|                     applyPluginExternally(pluginManager, BUILD_SCAN_PLUGIN_CLASS) |  | ||||||
|                     buildScan.server = geUrl |  | ||||||
|                     buildScan.allowUntrustedServer = geAllowUntrustedServer |  | ||||||
|                     buildScan.publishAlways() |  | ||||||
|                     if (buildScan.metaClass.respondsTo(buildScan, 'setUploadInBackground', Boolean)) buildScan.uploadInBackground = buildScanUploadInBackground  // uploadInBackground not available for build-scan-plugin 1.16 |  | ||||||
|                     buildScan.value CI_AUTO_INJECTION_CUSTOM_VALUE_NAME, CI_AUTO_INJECTION_CUSTOM_VALUE_VALUE |  | ||||||
|                 } |  | ||||||
|  |  | ||||||
|                 if (geUrl && geEnforceUrl) { |  | ||||||
|                     pluginManager.withPlugin(BUILD_SCAN_PLUGIN_ID) { |  | ||||||
|                         afterEvaluate { |  | ||||||
|                             logger.quiet("Enforcing Gradle Enterprise: $geUrl, allowUntrustedServer: $geAllowUntrustedServer") |  | ||||||
|                             buildScan.server = geUrl |  | ||||||
|                             buildScan.allowUntrustedServer = geAllowUntrustedServer |  | ||||||
|                         } |  | ||||||
|                     } |  | ||||||
|                 } |  | ||||||
|             } |  | ||||||
|  |  | ||||||
|             if (ccudPluginVersion && atLeastGradle4) { |  | ||||||
|                 def ccudPluginComponent = resolutionResult.allComponents.find { |  | ||||||
|                     it.moduleVersion.with { group == "com.gradle" && name == "common-custom-user-data-gradle-plugin" } |  | ||||||
|                 } |  | ||||||
|                 if (!ccudPluginComponent) { |  | ||||||
|                     logger.quiet("Applying $CCUD_PLUGIN_CLASS via init script") |  | ||||||
|                     pluginManager.apply(initscript.classLoader.loadClass(CCUD_PLUGIN_CLASS)) |  | ||||||
|                 } |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| } else { |  | ||||||
|     gradle.settingsEvaluated { settings -> |  | ||||||
|         if (gePluginVersion) { |  | ||||||
|             if (!settings.pluginManager.hasPlugin(GRADLE_ENTERPRISE_PLUGIN_ID)) { |  | ||||||
|                 logger.quiet("Applying $GRADLE_ENTERPRISE_PLUGIN_CLASS via init script") |  | ||||||
|                 logger.quiet("Connection to Gradle Enterprise: $geUrl, allowUntrustedServer: $geAllowUntrustedServer") |  | ||||||
|                 applyPluginExternally(settings.pluginManager, GRADLE_ENTERPRISE_PLUGIN_CLASS) |  | ||||||
|                 extensionsWithPublicType(settings, GRADLE_ENTERPRISE_EXTENSION_CLASS).collect { settings[it.name] }.each { ext -> |  | ||||||
|                     ext.server = geUrl |  | ||||||
|                     ext.allowUntrustedServer = geAllowUntrustedServer |  | ||||||
|                     ext.buildScan.publishAlways() |  | ||||||
|                     ext.buildScan.uploadInBackground = buildScanUploadInBackground |  | ||||||
|                     ext.buildScan.value CI_AUTO_INJECTION_CUSTOM_VALUE_NAME, CI_AUTO_INJECTION_CUSTOM_VALUE_VALUE |  | ||||||
|                 } |  | ||||||
|             } |  | ||||||
|  |  | ||||||
|             if (geUrl && geEnforceUrl) { |  | ||||||
|                 extensionsWithPublicType(settings, GRADLE_ENTERPRISE_EXTENSION_CLASS).collect { settings[it.name] }.each { ext -> |  | ||||||
|                     logger.quiet("Enforcing Gradle Enterprise: $geUrl, allowUntrustedServer: $geAllowUntrustedServer") |  | ||||||
|                     ext.server = geUrl |  | ||||||
|                     ext.allowUntrustedServer = geAllowUntrustedServer |  | ||||||
|                 } |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
|  |  | ||||||
|         if (ccudPluginVersion) { |  | ||||||
|             if (!settings.pluginManager.hasPlugin(CCUD_PLUGIN_ID)) { |  | ||||||
|                 logger.quiet("Applying $CCUD_PLUGIN_CLASS via init script") |  | ||||||
|                 settings.pluginManager.apply(initscript.classLoader.loadClass(CCUD_PLUGIN_CLASS)) |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| } |  | ||||||
|  |  | ||||||
| void applyPluginExternally(def pluginManager, String pluginClassName) { |  | ||||||
|     def externallyApplied = 'gradle.enterprise.externally-applied' |  | ||||||
|     def oldValue = System.getProperty(externallyApplied) |  | ||||||
|     System.setProperty(externallyApplied, 'true') |  | ||||||
|     try { |  | ||||||
|         pluginManager.apply(initscript.classLoader.loadClass(pluginClassName)) |  | ||||||
|     } finally { |  | ||||||
|         if (oldValue == null) { |  | ||||||
|             System.clearProperty(externallyApplied) |  | ||||||
|         } else { |  | ||||||
|             System.setProperty(externallyApplied, oldValue) |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| } |  | ||||||
|  |  | ||||||
| static def extensionsWithPublicType(def container, String publicType) { |  | ||||||
|     container.extensions.extensionsSchema.elements.findAll { it.publicType.concreteClass.name == publicType } |  | ||||||
| } |  | ||||||
|  |  | ||||||
| static boolean isNotAtLeast(String versionUnderTest, String referenceVersion) { |  | ||||||
|     GradleVersion.version(versionUnderTest) < GradleVersion.version(referenceVersion) |  | ||||||
| } |  | ||||||
| @@ -1,102 +0,0 @@ | |||||||
| import * as core from '@actions/core' |  | ||||||
| import * as exec from '@actions/exec' |  | ||||||
| import {SUMMARY_ENV_VAR} from '@actions/core/lib/summary' |  | ||||||
| import * as path from 'path' |  | ||||||
| import * as os from 'os' |  | ||||||
| import * as caches from './caches' |  | ||||||
| import * as layout from './repository-layout' |  | ||||||
| import * as params from './input-params' |  | ||||||
| import * as dependencyGraph from './dependency-graph' |  | ||||||
|  |  | ||||||
| import {logJobSummary, writeJobSummary} from './job-summary' |  | ||||||
| import {loadBuildResults} from './build-results' |  | ||||||
| import {CacheListener} from './cache-reporting' |  | ||||||
| import {DaemonController} from './daemon-controller' |  | ||||||
|  |  | ||||||
| const GRADLE_SETUP_VAR = 'GRADLE_BUILD_ACTION_SETUP_COMPLETED' |  | ||||||
| const GRADLE_USER_HOME = 'GRADLE_USER_HOME' |  | ||||||
| const CACHE_LISTENER = 'CACHE_LISTENER' |  | ||||||
|  |  | ||||||
| export async function setup(): Promise<void> { |  | ||||||
|     const gradleUserHome = await determineGradleUserHome() |  | ||||||
|  |  | ||||||
|     // Bypass setup on all but first action step in workflow. |  | ||||||
|     if (process.env[GRADLE_SETUP_VAR]) { |  | ||||||
|         core.info('Gradle setup only performed on first gradle-build-action step in workflow.') |  | ||||||
|         return |  | ||||||
|     } |  | ||||||
|     // Record setup complete: visible to all subsequent actions and prevents duplicate setup |  | ||||||
|     core.exportVariable(GRADLE_SETUP_VAR, true) |  | ||||||
|     // Record setup complete: visible in post-action, to control action completion |  | ||||||
|     core.saveState(GRADLE_SETUP_VAR, true) |  | ||||||
|  |  | ||||||
|     // Save the Gradle User Home for use in the post-action step. |  | ||||||
|     core.saveState(GRADLE_USER_HOME, gradleUserHome) |  | ||||||
|  |  | ||||||
|     const cacheListener = new CacheListener() |  | ||||||
|     await caches.restore(gradleUserHome, cacheListener) |  | ||||||
|  |  | ||||||
|     core.saveState(CACHE_LISTENER, cacheListener.stringify()) |  | ||||||
|  |  | ||||||
|     await dependencyGraph.setup(params.getDependencyGraphOption()) |  | ||||||
| } |  | ||||||
|  |  | ||||||
| export async function complete(): Promise<void> { |  | ||||||
|     if (!core.getState(GRADLE_SETUP_VAR)) { |  | ||||||
|         core.info('Gradle setup post-action only performed for first gradle-build-action step in workflow.') |  | ||||||
|         return |  | ||||||
|     } |  | ||||||
|     core.info('In final post-action step, saving state and writing summary') |  | ||||||
|  |  | ||||||
|     const buildResults = loadBuildResults() |  | ||||||
|  |  | ||||||
|     const gradleUserHome = core.getState(GRADLE_USER_HOME) |  | ||||||
|     const cacheListener: CacheListener = CacheListener.rehydrate(core.getState(CACHE_LISTENER)) |  | ||||||
|     const daemonController = new DaemonController(buildResults) |  | ||||||
|  |  | ||||||
|     await caches.save(gradleUserHome, cacheListener, daemonController) |  | ||||||
|  |  | ||||||
|     if (shouldGenerateJobSummary()) { |  | ||||||
|         await writeJobSummary(buildResults, cacheListener) |  | ||||||
|     } else { |  | ||||||
|         logJobSummary(buildResults, cacheListener) |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     await dependencyGraph.complete(params.getDependencyGraphOption()) |  | ||||||
| } |  | ||||||
|  |  | ||||||
| async function determineGradleUserHome(): Promise<string> { |  | ||||||
|     const customGradleUserHome = process.env['GRADLE_USER_HOME'] |  | ||||||
|     if (customGradleUserHome) { |  | ||||||
|         const rootDir = layout.workspaceDirectory() |  | ||||||
|         return path.resolve(rootDir, customGradleUserHome) |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     return path.resolve(await determineUserHome(), '.gradle') |  | ||||||
| } |  | ||||||
|  |  | ||||||
| /** |  | ||||||
|  * Different values can be returned by os.homedir() in Javascript and System.getProperty('user.home') in Java. |  | ||||||
|  * In order to determine the correct Gradle User Home, we ask Java for the user home instead of using os.homedir(). |  | ||||||
|  */ |  | ||||||
| async function determineUserHome(): Promise<string> { |  | ||||||
|     const output = await exec.getExecOutput('java', ['-XshowSettings:properties', '-version'], {silent: true}) |  | ||||||
|     const regex = /user\.home = (\S*)/i |  | ||||||
|     const found = output.stderr.match(regex) |  | ||||||
|     if (found == null || found.length <= 1) { |  | ||||||
|         core.info('Could not determine user.home from java -version output. Using os.homedir().') |  | ||||||
|         return os.homedir() |  | ||||||
|     } |  | ||||||
|     const userHome = found[1] |  | ||||||
|     core.debug(`Determined user.home from java -version output: '${userHome}'`) |  | ||||||
|     return userHome |  | ||||||
| } |  | ||||||
|  |  | ||||||
| function shouldGenerateJobSummary(): boolean { |  | ||||||
|     // Check if Job Summary is supported on this platform |  | ||||||
|     if (!process.env[SUMMARY_ENV_VAR]) { |  | ||||||
|         return false |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     return params.isJobSummaryEnabled() |  | ||||||
| } |  | ||||||
							
								
								
									
										2
									
								
								test/init-scripts/.gitignore
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										2
									
								
								test/init-scripts/.gitignore
									
									
									
									
										vendored
									
									
								
							| @@ -1,2 +0,0 @@ | |||||||
| build |  | ||||||
| .gradle |  | ||||||
| @@ -1,28 +0,0 @@ | |||||||
| plugins { |  | ||||||
|     id 'groovy' |  | ||||||
| } |  | ||||||
|  |  | ||||||
| java { |  | ||||||
|     toolchain { |  | ||||||
|         languageVersion = JavaLanguageVersion.of(8) |  | ||||||
|     } |  | ||||||
| } |  | ||||||
|  |  | ||||||
| repositories { |  | ||||||
|     mavenCentral() |  | ||||||
| } |  | ||||||
|  |  | ||||||
| dependencies { |  | ||||||
|     testImplementation gradleTestKit() |  | ||||||
|     testImplementation 'org.spockframework:spock-core:2.3-groovy-3.0' |  | ||||||
|     testImplementation('org.spockframework:spock-junit4:2.3-groovy-3.0') |  | ||||||
|  |  | ||||||
|     testImplementation ('io.ratpack:ratpack-groovy-test:1.9.0') { |  | ||||||
|         exclude group: 'org.codehaus.groovy', module: 'groovy-all' |  | ||||||
|     } |  | ||||||
|     testImplementation 'com.fasterxml.jackson.dataformat:jackson-dataformat-smile:2.16.0' |  | ||||||
| } |  | ||||||
|  |  | ||||||
| test { |  | ||||||
|     useJUnitPlatform() |  | ||||||
| } |  | ||||||
							
								
								
									
										
											BIN
										
									
								
								test/init-scripts/gradle/wrapper/gradle-wrapper.jar
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										
											BIN
										
									
								
								test/init-scripts/gradle/wrapper/gradle-wrapper.jar
									
									
									
									
										vendored
									
									
								
							
										
											Binary file not shown.
										
									
								
							| @@ -1,8 +0,0 @@ | |||||||
| distributionBase=GRADLE_USER_HOME |  | ||||||
| distributionPath=wrapper/dists |  | ||||||
| distributionSha256Sum=591855b517fc635b9e04de1d05d5e76ada3f89f5fc76f87978d1b245b4f69225 |  | ||||||
| distributionUrl=https\://services.gradle.org/distributions/gradle-8.3-bin.zip |  | ||||||
| networkTimeout=10000 |  | ||||||
| validateDistributionUrl=true |  | ||||||
| zipStoreBase=GRADLE_USER_HOME |  | ||||||
| zipStorePath=wrapper/dists |  | ||||||
							
								
								
									
										249
									
								
								test/init-scripts/gradlew
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										249
									
								
								test/init-scripts/gradlew
									
									
									
									
										vendored
									
									
								
							| @@ -1,249 +0,0 @@ | |||||||
| #!/bin/sh |  | ||||||
|  |  | ||||||
| # |  | ||||||
| # Copyright © 2015-2021 the original authors. |  | ||||||
| # |  | ||||||
| # Licensed under the Apache License, Version 2.0 (the "License"); |  | ||||||
| # you may not use this file except in compliance with the License. |  | ||||||
| # You may obtain a copy of the License at |  | ||||||
| # |  | ||||||
| #      https://www.apache.org/licenses/LICENSE-2.0 |  | ||||||
| # |  | ||||||
| # Unless required by applicable law or agreed to in writing, software |  | ||||||
| # distributed under the License is distributed on an "AS IS" BASIS, |  | ||||||
| # WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. |  | ||||||
| # See the License for the specific language governing permissions and |  | ||||||
| # limitations under the License. |  | ||||||
| # |  | ||||||
|  |  | ||||||
| ############################################################################## |  | ||||||
| # |  | ||||||
| #   Gradle start up script for POSIX generated by Gradle. |  | ||||||
| # |  | ||||||
| #   Important for running: |  | ||||||
| # |  | ||||||
| #   (1) You need a POSIX-compliant shell to run this script. If your /bin/sh is |  | ||||||
| #       noncompliant, but you have some other compliant shell such as ksh or |  | ||||||
| #       bash, then to run this script, type that shell name before the whole |  | ||||||
| #       command line, like: |  | ||||||
| # |  | ||||||
| #           ksh Gradle |  | ||||||
| # |  | ||||||
| #       Busybox and similar reduced shells will NOT work, because this script |  | ||||||
| #       requires all of these POSIX shell features: |  | ||||||
| #         * functions; |  | ||||||
| #         * expansions «$var», «${var}», «${var:-default}», «${var+SET}», |  | ||||||
| #           «${var#prefix}», «${var%suffix}», and «$( cmd )»; |  | ||||||
| #         * compound commands having a testable exit status, especially «case»; |  | ||||||
| #         * various built-in commands including «command», «set», and «ulimit». |  | ||||||
| # |  | ||||||
| #   Important for patching: |  | ||||||
| # |  | ||||||
| #   (2) This script targets any POSIX shell, so it avoids extensions provided |  | ||||||
| #       by Bash, Ksh, etc; in particular arrays are avoided. |  | ||||||
| # |  | ||||||
| #       The "traditional" practice of packing multiple parameters into a |  | ||||||
| #       space-separated string is a well documented source of bugs and security |  | ||||||
| #       problems, so this is (mostly) avoided, by progressively accumulating |  | ||||||
| #       options in "$@", and eventually passing that to Java. |  | ||||||
| # |  | ||||||
| #       Where the inherited environment variables (DEFAULT_JVM_OPTS, JAVA_OPTS, |  | ||||||
| #       and GRADLE_OPTS) rely on word-splitting, this is performed explicitly; |  | ||||||
| #       see the in-line comments for details. |  | ||||||
| # |  | ||||||
| #       There are tweaks for specific operating systems such as AIX, CygWin, |  | ||||||
| #       Darwin, MinGW, and NonStop. |  | ||||||
| # |  | ||||||
| #   (3) This script is generated from the Groovy template |  | ||||||
| #       https://github.com/gradle/gradle/blob/HEAD/subprojects/plugins/src/main/resources/org/gradle/api/internal/plugins/unixStartScript.txt |  | ||||||
| #       within the Gradle project. |  | ||||||
| # |  | ||||||
| #       You can find Gradle at https://github.com/gradle/gradle/. |  | ||||||
| # |  | ||||||
| ############################################################################## |  | ||||||
|  |  | ||||||
| # Attempt to set APP_HOME |  | ||||||
|  |  | ||||||
| # Resolve links: $0 may be a link |  | ||||||
| app_path=$0 |  | ||||||
|  |  | ||||||
| # Need this for daisy-chained symlinks. |  | ||||||
| while |  | ||||||
|     APP_HOME=${app_path%"${app_path##*/}"}  # leaves a trailing /; empty if no leading path |  | ||||||
|     [ -h "$app_path" ] |  | ||||||
| do |  | ||||||
|     ls=$( ls -ld "$app_path" ) |  | ||||||
|     link=${ls#*' -> '} |  | ||||||
|     case $link in             #( |  | ||||||
|       /*)   app_path=$link ;; #( |  | ||||||
|       *)    app_path=$APP_HOME$link ;; |  | ||||||
|     esac |  | ||||||
| done |  | ||||||
|  |  | ||||||
| # This is normally unused |  | ||||||
| # shellcheck disable=SC2034 |  | ||||||
| APP_BASE_NAME=${0##*/} |  | ||||||
| # Discard cd standard output in case $CDPATH is set (https://github.com/gradle/gradle/issues/25036) |  | ||||||
| APP_HOME=$( cd "${APP_HOME:-./}" > /dev/null && pwd -P ) || exit |  | ||||||
|  |  | ||||||
| # Use the maximum available, or set MAX_FD != -1 to use that value. |  | ||||||
| MAX_FD=maximum |  | ||||||
|  |  | ||||||
| warn () { |  | ||||||
|     echo "$*" |  | ||||||
| } >&2 |  | ||||||
|  |  | ||||||
| die () { |  | ||||||
|     echo |  | ||||||
|     echo "$*" |  | ||||||
|     echo |  | ||||||
|     exit 1 |  | ||||||
| } >&2 |  | ||||||
|  |  | ||||||
| # OS specific support (must be 'true' or 'false'). |  | ||||||
| cygwin=false |  | ||||||
| msys=false |  | ||||||
| darwin=false |  | ||||||
| nonstop=false |  | ||||||
| case "$( uname )" in                #( |  | ||||||
|   CYGWIN* )         cygwin=true  ;; #( |  | ||||||
|   Darwin* )         darwin=true  ;; #( |  | ||||||
|   MSYS* | MINGW* )  msys=true    ;; #( |  | ||||||
|   NONSTOP* )        nonstop=true ;; |  | ||||||
| esac |  | ||||||
|  |  | ||||||
| CLASSPATH=$APP_HOME/gradle/wrapper/gradle-wrapper.jar |  | ||||||
|  |  | ||||||
|  |  | ||||||
| # Determine the Java command to use to start the JVM. |  | ||||||
| if [ -n "$JAVA_HOME" ] ; then |  | ||||||
|     if [ -x "$JAVA_HOME/jre/sh/java" ] ; then |  | ||||||
|         # IBM's JDK on AIX uses strange locations for the executables |  | ||||||
|         JAVACMD=$JAVA_HOME/jre/sh/java |  | ||||||
|     else |  | ||||||
|         JAVACMD=$JAVA_HOME/bin/java |  | ||||||
|     fi |  | ||||||
|     if [ ! -x "$JAVACMD" ] ; then |  | ||||||
|         die "ERROR: JAVA_HOME is set to an invalid directory: $JAVA_HOME |  | ||||||
|  |  | ||||||
| Please set the JAVA_HOME variable in your environment to match the |  | ||||||
| location of your Java installation." |  | ||||||
|     fi |  | ||||||
| else |  | ||||||
|     JAVACMD=java |  | ||||||
|     if ! command -v java >/dev/null 2>&1 |  | ||||||
|     then |  | ||||||
|         die "ERROR: JAVA_HOME is not set and no 'java' command could be found in your PATH. |  | ||||||
|  |  | ||||||
| Please set the JAVA_HOME variable in your environment to match the |  | ||||||
| location of your Java installation." |  | ||||||
|     fi |  | ||||||
| fi |  | ||||||
|  |  | ||||||
| # Increase the maximum file descriptors if we can. |  | ||||||
| if ! "$cygwin" && ! "$darwin" && ! "$nonstop" ; then |  | ||||||
|     case $MAX_FD in #( |  | ||||||
|       max*) |  | ||||||
|         # In POSIX sh, ulimit -H is undefined. That's why the result is checked to see if it worked. |  | ||||||
|         # shellcheck disable=SC3045 |  | ||||||
|         MAX_FD=$( ulimit -H -n ) || |  | ||||||
|             warn "Could not query maximum file descriptor limit" |  | ||||||
|     esac |  | ||||||
|     case $MAX_FD in  #( |  | ||||||
|       '' | soft) :;; #( |  | ||||||
|       *) |  | ||||||
|         # In POSIX sh, ulimit -n is undefined. That's why the result is checked to see if it worked. |  | ||||||
|         # shellcheck disable=SC3045 |  | ||||||
|         ulimit -n "$MAX_FD" || |  | ||||||
|             warn "Could not set maximum file descriptor limit to $MAX_FD" |  | ||||||
|     esac |  | ||||||
| fi |  | ||||||
|  |  | ||||||
| # Collect all arguments for the java command, stacking in reverse order: |  | ||||||
| #   * args from the command line |  | ||||||
| #   * the main class name |  | ||||||
| #   * -classpath |  | ||||||
| #   * -D...appname settings |  | ||||||
| #   * --module-path (only if needed) |  | ||||||
| #   * DEFAULT_JVM_OPTS, JAVA_OPTS, and GRADLE_OPTS environment variables. |  | ||||||
|  |  | ||||||
| # For Cygwin or MSYS, switch paths to Windows format before running java |  | ||||||
| if "$cygwin" || "$msys" ; then |  | ||||||
|     APP_HOME=$( cygpath --path --mixed "$APP_HOME" ) |  | ||||||
|     CLASSPATH=$( cygpath --path --mixed "$CLASSPATH" ) |  | ||||||
|  |  | ||||||
|     JAVACMD=$( cygpath --unix "$JAVACMD" ) |  | ||||||
|  |  | ||||||
|     # Now convert the arguments - kludge to limit ourselves to /bin/sh |  | ||||||
|     for arg do |  | ||||||
|         if |  | ||||||
|             case $arg in                                #( |  | ||||||
|               -*)   false ;;                            # don't mess with options #( |  | ||||||
|               /?*)  t=${arg#/} t=/${t%%/*}              # looks like a POSIX filepath |  | ||||||
|                     [ -e "$t" ] ;;                      #( |  | ||||||
|               *)    false ;; |  | ||||||
|             esac |  | ||||||
|         then |  | ||||||
|             arg=$( cygpath --path --ignore --mixed "$arg" ) |  | ||||||
|         fi |  | ||||||
|         # Roll the args list around exactly as many times as the number of |  | ||||||
|         # args, so each arg winds up back in the position where it started, but |  | ||||||
|         # possibly modified. |  | ||||||
|         # |  | ||||||
|         # NB: a `for` loop captures its iteration list before it begins, so |  | ||||||
|         # changing the positional parameters here affects neither the number of |  | ||||||
|         # iterations, nor the values presented in `arg`. |  | ||||||
|         shift                   # remove old arg |  | ||||||
|         set -- "$@" "$arg"      # push replacement arg |  | ||||||
|     done |  | ||||||
| fi |  | ||||||
|  |  | ||||||
|  |  | ||||||
| # Add default JVM options here. You can also use JAVA_OPTS and GRADLE_OPTS to pass JVM options to this script. |  | ||||||
| DEFAULT_JVM_OPTS='"-Xmx64m" "-Xms64m"' |  | ||||||
|  |  | ||||||
| # Collect all arguments for the java command; |  | ||||||
| #   * $DEFAULT_JVM_OPTS, $JAVA_OPTS, and $GRADLE_OPTS can contain fragments of |  | ||||||
| #     shell script including quotes and variable substitutions, so put them in |  | ||||||
| #     double quotes to make sure that they get re-expanded; and |  | ||||||
| #   * put everything else in single quotes, so that it's not re-expanded. |  | ||||||
|  |  | ||||||
| set -- \ |  | ||||||
|         "-Dorg.gradle.appname=$APP_BASE_NAME" \ |  | ||||||
|         -classpath "$CLASSPATH" \ |  | ||||||
|         org.gradle.wrapper.GradleWrapperMain \ |  | ||||||
|         "$@" |  | ||||||
|  |  | ||||||
| # Stop when "xargs" is not available. |  | ||||||
| if ! command -v xargs >/dev/null 2>&1 |  | ||||||
| then |  | ||||||
|     die "xargs is not available" |  | ||||||
| fi |  | ||||||
|  |  | ||||||
| # Use "xargs" to parse quoted args. |  | ||||||
| # |  | ||||||
| # With -n1 it outputs one arg per line, with the quotes and backslashes removed. |  | ||||||
| # |  | ||||||
| # In Bash we could simply go: |  | ||||||
| # |  | ||||||
| #   readarray ARGS < <( xargs -n1 <<<"$var" ) && |  | ||||||
| #   set -- "${ARGS[@]}" "$@" |  | ||||||
| # |  | ||||||
| # but POSIX shell has neither arrays nor command substitution, so instead we |  | ||||||
| # post-process each arg (as a line of input to sed) to backslash-escape any |  | ||||||
| # character that might be a shell metacharacter, then use eval to reverse |  | ||||||
| # that process (while maintaining the separation between arguments), and wrap |  | ||||||
| # the whole thing up as a single "set" statement. |  | ||||||
| # |  | ||||||
| # This will of course break if any of these variables contains a newline or |  | ||||||
| # an unmatched quote. |  | ||||||
| # |  | ||||||
|  |  | ||||||
| eval "set -- $( |  | ||||||
|         printf '%s\n' "$DEFAULT_JVM_OPTS $JAVA_OPTS $GRADLE_OPTS" | |  | ||||||
|         xargs -n1 | |  | ||||||
|         sed ' s~[^-[:alnum:]+,./:=@_]~\\&~g; ' | |  | ||||||
|         tr '\n' ' ' |  | ||||||
|     )" '"$@"' |  | ||||||
|  |  | ||||||
| exec "$JAVACMD" "$@" |  | ||||||
							
								
								
									
										92
									
								
								test/init-scripts/gradlew.bat
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										92
									
								
								test/init-scripts/gradlew.bat
									
									
									
									
										vendored
									
									
								
							| @@ -1,92 +0,0 @@ | |||||||
| @rem |  | ||||||
| @rem Copyright 2015 the original author or authors. |  | ||||||
| @rem |  | ||||||
| @rem Licensed under the Apache License, Version 2.0 (the "License"); |  | ||||||
| @rem you may not use this file except in compliance with the License. |  | ||||||
| @rem You may obtain a copy of the License at |  | ||||||
| @rem |  | ||||||
| @rem      https://www.apache.org/licenses/LICENSE-2.0 |  | ||||||
| @rem |  | ||||||
| @rem Unless required by applicable law or agreed to in writing, software |  | ||||||
| @rem distributed under the License is distributed on an "AS IS" BASIS, |  | ||||||
| @rem WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. |  | ||||||
| @rem See the License for the specific language governing permissions and |  | ||||||
| @rem limitations under the License. |  | ||||||
| @rem |  | ||||||
|  |  | ||||||
| @if "%DEBUG%"=="" @echo off |  | ||||||
| @rem ########################################################################## |  | ||||||
| @rem |  | ||||||
| @rem  Gradle startup script for Windows |  | ||||||
| @rem |  | ||||||
| @rem ########################################################################## |  | ||||||
|  |  | ||||||
| @rem Set local scope for the variables with windows NT shell |  | ||||||
| if "%OS%"=="Windows_NT" setlocal |  | ||||||
|  |  | ||||||
| set DIRNAME=%~dp0 |  | ||||||
| if "%DIRNAME%"=="" set DIRNAME=. |  | ||||||
| @rem This is normally unused |  | ||||||
| set APP_BASE_NAME=%~n0 |  | ||||||
| set APP_HOME=%DIRNAME% |  | ||||||
|  |  | ||||||
| @rem Resolve any "." and ".." in APP_HOME to make it shorter. |  | ||||||
| for %%i in ("%APP_HOME%") do set APP_HOME=%%~fi |  | ||||||
|  |  | ||||||
| @rem Add default JVM options here. You can also use JAVA_OPTS and GRADLE_OPTS to pass JVM options to this script. |  | ||||||
| set DEFAULT_JVM_OPTS="-Xmx64m" "-Xms64m" |  | ||||||
|  |  | ||||||
| @rem Find java.exe |  | ||||||
| if defined JAVA_HOME goto findJavaFromJavaHome |  | ||||||
|  |  | ||||||
| set JAVA_EXE=java.exe |  | ||||||
| %JAVA_EXE% -version >NUL 2>&1 |  | ||||||
| if %ERRORLEVEL% equ 0 goto execute |  | ||||||
|  |  | ||||||
| echo. |  | ||||||
| echo ERROR: JAVA_HOME is not set and no 'java' command could be found in your PATH. |  | ||||||
| echo. |  | ||||||
| echo Please set the JAVA_HOME variable in your environment to match the |  | ||||||
| echo location of your Java installation. |  | ||||||
|  |  | ||||||
| goto fail |  | ||||||
|  |  | ||||||
| :findJavaFromJavaHome |  | ||||||
| set JAVA_HOME=%JAVA_HOME:"=% |  | ||||||
| set JAVA_EXE=%JAVA_HOME%/bin/java.exe |  | ||||||
|  |  | ||||||
| if exist "%JAVA_EXE%" goto execute |  | ||||||
|  |  | ||||||
| echo. |  | ||||||
| echo ERROR: JAVA_HOME is set to an invalid directory: %JAVA_HOME% |  | ||||||
| echo. |  | ||||||
| echo Please set the JAVA_HOME variable in your environment to match the |  | ||||||
| echo location of your Java installation. |  | ||||||
|  |  | ||||||
| goto fail |  | ||||||
|  |  | ||||||
| :execute |  | ||||||
| @rem Setup the command line |  | ||||||
|  |  | ||||||
| set CLASSPATH=%APP_HOME%\gradle\wrapper\gradle-wrapper.jar |  | ||||||
|  |  | ||||||
|  |  | ||||||
| @rem Execute Gradle |  | ||||||
| "%JAVA_EXE%" %DEFAULT_JVM_OPTS% %JAVA_OPTS% %GRADLE_OPTS% "-Dorg.gradle.appname=%APP_BASE_NAME%" -classpath "%CLASSPATH%" org.gradle.wrapper.GradleWrapperMain %* |  | ||||||
|  |  | ||||||
| :end |  | ||||||
| @rem End local scope for the variables with windows NT shell |  | ||||||
| if %ERRORLEVEL% equ 0 goto mainEnd |  | ||||||
|  |  | ||||||
| :fail |  | ||||||
| rem Set variable GRADLE_EXIT_CONSOLE if you need the _script_ return code instead of |  | ||||||
| rem the _cmd.exe /c_ return code! |  | ||||||
| set EXIT_CODE=%ERRORLEVEL% |  | ||||||
| if %EXIT_CODE% equ 0 set EXIT_CODE=1 |  | ||||||
| if not ""=="%GRADLE_EXIT_CONSOLE%" exit %EXIT_CODE% |  | ||||||
| exit /b %EXIT_CODE% |  | ||||||
|  |  | ||||||
| :mainEnd |  | ||||||
| if "%OS%"=="Windows_NT" endlocal |  | ||||||
|  |  | ||||||
| :omega |  | ||||||
| @@ -1,14 +0,0 @@ | |||||||
| plugins { |  | ||||||
|     id "com.gradle.enterprise" version "3.15.1" |  | ||||||
|     id "com.gradle.common-custom-user-data-gradle-plugin" version "1.12" |  | ||||||
| } |  | ||||||
|  |  | ||||||
| gradleEnterprise { |  | ||||||
|     buildScan { |  | ||||||
|         termsOfServiceUrl = "https://gradle.com/terms-of-service" |  | ||||||
|         termsOfServiceAgree = "yes" |  | ||||||
|         publishAlways() |  | ||||||
|         uploadInBackground = false |  | ||||||
|     } |  | ||||||
| } |  | ||||||
| rootProject.name = 'test-init-scripts' |  | ||||||
| @@ -1,261 +0,0 @@ | |||||||
| package com.gradle.gradlebuildaction |  | ||||||
|  |  | ||||||
| import com.fasterxml.jackson.core.JsonFactory |  | ||||||
| import com.fasterxml.jackson.databind.ObjectMapper |  | ||||||
| import com.fasterxml.jackson.dataformat.smile.SmileFactory |  | ||||||
| import org.gradle.testkit.runner.BuildResult |  | ||||||
| import org.gradle.testkit.runner.GradleRunner |  | ||||||
| import org.gradle.testkit.runner.internal.DefaultGradleRunner |  | ||||||
| import org.gradle.util.GradleVersion |  | ||||||
| import ratpack.groovy.test.embed.GroovyEmbeddedApp |  | ||||||
| import spock.lang.AutoCleanup |  | ||||||
| import spock.lang.Specification |  | ||||||
| import spock.lang.TempDir |  | ||||||
|  |  | ||||||
| import java.nio.file.Files |  | ||||||
| import java.util.zip.GZIPOutputStream |  | ||||||
|  |  | ||||||
| class BaseInitScriptTest extends Specification { |  | ||||||
|     static final String GE_PLUGIN_VERSION = '3.15.1' |  | ||||||
|     static final String CCUD_PLUGIN_VERSION = '1.12' |  | ||||||
|  |  | ||||||
|     static final TestGradleVersion GRADLE_3_X = new TestGradleVersion(GradleVersion.version('3.5.1'), 7, 9) |  | ||||||
|     static final TestGradleVersion GRADLE_4_X = new TestGradleVersion(GradleVersion.version('4.10.3'), 7, 10) |  | ||||||
|     static final TestGradleVersion GRADLE_5_X = new TestGradleVersion(GradleVersion.version('5.6.4'), 8, 12) |  | ||||||
|     static final TestGradleVersion GRADLE_6_NO_BUILD_SERVICE = new TestGradleVersion(GradleVersion.version('6.5.1'), 8, 14) |  | ||||||
|     static final TestGradleVersion GRADLE_6_X = new TestGradleVersion(GradleVersion.version('6.9.4'), 8, 15) |  | ||||||
|     static final TestGradleVersion GRADLE_7_X = new TestGradleVersion(GradleVersion.version('7.6.2'), 8, 19) |  | ||||||
|     static final TestGradleVersion GRADLE_8_0 = new TestGradleVersion(GradleVersion.version('8.0.2'), 8, 19) |  | ||||||
|     static final TestGradleVersion GRADLE_8_X = new TestGradleVersion(GradleVersion.version('8.5'), 8, 19) |  | ||||||
|  |  | ||||||
|     static final List<TestGradleVersion> ALL_VERSIONS = [ |  | ||||||
|         GRADLE_3_X, // First version where TestKit supports environment variables |  | ||||||
|         GRADLE_4_X, |  | ||||||
|         GRADLE_5_X, |  | ||||||
|         GRADLE_6_NO_BUILD_SERVICE, // Last version without build service support |  | ||||||
|         GRADLE_6_X, |  | ||||||
|         GRADLE_7_X, |  | ||||||
|         GRADLE_8_0, |  | ||||||
|         GRADLE_8_X, |  | ||||||
|     ] |  | ||||||
|  |  | ||||||
|     static final List<TestGradleVersion> CONFIGURATION_CACHE_VERSIONS = |  | ||||||
|         [GRADLE_7_X, GRADLE_8_0, GRADLE_8_X] |  | ||||||
|  |  | ||||||
|     static final List<TestGradleVersion> SETTINGS_PLUGIN_VERSIONS = |  | ||||||
|         [GRADLE_6_X, GRADLE_7_X, GRADLE_8_0, GRADLE_8_X] |  | ||||||
|  |  | ||||||
|     static final String PUBLIC_BUILD_SCAN_ID = 'i2wepy2gr7ovw' |  | ||||||
|     static final String DEFAULT_SCAN_UPLOAD_TOKEN = 'scan-upload-token' |  | ||||||
|     static final String ROOT_PROJECT_NAME = 'test-init-script' |  | ||||||
|     boolean failScanUpload = false |  | ||||||
|  |  | ||||||
|     File settingsFile |  | ||||||
|     File buildFile |  | ||||||
|  |  | ||||||
|     @TempDir |  | ||||||
|     File testProjectDir |  | ||||||
|  |  | ||||||
|     @AutoCleanup |  | ||||||
|     def mockScansServer = GroovyEmbeddedApp.of { |  | ||||||
|         def jsonWriter = new ObjectMapper(new JsonFactory()).writer() |  | ||||||
|         def smileWriter = new ObjectMapper(new SmileFactory()).writer() |  | ||||||
|  |  | ||||||
|         handlers { |  | ||||||
|             post('in/:gradleVersion/:pluginVersion') { |  | ||||||
|                 if (failScanUpload) { |  | ||||||
|                     context.response.status(401).send() |  | ||||||
|                     return |  | ||||||
|                 } |  | ||||||
|                 def scanUrlString = "${mockScansServer.address}s/$PUBLIC_BUILD_SCAN_ID" |  | ||||||
|                 def body = [ |  | ||||||
|                     id     : PUBLIC_BUILD_SCAN_ID, |  | ||||||
|                     scanUrl: scanUrlString.toString(), |  | ||||||
|                 ] |  | ||||||
|                 def out = new ByteArrayOutputStream() |  | ||||||
|                 new GZIPOutputStream(out).withStream { smileWriter.writeValue(it, body) } |  | ||||||
|                 context.response |  | ||||||
|                     .contentType('application/vnd.gradle.scan-ack') |  | ||||||
|                     .send(out.toByteArray()) |  | ||||||
|             } |  | ||||||
|             prefix('scans/publish') { |  | ||||||
|                 post('gradle/:pluginVersion/token') { |  | ||||||
|                     if (failScanUpload) { |  | ||||||
|                         context.response.status(401).send() |  | ||||||
|                         return |  | ||||||
|                     } |  | ||||||
|                     def pluginVersion = context.pathTokens.pluginVersion |  | ||||||
|                     def scanUrlString = "${mockScansServer.address}s/$PUBLIC_BUILD_SCAN_ID" |  | ||||||
|                     def body = [ |  | ||||||
|                         id             : PUBLIC_BUILD_SCAN_ID, |  | ||||||
|                         scanUrl        : scanUrlString.toString(), |  | ||||||
|                         scanUploadUrl  : "${mockScansServer.address.toString()}scans/publish/gradle/$pluginVersion/upload".toString(), |  | ||||||
|                         scanUploadToken: DEFAULT_SCAN_UPLOAD_TOKEN |  | ||||||
|                     ] |  | ||||||
|                     context.response |  | ||||||
|                         .contentType('application/vnd.gradle.scan-ack+json') |  | ||||||
|                         .send(jsonWriter.writeValueAsBytes(body)) |  | ||||||
|                 } |  | ||||||
|                 post('gradle/:pluginVersion/upload') { |  | ||||||
|                     if (failScanUpload) { |  | ||||||
|                         context.response.status(401).send() |  | ||||||
|                         return |  | ||||||
|                     } |  | ||||||
|                     context.request.getBody(1024 * 1024 * 10).then { |  | ||||||
|                         context.response |  | ||||||
|                             .contentType('application/vnd.gradle.scan-upload-ack+json') |  | ||||||
|                             .send() |  | ||||||
|                     } |  | ||||||
|                 } |  | ||||||
|                 notFound() |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     def setup() { |  | ||||||
|         settingsFile = new File(testProjectDir, 'settings.gradle') |  | ||||||
|         buildFile = new File(testProjectDir, 'build.gradle') |  | ||||||
|  |  | ||||||
|         File srcInitScriptsDir = new File("../../src/resources/init-scripts") |  | ||||||
|         File targetInitScriptsDir = new File(testProjectDir, "initScripts") |  | ||||||
|         targetInitScriptsDir.mkdirs() |  | ||||||
|  |  | ||||||
|         for (File srcInitScript : srcInitScriptsDir.listFiles()) { |  | ||||||
|             File targetInitScript = new File(targetInitScriptsDir, srcInitScript.name) |  | ||||||
|             Files.copy(srcInitScript.toPath(), targetInitScript.toPath()) |  | ||||||
|         } |  | ||||||
|         settingsFile << "rootProject.name = '${ROOT_PROJECT_NAME}'\n" |  | ||||||
|         buildFile << '' |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     def declareGePluginApplication(GradleVersion gradleVersion, URI serverUrl = mockScansServer.address) { |  | ||||||
|         settingsFile.text = maybeAddPluginsToSettings(gradleVersion, null, serverUrl) + settingsFile.text |  | ||||||
|         buildFile.text = maybeAddPluginsToRootProject(gradleVersion, null, serverUrl) + buildFile.text |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     def declareGePluginAndCcudPluginApplication(GradleVersion gradleVersion, URI serverUrl = mockScansServer.address) { |  | ||||||
|         settingsFile.text = maybeAddPluginsToSettings(gradleVersion, CCUD_PLUGIN_VERSION, serverUrl) + settingsFile.text |  | ||||||
|         buildFile.text = maybeAddPluginsToRootProject(gradleVersion, CCUD_PLUGIN_VERSION, serverUrl) + buildFile.text |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     String maybeAddPluginsToSettings(GradleVersion gradleVersion, String ccudPluginVersion, URI serverUri) { |  | ||||||
|         if (gradleVersion < GradleVersion.version('5.0')) { |  | ||||||
|             '' // applied in build.gradle |  | ||||||
|         } else if (gradleVersion < GradleVersion.version('6.0')) { |  | ||||||
|             '' // applied in build.gradle |  | ||||||
|         } else { |  | ||||||
|             """ |  | ||||||
|               plugins { |  | ||||||
|                 id 'com.gradle.enterprise' version '${GE_PLUGIN_VERSION}' |  | ||||||
|                 ${ccudPluginVersion ? "id 'com.gradle.common-custom-user-data-gradle-plugin' version '$ccudPluginVersion'" : ""} |  | ||||||
|               } |  | ||||||
|               gradleEnterprise { |  | ||||||
|                 server = '$serverUri' |  | ||||||
|                 buildScan { |  | ||||||
|                   publishAlways() |  | ||||||
|                 } |  | ||||||
|               } |  | ||||||
|             """ |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     String maybeAddPluginsToRootProject(GradleVersion gradleVersion, String ccudPluginVersion, URI serverUrl) { |  | ||||||
|         if (gradleVersion < GradleVersion.version('5.0')) { |  | ||||||
|             """ |  | ||||||
|               plugins { |  | ||||||
|                 id 'com.gradle.build-scan' version '1.16' |  | ||||||
|                 ${ccudPluginVersion ? "id 'com.gradle.common-custom-user-data-gradle-plugin' version '$ccudPluginVersion'" : ""} |  | ||||||
|               } |  | ||||||
|               buildScan { |  | ||||||
|                 server = '$serverUrl' |  | ||||||
|                 publishAlways() |  | ||||||
|               } |  | ||||||
|             """ |  | ||||||
|         } else if (gradleVersion < GradleVersion.version('6.0')) { |  | ||||||
|             """ |  | ||||||
|               plugins { |  | ||||||
|                 id 'com.gradle.build-scan' version '${GE_PLUGIN_VERSION}' |  | ||||||
|                 ${ccudPluginVersion ? "id 'com.gradle.common-custom-user-data-gradle-plugin' version '$ccudPluginVersion'" : ""} |  | ||||||
|               } |  | ||||||
|               gradleEnterprise { |  | ||||||
|                 server = '$serverUrl' |  | ||||||
|                 buildScan { |  | ||||||
|                   publishAlways() |  | ||||||
|                 } |  | ||||||
|               } |  | ||||||
|             """ |  | ||||||
|         } else { |  | ||||||
|             '' // applied in settings.gradle |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     def addFailingTaskToBuild() { |  | ||||||
|         buildFile << ''' |  | ||||||
| task expectFailure { |  | ||||||
|     doLast { |  | ||||||
|         throw new RuntimeException("Expected to fail") |  | ||||||
|     } |  | ||||||
| } |  | ||||||
| ''' |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     BuildResult run(List<String> args, String initScript, GradleVersion gradleVersion = GradleVersion.current(), List<String> jvmArgs = [], Map<String, String> envVars = [:]) { |  | ||||||
|         createRunner(initScript, args, gradleVersion, jvmArgs, envVars).build() |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     BuildResult runAndFail(List<String> args, String initScript, GradleVersion gradleVersion = GradleVersion.current(), List<String> jvmArgs = [], Map<String, String> envVars = [:]) { |  | ||||||
|         createRunner(initScript, args, gradleVersion, jvmArgs, envVars).buildAndFail() |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     GradleRunner createRunner(String initScript, List<String> args, GradleVersion gradleVersion = GradleVersion.current(), List<String> jvmArgs = [], Map<String, String> envVars = [:]) { |  | ||||||
|         File initScriptsDir = new File(testProjectDir, "initScripts") |  | ||||||
|         args << '-I' << new File(initScriptsDir, initScript).absolutePath |  | ||||||
|  |  | ||||||
|         envVars.putIfAbsent('RUNNER_TEMP', testProjectDir.absolutePath) |  | ||||||
|         envVars.putIfAbsent('GITHUB_ACTION', 'github-step-id') |  | ||||||
|  |  | ||||||
|         def runner = ((DefaultGradleRunner) GradleRunner.create()) |  | ||||||
|             .withJvmArguments(jvmArgs) |  | ||||||
|             .withGradleVersion(gradleVersion.version) |  | ||||||
|             .withProjectDir(testProjectDir) |  | ||||||
|             .withArguments(args) |  | ||||||
|             .withEnvironment(envVars) |  | ||||||
|             .forwardOutput() |  | ||||||
|  |  | ||||||
|         runner |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     static final class TestGradleVersion { |  | ||||||
|  |  | ||||||
|         final GradleVersion gradleVersion |  | ||||||
|         private final Integer jdkMin |  | ||||||
|         private final Integer jdkMax |  | ||||||
|  |  | ||||||
|         TestGradleVersion(GradleVersion gradleVersion, Integer jdkMin, Integer jdkMax) { |  | ||||||
|             this.gradleVersion = gradleVersion |  | ||||||
|             this.jdkMin = jdkMin |  | ||||||
|             this.jdkMax = jdkMax |  | ||||||
|         } |  | ||||||
|  |  | ||||||
|         boolean isCompatibleWithCurrentJvm() { |  | ||||||
|             def jvmVersion = getJvmVersion() |  | ||||||
|             jdkMin <= jvmVersion && jvmVersion <= jdkMax |  | ||||||
|         } |  | ||||||
|  |  | ||||||
|         private static int getJvmVersion() { |  | ||||||
|             String version = System.getProperty('java.version') |  | ||||||
|             if (version.startsWith('1.')) { |  | ||||||
|                 Integer.parseInt(version.substring(2, 3)) |  | ||||||
|             } else { |  | ||||||
|                 Integer.parseInt(version.substring(0, version.indexOf('.'))) |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
|  |  | ||||||
|         @Override |  | ||||||
|         String toString() { |  | ||||||
|             return "Gradle " + gradleVersion.version |  | ||||||
|         } |  | ||||||
|  |  | ||||||
|     } |  | ||||||
|  |  | ||||||
| } |  | ||||||
| @@ -1,217 +0,0 @@ | |||||||
| package com.gradle.gradlebuildaction |  | ||||||
|  |  | ||||||
| import groovy.json.JsonSlurper |  | ||||||
|  |  | ||||||
| import static org.junit.Assume.assumeTrue |  | ||||||
|  |  | ||||||
| class TestBuildResultRecorder extends BaseInitScriptTest { |  | ||||||
|     def initScript = 'gradle-build-action.build-result-capture.init.gradle' |  | ||||||
|  |  | ||||||
|     def "produces build results file for build with #testGradleVersion"() { |  | ||||||
|         assumeTrue testGradleVersion.compatibleWithCurrentJvm |  | ||||||
|  |  | ||||||
|         when: |  | ||||||
|         run(['help'], initScript, testGradleVersion.gradleVersion) |  | ||||||
|  |  | ||||||
|         then: |  | ||||||
|         assertResults('help', testGradleVersion, false, false) |  | ||||||
|  |  | ||||||
|         where: |  | ||||||
|         testGradleVersion << ALL_VERSIONS |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     def "produces build results file for failing build with #testGradleVersion"() { |  | ||||||
|         assumeTrue testGradleVersion.compatibleWithCurrentJvm |  | ||||||
|  |  | ||||||
|         when: |  | ||||||
|         addFailingTaskToBuild() |  | ||||||
|         runAndFail(['expectFailure'], initScript, testGradleVersion.gradleVersion) |  | ||||||
|  |  | ||||||
|         then: |  | ||||||
|         assertResults('expectFailure', testGradleVersion, true, false) |  | ||||||
|  |  | ||||||
|         where: |  | ||||||
|         testGradleVersion << ALL_VERSIONS |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     def "produces build results file for build with --configuration-cache on #testGradleVersion"() { |  | ||||||
|         assumeTrue testGradleVersion.compatibleWithCurrentJvm |  | ||||||
|  |  | ||||||
|         when: |  | ||||||
|         run(['help', '--configuration-cache'], initScript, testGradleVersion.gradleVersion) |  | ||||||
|  |  | ||||||
|         then: |  | ||||||
|         assertResults('help', testGradleVersion, false, false) |  | ||||||
|         assert buildResultFile.delete() |  | ||||||
|  |  | ||||||
|         when: |  | ||||||
|         run(['help', '--configuration-cache'], initScript, testGradleVersion.gradleVersion) |  | ||||||
|  |  | ||||||
|         then: |  | ||||||
|         assertResults('help', testGradleVersion, false, false) |  | ||||||
|  |  | ||||||
|         where: |  | ||||||
|         testGradleVersion << CONFIGURATION_CACHE_VERSIONS |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     def "produces build results file for #testGradleVersion with build scan published"() { |  | ||||||
|         assumeTrue testGradleVersion.compatibleWithCurrentJvm |  | ||||||
|  |  | ||||||
|         when: |  | ||||||
|         declareGePluginApplication(testGradleVersion.gradleVersion) |  | ||||||
|         run(['help'], initScript, testGradleVersion.gradleVersion) |  | ||||||
|  |  | ||||||
|         then: |  | ||||||
|         assertResults('help', testGradleVersion, false, true) |  | ||||||
|  |  | ||||||
|         where: |  | ||||||
|         testGradleVersion << ALL_VERSIONS |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     def "produces build results file for #testGradleVersion with ge-plugin and no build scan published"() { |  | ||||||
|         assumeTrue testGradleVersion.compatibleWithCurrentJvm |  | ||||||
|  |  | ||||||
|         when: |  | ||||||
|         declareGePluginApplication(testGradleVersion.gradleVersion) |  | ||||||
|         run(['help', '--no-scan'], initScript, testGradleVersion.gradleVersion) |  | ||||||
|  |  | ||||||
|         then: |  | ||||||
|         assertResults('help', testGradleVersion, false, false) |  | ||||||
|  |  | ||||||
|         where: |  | ||||||
|         testGradleVersion << ALL_VERSIONS |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     def "produces build results file for failing build on #testGradleVersion with build scan published"() { |  | ||||||
|         assumeTrue testGradleVersion.compatibleWithCurrentJvm |  | ||||||
|  |  | ||||||
|         when: |  | ||||||
|         declareGePluginApplication(testGradleVersion.gradleVersion) |  | ||||||
|         addFailingTaskToBuild() |  | ||||||
|         runAndFail(['expectFailure'], initScript, testGradleVersion.gradleVersion) |  | ||||||
|  |  | ||||||
|         then: |  | ||||||
|         assertResults('expectFailure', testGradleVersion, true, true) |  | ||||||
|  |  | ||||||
|         where: |  | ||||||
|         testGradleVersion << ALL_VERSIONS |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     def "produces build results file for build with --configuration-cache on #testGradleVersion with build scan published"() { |  | ||||||
|         assumeTrue testGradleVersion.compatibleWithCurrentJvm |  | ||||||
|  |  | ||||||
|         when: |  | ||||||
|         declareGePluginApplication(testGradleVersion.gradleVersion) |  | ||||||
|         run(['help', '--configuration-cache'], initScript, testGradleVersion.gradleVersion) |  | ||||||
|  |  | ||||||
|         then: |  | ||||||
|         assertResults('help', testGradleVersion, false, true) |  | ||||||
|         assert buildResultFile.delete() |  | ||||||
|  |  | ||||||
|         when: |  | ||||||
|         run(['help', '--configuration-cache'], initScript, testGradleVersion.gradleVersion) |  | ||||||
|  |  | ||||||
|         then: |  | ||||||
|         assertResults('help', testGradleVersion, false, true) |  | ||||||
|  |  | ||||||
|         where: |  | ||||||
|         testGradleVersion << CONFIGURATION_CACHE_VERSIONS |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     def "produces build results file for failing build on #testGradleVersion when build scan publish fails"() { |  | ||||||
|         assumeTrue testGradleVersion.compatibleWithCurrentJvm |  | ||||||
|  |  | ||||||
|         when: |  | ||||||
|         declareGePluginApplication(testGradleVersion.gradleVersion) |  | ||||||
|         addFailingTaskToBuild() |  | ||||||
|         failScanUpload = true |  | ||||||
|         runAndFail(['expectFailure'], initScript, testGradleVersion.gradleVersion) |  | ||||||
|  |  | ||||||
|         then: |  | ||||||
|         assertResults('expectFailure', testGradleVersion, true, false, true) |  | ||||||
|  |  | ||||||
|         where: |  | ||||||
|         testGradleVersion << ALL_VERSIONS |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     def "produces no build results file when GitHub env vars not set with #testGradleVersion"() { |  | ||||||
|         assumeTrue testGradleVersion.compatibleWithCurrentJvm |  | ||||||
|  |  | ||||||
|         when: |  | ||||||
|         run(['help'], initScript, testGradleVersion.gradleVersion, [], [RUNNER_TEMP: '', GITHUB_ACTION: '']) |  | ||||||
|  |  | ||||||
|         then: |  | ||||||
|         def buildResultsDir = new File(testProjectDir, '.build-results') |  | ||||||
|         assert !buildResultsDir.exists() |  | ||||||
|  |  | ||||||
|         where: |  | ||||||
|         testGradleVersion << ALL_VERSIONS |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     def "produces no build results file when RUNNER_TEMP dir is not a writable directory with #testGradleVersion"() { |  | ||||||
|         assumeTrue testGradleVersion.compatibleWithCurrentJvm |  | ||||||
|  |  | ||||||
|         when: |  | ||||||
|         def invalidDir = new File(testProjectDir, 'invalid-runner-temp') |  | ||||||
|         invalidDir.createNewFile() |  | ||||||
|  |  | ||||||
|         run(['help'], initScript, testGradleVersion.gradleVersion, [], [RUNNER_TEMP: invalidDir.absolutePath]) |  | ||||||
|  |  | ||||||
|         then: |  | ||||||
|         def buildResultsDir = new File(testProjectDir, '.build-results') |  | ||||||
|         assert !buildResultsDir.exists() |  | ||||||
|  |  | ||||||
|         where: |  | ||||||
|         testGradleVersion << ALL_VERSIONS |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     def "produces build results file with build scan when GE plugin is applied in settingsEvaluated"() { |  | ||||||
|         assumeTrue testGradleVersion.compatibleWithCurrentJvm |  | ||||||
|  |  | ||||||
|         when: |  | ||||||
|         settingsFile.text = """ |  | ||||||
|             plugins { |  | ||||||
|                 id 'com.gradle.enterprise' version '3.15.1' apply(false) |  | ||||||
|             } |  | ||||||
|             gradle.settingsEvaluated { |  | ||||||
|                 apply plugin: 'com.gradle.enterprise' |  | ||||||
|                 gradleEnterprise { |  | ||||||
|                     server = '$mockScansServer.address' |  | ||||||
|                     buildScan { |  | ||||||
|                         publishAlways() |  | ||||||
|                     } |  | ||||||
|                 } |  | ||||||
|             } |  | ||||||
|         """ + settingsFile.text |  | ||||||
|          |  | ||||||
|         run(['help'], initScript, testGradleVersion.gradleVersion) |  | ||||||
|  |  | ||||||
|         then: |  | ||||||
|         assertResults('help', testGradleVersion, false, true) |  | ||||||
|  |  | ||||||
|         where: |  | ||||||
|         testGradleVersion << SETTINGS_PLUGIN_VERSIONS |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     void assertResults(String task, TestGradleVersion testGradleVersion, boolean hasFailure, boolean hasBuildScan, boolean scanUploadFailed = false) { |  | ||||||
|         def results = new JsonSlurper().parse(buildResultFile) |  | ||||||
|         assert results['rootProjectName'] == ROOT_PROJECT_NAME |  | ||||||
|         assert results['rootProjectDir'] == testProjectDir.canonicalPath |  | ||||||
|         assert results['requestedTasks'] == task |  | ||||||
|         assert results['gradleVersion'] == testGradleVersion.gradleVersion.version |  | ||||||
|         assert results['gradleHomeDir'] != null |  | ||||||
|         assert results['buildFailed'] == hasFailure |  | ||||||
|         assert results['buildScanUri'] == (hasBuildScan ? "${mockScansServer.address}s/${PUBLIC_BUILD_SCAN_ID}" : null) |  | ||||||
|         assert results['buildScanFailed'] == scanUploadFailed |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     private File getBuildResultFile() { |  | ||||||
|         def buildResultsDir = new File(testProjectDir, '.build-results') |  | ||||||
|         assert buildResultsDir.directory |  | ||||||
|         assert buildResultsDir.listFiles().size() == 1 |  | ||||||
|         def resultsFile = buildResultsDir.listFiles()[0] |  | ||||||
|         assert resultsFile.name.startsWith('github-step-id') |  | ||||||
|         assert resultsFile.text.count('rootProjectName') == 1 |  | ||||||
|         return resultsFile |  | ||||||
|     } |  | ||||||
| } |  | ||||||
| @@ -1,134 +0,0 @@ | |||||||
| package com.gradle.gradlebuildaction |  | ||||||
|  |  | ||||||
| import static org.junit.Assume.assumeTrue |  | ||||||
|  |  | ||||||
| class TestDependencyGraph extends BaseInitScriptTest { |  | ||||||
|     def initScript = 'gradle-build-action.github-dependency-graph.init.gradle' |  | ||||||
|  |  | ||||||
|     static final List<TestGradleVersion> NO_DEPENDENCY_GRAPH_VERSIONS = [GRADLE_3_X, GRADLE_4_X] |  | ||||||
|     static final List<TestGradleVersion> DEPENDENCY_GRAPH_VERSIONS = ALL_VERSIONS - NO_DEPENDENCY_GRAPH_VERSIONS |  | ||||||
|  |  | ||||||
|     def "does not produce dependency graph when not enabled"() { |  | ||||||
|         assumeTrue testGradleVersion.compatibleWithCurrentJvm |  | ||||||
|  |  | ||||||
|         when: |  | ||||||
|         run(['help'], initScript, testGradleVersion.gradleVersion) |  | ||||||
|  |  | ||||||
|         then: |  | ||||||
|         assert !reportsDir.exists() |  | ||||||
|  |  | ||||||
|         where: |  | ||||||
|         testGradleVersion << ALL_VERSIONS |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     def "produces dependency graph when enabled"() { |  | ||||||
|         assumeTrue testGradleVersion.compatibleWithCurrentJvm |  | ||||||
|  |  | ||||||
|         when: |  | ||||||
|         run(['help'], initScript, testGradleVersion.gradleVersion, [], envVars) |  | ||||||
|  |  | ||||||
|         then: |  | ||||||
|         assert reportFile.exists() |  | ||||||
|         assert gitHubOutputFile.text == "dependency-graph-file=${reportFile.absolutePath}\n" |  | ||||||
|  |  | ||||||
|         where: |  | ||||||
|         testGradleVersion << [GRADLE_8_X] |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     def "produces dependency graph with configuration-cache on latest Gradle"() { |  | ||||||
|         assumeTrue testGradleVersion.compatibleWithCurrentJvm |  | ||||||
|  |  | ||||||
|         when: |  | ||||||
|         run(['help', '--configuration-cache'], initScript, testGradleVersion.gradleVersion, [], envVars) |  | ||||||
|  |  | ||||||
|         then: |  | ||||||
|         assert reportFile.exists() |  | ||||||
|  |  | ||||||
|         where: |  | ||||||
|         // Dependency-graph plugin doesn't support config-cache for 8.0 of Gradle |  | ||||||
|         testGradleVersion << [GRADLE_8_X] |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     def "warns and produces no dependency graph when enabled for older Gradle versions"() { |  | ||||||
|         assumeTrue testGradleVersion.compatibleWithCurrentJvm |  | ||||||
|  |  | ||||||
|         when: |  | ||||||
|         def result = run(['help'], initScript, testGradleVersion.gradleVersion, [], envVars) |  | ||||||
|  |  | ||||||
|         then: |  | ||||||
|         assert !reportsDir.exists() |  | ||||||
|         assert result.output.contains("::warning::Dependency Graph is not supported") |  | ||||||
|  |  | ||||||
|         where: |  | ||||||
|         testGradleVersion << NO_DEPENDENCY_GRAPH_VERSIONS |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     def "constructs unique job correlator for each build invocation"() { |  | ||||||
|         assumeTrue testGradleVersion.compatibleWithCurrentJvm |  | ||||||
|  |  | ||||||
|         def reportFile1 = new File(reportsDir, "CORRELATOR-1.json") |  | ||||||
|         def reportFile2 = new File(reportsDir, "CORRELATOR-2.json") |  | ||||||
|  |  | ||||||
|         buildFile << """ |  | ||||||
|             task firstTask { |  | ||||||
|                 doLast { |  | ||||||
|                     println "First" |  | ||||||
|                 } |  | ||||||
|             } |  | ||||||
|             task secondTask { |  | ||||||
|                 doLast { |  | ||||||
|                     println "Second" |  | ||||||
|                 } |  | ||||||
|             } |  | ||||||
|         """ |  | ||||||
|  |  | ||||||
|         when: |  | ||||||
|         run(['help'], initScript, testGradleVersion.gradleVersion, [], envVars) |  | ||||||
|  |  | ||||||
|         then: |  | ||||||
|         assert reportFile.exists() |  | ||||||
|  |  | ||||||
|         when: |  | ||||||
|         run(['first'], initScript, testGradleVersion.gradleVersion, [], envVars) |  | ||||||
|  |  | ||||||
|         then: |  | ||||||
|         assert reportFile.exists() |  | ||||||
|         assert reportFile1.exists() |  | ||||||
|          |  | ||||||
|         when: |  | ||||||
|         run(['second'], initScript, testGradleVersion.gradleVersion, [], envVars) |  | ||||||
|  |  | ||||||
|         then: |  | ||||||
|         assert reportFile.exists() |  | ||||||
|         assert reportFile1.exists() |  | ||||||
|         assert reportFile2.exists() |  | ||||||
|          |  | ||||||
|         where: |  | ||||||
|         testGradleVersion << DEPENDENCY_GRAPH_VERSIONS |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     def getEnvVars() { |  | ||||||
|         return [ |  | ||||||
|             GITHUB_DEPENDENCY_GRAPH_ENABLED: "true", |  | ||||||
|             GITHUB_DEPENDENCY_GRAPH_JOB_CORRELATOR: "CORRELATOR", |  | ||||||
|             GITHUB_DEPENDENCY_GRAPH_JOB_ID: "1", |  | ||||||
|             GITHUB_DEPENDENCY_GRAPH_REF: "main", |  | ||||||
|             GITHUB_DEPENDENCY_GRAPH_SHA: "123456", |  | ||||||
|             GITHUB_DEPENDENCY_GRAPH_WORKSPACE: testProjectDir.absolutePath, |  | ||||||
|             DEPENDENCY_GRAPH_REPORT_DIR: reportsDir.absolutePath, |  | ||||||
|             GITHUB_OUTPUT: gitHubOutputFile.absolutePath |  | ||||||
|         ] |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     def getReportsDir() { |  | ||||||
|         return new File(testProjectDir, 'build/reports/github-dependency-graph-snapshots') |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     def getReportFile() { |  | ||||||
|         return new File(reportsDir, "CORRELATOR.json") |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     def getGitHubOutputFile() { |  | ||||||
|         return new File(testProjectDir, "GITHUB_OUTPUT") |  | ||||||
|     } |  | ||||||
| } |  | ||||||
| @@ -1,401 +0,0 @@ | |||||||
| package com.gradle.gradlebuildaction |  | ||||||
|  |  | ||||||
| import org.gradle.testkit.runner.BuildResult |  | ||||||
| import org.gradle.util.GradleVersion |  | ||||||
|  |  | ||||||
| import static org.junit.Assume.assumeTrue |  | ||||||
|  |  | ||||||
| class TestGradleEnterpriseInjection  extends BaseInitScriptTest { |  | ||||||
|     static final List<TestGradleVersion> CCUD_COMPATIBLE_VERSIONS = ALL_VERSIONS - [GRADLE_3_X] |  | ||||||
|  |  | ||||||
|     def initScript = 'gradle-build-action.inject-gradle-enterprise.init.gradle' |  | ||||||
|  |  | ||||||
|     private static final GradleVersion GRADLE_6 = GradleVersion.version('6.0') |  | ||||||
|  |  | ||||||
|     def "does not apply GE plugins when not requested"() { |  | ||||||
|         assumeTrue testGradleVersion.compatibleWithCurrentJvm |  | ||||||
|  |  | ||||||
|         when: |  | ||||||
|         def result = run([], initScript, testGradleVersion.gradleVersion) |  | ||||||
|  |  | ||||||
|         then: |  | ||||||
|         outputMissesGePluginApplicationViaInitScript(result) |  | ||||||
|         outputMissesCcudPluginApplicationViaInitScript(result) |  | ||||||
|  |  | ||||||
|         where: |  | ||||||
|         testGradleVersion << ALL_VERSIONS |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     def "does not override GE plugin when already defined in project"() { |  | ||||||
|         assumeTrue testGradleVersion.compatibleWithCurrentJvm |  | ||||||
|  |  | ||||||
|         given: |  | ||||||
|         declareGePluginApplication(testGradleVersion.gradleVersion) |  | ||||||
|  |  | ||||||
|         when: |  | ||||||
|         def result = run(testGradleVersion, testConfig()) |  | ||||||
|  |  | ||||||
|         then: |  | ||||||
|         outputMissesGePluginApplicationViaInitScript(result) |  | ||||||
|         outputMissesCcudPluginApplicationViaInitScript(result) |  | ||||||
|  |  | ||||||
|         and: |  | ||||||
|         outputContainsBuildScanUrl(result) |  | ||||||
|  |  | ||||||
|         where: |  | ||||||
|         testGradleVersion << ALL_VERSIONS |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     def "applies GE plugin via init script when not defined in project"() { |  | ||||||
|         assumeTrue testGradleVersion.compatibleWithCurrentJvm |  | ||||||
|  |  | ||||||
|         when: |  | ||||||
|         def result = run(testGradleVersion, testConfig()) |  | ||||||
|  |  | ||||||
|         then: |  | ||||||
|         outputContainsGePluginApplicationViaInitScript(result, testGradleVersion.gradleVersion) |  | ||||||
|         outputMissesCcudPluginApplicationViaInitScript(result) |  | ||||||
|  |  | ||||||
|         and: |  | ||||||
|         outputContainsBuildScanUrl(result) |  | ||||||
|  |  | ||||||
|         where: |  | ||||||
|         testGradleVersion << ALL_VERSIONS |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     def "applies GE and CCUD plugins via init script when not defined in project"() { |  | ||||||
|         assumeTrue testGradleVersion.compatibleWithCurrentJvm |  | ||||||
|  |  | ||||||
|         when: |  | ||||||
|         def result = run(testGradleVersion, testConfig().withCCUDPlugin()) |  | ||||||
|  |  | ||||||
|         then: |  | ||||||
|         outputContainsGePluginApplicationViaInitScript(result, testGradleVersion.gradleVersion) |  | ||||||
|         outputContainsCcudPluginApplicationViaInitScript(result) |  | ||||||
|  |  | ||||||
|         and: |  | ||||||
|         outputContainsBuildScanUrl(result) |  | ||||||
|  |  | ||||||
|         where: |  | ||||||
|         testGradleVersion << CCUD_COMPATIBLE_VERSIONS |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     def "applies CCUD plugin via init script where GE plugin already applied"() { |  | ||||||
|         assumeTrue testGradleVersion.compatibleWithCurrentJvm |  | ||||||
|  |  | ||||||
|         given: |  | ||||||
|         declareGePluginApplication(testGradleVersion.gradleVersion) |  | ||||||
|  |  | ||||||
|         when: |  | ||||||
|         def result = run(testGradleVersion, testConfig().withCCUDPlugin()) |  | ||||||
|  |  | ||||||
|         then: |  | ||||||
|         outputMissesGePluginApplicationViaInitScript(result) |  | ||||||
|         outputContainsCcudPluginApplicationViaInitScript(result) |  | ||||||
|  |  | ||||||
|         and: |  | ||||||
|         outputContainsBuildScanUrl(result) |  | ||||||
|  |  | ||||||
|         where: |  | ||||||
|         testGradleVersion << CCUD_COMPATIBLE_VERSIONS |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     def "does not override CCUD plugin when already defined in project"() { |  | ||||||
|         assumeTrue testGradleVersion.compatibleWithCurrentJvm |  | ||||||
|  |  | ||||||
|         given: |  | ||||||
|         declareGePluginAndCcudPluginApplication(testGradleVersion.gradleVersion) |  | ||||||
|  |  | ||||||
|         when: |  | ||||||
|         def result = run(testGradleVersion, testConfig().withCCUDPlugin()) |  | ||||||
|  |  | ||||||
|         then: |  | ||||||
|         outputMissesGePluginApplicationViaInitScript(result) |  | ||||||
|         outputMissesCcudPluginApplicationViaInitScript(result) |  | ||||||
|  |  | ||||||
|         and: |  | ||||||
|         outputContainsBuildScanUrl(result) |  | ||||||
|  |  | ||||||
|         where: |  | ||||||
|         testGradleVersion << CCUD_COMPATIBLE_VERSIONS |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     def "ignores GE URL and allowUntrustedServer when GE plugin is not applied by the init script"() { |  | ||||||
|         assumeTrue testGradleVersion.compatibleWithCurrentJvm |  | ||||||
|  |  | ||||||
|         given: |  | ||||||
|         declareGePluginApplication(testGradleVersion.gradleVersion) |  | ||||||
|  |  | ||||||
|         when: |  | ||||||
|         def config = testConfig().withServer(URI.create('https://ge-server.invalid')) |  | ||||||
|         def result = run(testGradleVersion, config) |  | ||||||
|  |  | ||||||
|         then: |  | ||||||
|         outputMissesGePluginApplicationViaInitScript(result) |  | ||||||
|         outputMissesCcudPluginApplicationViaInitScript(result) |  | ||||||
|  |  | ||||||
|         and: |  | ||||||
|         outputContainsBuildScanUrl(result) |  | ||||||
|  |  | ||||||
|         where: |  | ||||||
|         testGradleVersion << ALL_VERSIONS |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     def "configures GE URL and allowUntrustedServer when GE plugin is applied by the init script"() { |  | ||||||
|         assumeTrue testGradleVersion.compatibleWithCurrentJvm |  | ||||||
|  |  | ||||||
|         when: |  | ||||||
|         def config = testConfig().withServer(mockScansServer.address) |  | ||||||
|         def result = run(testGradleVersion, config) |  | ||||||
|  |  | ||||||
|         then: |  | ||||||
|         outputContainsGePluginApplicationViaInitScript(result, testGradleVersion.gradleVersion) |  | ||||||
|         outputContainsGeConnectionInfo(result, mockScansServer.address.toString(), true) |  | ||||||
|         outputMissesCcudPluginApplicationViaInitScript(result) |  | ||||||
|         outputContainsPluginRepositoryInfo(result, 'https://plugins.gradle.org/m2') |  | ||||||
|  |  | ||||||
|         and: |  | ||||||
|         outputContainsBuildScanUrl(result) |  | ||||||
|  |  | ||||||
|         where: |  | ||||||
|         testGradleVersion << ALL_VERSIONS |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     def "enforces GE URL and allowUntrustedServer in project if enforce url parameter is enabled"() { |  | ||||||
|         assumeTrue testGradleVersion.compatibleWithCurrentJvm |  | ||||||
|  |  | ||||||
|         given: |  | ||||||
|         declareGePluginApplication(testGradleVersion.gradleVersion, URI.create('https://ge-server.invalid')) |  | ||||||
|  |  | ||||||
|         when: |  | ||||||
|         def config = testConfig().withServer(mockScansServer.address, true) |  | ||||||
|         def result = run(testGradleVersion, config) |  | ||||||
|  |  | ||||||
|         then: |  | ||||||
|         outputMissesGePluginApplicationViaInitScript(result) |  | ||||||
|         outputMissesCcudPluginApplicationViaInitScript(result) |  | ||||||
|  |  | ||||||
|         and: |  | ||||||
|         outputEnforcesGeUrl(result, mockScansServer.address.toString(), true) |  | ||||||
|  |  | ||||||
|         and: |  | ||||||
|         outputContainsBuildScanUrl(result) |  | ||||||
|  |  | ||||||
|         where: |  | ||||||
|         testGradleVersion << ALL_VERSIONS |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     def "can configure alternative repository for plugins when GE plugin is applied by the init script"() { |  | ||||||
|         assumeTrue testGradleVersion.compatibleWithCurrentJvm |  | ||||||
|  |  | ||||||
|         when: |  | ||||||
|         def config = testConfig().withPluginRepository(new URI('https://plugins.grdev.net/m2')) |  | ||||||
|         def result = run(testGradleVersion, config) |  | ||||||
|  |  | ||||||
|         then: |  | ||||||
|         outputContainsGePluginApplicationViaInitScript(result, testGradleVersion.gradleVersion) |  | ||||||
|         outputContainsGeConnectionInfo(result, mockScansServer.address.toString(), true) |  | ||||||
|         outputMissesCcudPluginApplicationViaInitScript(result) |  | ||||||
|         outputContainsPluginRepositoryInfo(result, 'https://plugins.grdev.net/m2') |  | ||||||
|  |  | ||||||
|         and: |  | ||||||
|         outputContainsBuildScanUrl(result) |  | ||||||
|  |  | ||||||
|         where: |  | ||||||
|         testGradleVersion << ALL_VERSIONS |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     def "stops gracefully when requested CCUD plugin version is <1.7"() { |  | ||||||
|         assumeTrue testGradleVersion.compatibleWithCurrentJvm |  | ||||||
|  |  | ||||||
|         when: |  | ||||||
|         def config = testConfig().withCCUDPlugin("1.6.6") |  | ||||||
|         def result = run(testGradleVersion, config) |  | ||||||
|  |  | ||||||
|         then: |  | ||||||
|         outputMissesGePluginApplicationViaInitScript(result) |  | ||||||
|         outputMissesCcudPluginApplicationViaInitScript(result) |  | ||||||
|         result.output.contains('Common Custom User Data Gradle plugin must be at least 1.7. Configured version is 1.6.6.') |  | ||||||
|  |  | ||||||
|         where: |  | ||||||
|         testGradleVersion << ALL_VERSIONS |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     def "can configure GE via CCUD system property overrides when CCUD plugin is inject via init script"() { |  | ||||||
|         assumeTrue testGradleVersion.compatibleWithCurrentJvm |  | ||||||
|  |  | ||||||
|         when: |  | ||||||
|         def config = testConfig().withCCUDPlugin().withServer(URI.create('https://ge-server.invalid')) |  | ||||||
|         def result = run(testGradleVersion, config, ["help", "-Dgradle.enterprise.url=${mockScansServer.address}".toString()]) |  | ||||||
|  |  | ||||||
|         then: |  | ||||||
|         outputContainsGePluginApplicationViaInitScript(result, testGradleVersion.gradleVersion) |  | ||||||
|         outputContainsCcudPluginApplicationViaInitScript(result) |  | ||||||
|  |  | ||||||
|         and: |  | ||||||
|         outputContainsBuildScanUrl(result) |  | ||||||
|  |  | ||||||
|         where: |  | ||||||
|         testGradleVersion << CCUD_COMPATIBLE_VERSIONS |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     def "init script is configuration cache compatible"() { |  | ||||||
|         assumeTrue testGradleVersion.compatibleWithCurrentJvm |  | ||||||
|  |  | ||||||
|         when: |  | ||||||
|         def config = testConfig().withCCUDPlugin() |  | ||||||
|         def result = run(testGradleVersion, config, ["help", "--configuration-cache"]) |  | ||||||
|  |  | ||||||
|         then: |  | ||||||
|         outputContainsGePluginApplicationViaInitScript(result, testGradleVersion.gradleVersion) |  | ||||||
|         outputContainsCcudPluginApplicationViaInitScript(result) |  | ||||||
|  |  | ||||||
|         and: |  | ||||||
|         outputContainsBuildScanUrl(result) |  | ||||||
|  |  | ||||||
|         when: |  | ||||||
|         result = run(testGradleVersion, config, ["help", "--configuration-cache"]) |  | ||||||
|  |  | ||||||
|         then: |  | ||||||
|         outputMissesGePluginApplicationViaInitScript(result) |  | ||||||
|         outputMissesCcudPluginApplicationViaInitScript(result) |  | ||||||
|  |  | ||||||
|         and: |  | ||||||
|         outputContainsBuildScanUrl(result) |  | ||||||
|  |  | ||||||
|         where: |  | ||||||
|         testGradleVersion << CONFIGURATION_CACHE_VERSIONS |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     void outputContainsBuildScanUrl(BuildResult result) { |  | ||||||
|         def message = "Publishing build scan...\n${mockScansServer.address}s/$PUBLIC_BUILD_SCAN_ID" |  | ||||||
|         assert result.output.contains(message) |  | ||||||
|         assert 1 == result.output.count(message) |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     void outputContainsGePluginApplicationViaInitScript(BuildResult result, GradleVersion gradleVersion) { |  | ||||||
|         def pluginApplicationLogMsgGradle4And5 = "Applying com.gradle.scan.plugin.BuildScanPlugin via init script" |  | ||||||
|         def pluginApplicationLogMsgGradle6AndHigher = "Applying com.gradle.enterprise.gradleplugin.GradleEnterprisePlugin via init script" |  | ||||||
|         if (gradleVersion < GRADLE_6) { |  | ||||||
|             assert result.output.contains(pluginApplicationLogMsgGradle4And5) |  | ||||||
|             assert 1 == result.output.count(pluginApplicationLogMsgGradle4And5) |  | ||||||
|             assert !result.output.contains(pluginApplicationLogMsgGradle6AndHigher) |  | ||||||
|         } else { |  | ||||||
|             assert result.output.contains(pluginApplicationLogMsgGradle6AndHigher) |  | ||||||
|             assert 1 == result.output.count(pluginApplicationLogMsgGradle6AndHigher) |  | ||||||
|             assert !result.output.contains(pluginApplicationLogMsgGradle4And5) |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     void outputMissesGePluginApplicationViaInitScript(BuildResult result) { |  | ||||||
|         def pluginApplicationLogMsgGradle4And5 = "Applying com.gradle.scan.plugin.BuildScanPlugin via init script" |  | ||||||
|         def pluginApplicationLogMsgGradle6AndHigher = "Applying com.gradle.enterprise.gradleplugin.GradleEnterprisePlugin via init script" |  | ||||||
|         assert !result.output.contains(pluginApplicationLogMsgGradle4And5) |  | ||||||
|         assert !result.output.contains(pluginApplicationLogMsgGradle6AndHigher) |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     void outputContainsCcudPluginApplicationViaInitScript(BuildResult result) { |  | ||||||
|         def pluginApplicationLogMsg = "Applying com.gradle.CommonCustomUserDataGradlePlugin via init script" |  | ||||||
|         assert result.output.contains(pluginApplicationLogMsg) |  | ||||||
|         assert 1 == result.output.count(pluginApplicationLogMsg) |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     void outputMissesCcudPluginApplicationViaInitScript(BuildResult result) { |  | ||||||
|         def pluginApplicationLogMsg = "Applying com.gradle.CommonCustomUserDataGradlePlugin via init script" |  | ||||||
|         assert !result.output.contains(pluginApplicationLogMsg) |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     void outputContainsGeConnectionInfo(BuildResult result, String geUrl, boolean geAllowUntrustedServer) { |  | ||||||
|         def geConnectionInfo = "Connection to Gradle Enterprise: $geUrl, allowUntrustedServer: $geAllowUntrustedServer" |  | ||||||
|         assert result.output.contains(geConnectionInfo) |  | ||||||
|         assert 1 == result.output.count(geConnectionInfo) |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     void outputContainsPluginRepositoryInfo(BuildResult result, String gradlePluginRepositoryUrl) { |  | ||||||
|         def repositoryInfo = "Gradle Enterprise plugins resolution: ${gradlePluginRepositoryUrl}" |  | ||||||
|         assert result.output.contains(repositoryInfo) |  | ||||||
|         assert 1 == result.output.count(repositoryInfo) |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     void outputEnforcesGeUrl(BuildResult result, String geUrl, boolean geAllowUntrustedServer) { |  | ||||||
|         def enforceUrl = "Enforcing Gradle Enterprise: $geUrl, allowUntrustedServer: $geAllowUntrustedServer" |  | ||||||
|         assert result.output.contains(enforceUrl) |  | ||||||
|         assert 1 == result.output.count(enforceUrl) |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     private BuildResult run(TestGradleVersion testGradleVersion, TestConfig config, List<String> args = ["help"]) { |  | ||||||
|         if (testKitSupportsEnvVars(testGradleVersion.gradleVersion)) { |  | ||||||
|             return run(args, initScript, testGradleVersion.gradleVersion, [], config.envVars) |  | ||||||
|         } else { |  | ||||||
|             return run(args, initScript, testGradleVersion.gradleVersion, config.jvmArgs, [:]) |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     private boolean testKitSupportsEnvVars(GradleVersion gradleVersion) { |  | ||||||
|         // TestKit supports env vars for Gradle 3.5+, except on M1 Mac where only 6.9+ is supported |  | ||||||
|         def isM1Mac = System.getProperty("os.arch") == "aarch64" |  | ||||||
|         if (isM1Mac) { |  | ||||||
|             return gradleVersion >= GRADLE_6_X.gradleVersion |  | ||||||
|         } else { |  | ||||||
|             return gradleVersion >= GRADLE_3_X.gradleVersion |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     private TestConfig testConfig() { |  | ||||||
|         new TestConfig() |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     class TestConfig { |  | ||||||
|         String serverUrl = mockScansServer.address.toString() |  | ||||||
|         boolean enforceUrl = false |  | ||||||
|         String ccudPluginVersion = null |  | ||||||
|         String pluginRepositoryUrl = null |  | ||||||
|  |  | ||||||
|         TestConfig withCCUDPlugin(String version = CCUD_PLUGIN_VERSION) { |  | ||||||
|             ccudPluginVersion = version |  | ||||||
|             return this |  | ||||||
|         } |  | ||||||
|  |  | ||||||
|         TestConfig withServer(URI url, boolean enforceUrl = false) { |  | ||||||
|             serverUrl = url.toASCIIString() |  | ||||||
|             this.enforceUrl = enforceUrl |  | ||||||
|             return this |  | ||||||
|         } |  | ||||||
|  |  | ||||||
|         TestConfig withPluginRepository(URI pluginRepositoryUrl) { |  | ||||||
|             this.pluginRepositoryUrl = pluginRepositoryUrl |  | ||||||
|             return this |  | ||||||
|         } |  | ||||||
|  |  | ||||||
|         def getEnvVars() { |  | ||||||
|             Map<String, String> envVars = [ |  | ||||||
|                 GRADLE_ENTERPRISE_INJECTION_ENABLED: "true", |  | ||||||
|                 GRADLE_ENTERPRISE_URL: serverUrl, |  | ||||||
|                 GRADLE_ENTERPRISE_ALLOW_UNTRUSTED_SERVER: "true", |  | ||||||
|                 GRADLE_ENTERPRISE_PLUGIN_VERSION: GE_PLUGIN_VERSION, |  | ||||||
|                 GRADLE_ENTERPRISE_BUILD_SCAN_UPLOAD_IN_BACKGROUND: "true" // Need to upload in background since our Mock server doesn't cope with foreground upload |  | ||||||
|             ] |  | ||||||
|             if (enforceUrl) envVars.put("GRADLE_ENTERPRISE_ENFORCE_URL", "true") |  | ||||||
|             if (ccudPluginVersion != null) envVars.put("GRADLE_ENTERPRISE_CCUD_PLUGIN_VERSION", ccudPluginVersion) |  | ||||||
|             if (pluginRepositoryUrl != null) envVars.put("GRADLE_ENTERPRISE_PLUGIN_REPOSITORY_URL", pluginRepositoryUrl) |  | ||||||
|  |  | ||||||
|             return envVars |  | ||||||
|         } |  | ||||||
|  |  | ||||||
|         def getJvmArgs() { |  | ||||||
|             List<String> jvmArgs = [ |  | ||||||
|                 "-Dgradle-enterprise.injection-enabled=true", |  | ||||||
|                 "-Dgradle-enterprise.url=$serverUrl", |  | ||||||
|                 "-Dgradle-enterprise.allow-untrusted-server=true", |  | ||||||
|                 "-Dgradle-enterprise.plugin.version=$GE_PLUGIN_VERSION", |  | ||||||
|                 "-Dgradle-enterprise.build-scan.upload-in-background=true" |  | ||||||
|             ] |  | ||||||
|  |  | ||||||
|             if (enforceUrl) jvmArgs.add("-Dgradle-enterprise.enforce-url=true") |  | ||||||
|             if (ccudPluginVersion != null) jvmArgs.add("-Dgradle-enterprise.ccud-plugin.version=$ccudPluginVersion") |  | ||||||
|             if (pluginRepositoryUrl != null) jvmArgs.add("-Dgradle-enterprise.plugin-repository.url=$pluginRepositoryUrl") |  | ||||||
|  |  | ||||||
|             return jvmArgs.collect { it.toString() } // Convert from GStrings |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| } |  | ||||||
| @@ -1,87 +0,0 @@ | |||||||
| import * as exec from '@actions/exec' |  | ||||||
| import fs from 'fs' |  | ||||||
| import path from 'path' |  | ||||||
| import {CacheCleaner} from '../../src/cache-cleaner' |  | ||||||
|  |  | ||||||
| jest.setTimeout(120000) |  | ||||||
|  |  | ||||||
| test('will cleanup unused dependency jars and build-cache entries', async () => { |  | ||||||
|     const projectRoot = prepareTestProject() |  | ||||||
|     const gradleUserHome = path.resolve(projectRoot, 'HOME') |  | ||||||
|     const tmpDir = path.resolve(projectRoot, 'tmp') |  | ||||||
|     const cacheCleaner = new CacheCleaner(gradleUserHome, tmpDir) |  | ||||||
|  |  | ||||||
|     await runGradleBuild(projectRoot, 'build', '3.1') |  | ||||||
|      |  | ||||||
|     await cacheCleaner.prepare() |  | ||||||
|  |  | ||||||
|     await runGradleBuild(projectRoot, 'build', '3.1.1') |  | ||||||
|  |  | ||||||
|     const commonsMath31 = path.resolve(gradleUserHome, "caches/modules-2/files-2.1/org.apache.commons/commons-math3/3.1") |  | ||||||
|     const commonsMath311 = path.resolve(gradleUserHome, "caches/modules-2/files-2.1/org.apache.commons/commons-math3/3.1.1") |  | ||||||
|     const buildCacheDir = path.resolve(gradleUserHome, "caches/build-cache-1") |  | ||||||
|  |  | ||||||
|     expect(fs.existsSync(commonsMath31)).toBe(true) |  | ||||||
|     expect(fs.existsSync(commonsMath311)).toBe(true) |  | ||||||
|     expect(fs.readdirSync(buildCacheDir).length).toBe(4) |  | ||||||
|  |  | ||||||
|     await cacheCleaner.forceCleanup() |  | ||||||
|  |  | ||||||
|     expect(fs.existsSync(commonsMath31)).toBe(false) |  | ||||||
|     expect(fs.existsSync(commonsMath311)).toBe(true) |  | ||||||
|     expect(fs.readdirSync(buildCacheDir).length).toBe(3) |  | ||||||
| }) |  | ||||||
|  |  | ||||||
| test('will cleanup unused gradle versions', async () => { |  | ||||||
|     const projectRoot = prepareTestProject() |  | ||||||
|     const gradleUserHome = path.resolve(projectRoot, 'HOME') |  | ||||||
|     const tmpDir = path.resolve(projectRoot, 'tmp') |  | ||||||
|     const cacheCleaner = new CacheCleaner(gradleUserHome, tmpDir) |  | ||||||
|  |  | ||||||
|     // Initialize HOME with 2 different Gradle versions |  | ||||||
|     await runGradleWrapperBuild(projectRoot, 'build') |  | ||||||
|     await runGradleBuild(projectRoot, 'build') |  | ||||||
|      |  | ||||||
|     await cacheCleaner.prepare() |  | ||||||
|  |  | ||||||
|     // Run with only one of these versions |  | ||||||
|     await runGradleBuild(projectRoot, 'build') |  | ||||||
|  |  | ||||||
|     const gradle802 = path.resolve(gradleUserHome, "caches/8.0.2") |  | ||||||
|     const wrapper802 = path.resolve(gradleUserHome, "wrapper/dists/gradle-8.0.2-bin") |  | ||||||
|     const gradleCurrent = path.resolve(gradleUserHome, "caches/8.5") |  | ||||||
|  |  | ||||||
|     expect(fs.existsSync(gradle802)).toBe(true) |  | ||||||
|     expect(fs.existsSync(wrapper802)).toBe(true) |  | ||||||
|     expect(fs.existsSync(gradleCurrent)).toBe(true) |  | ||||||
|  |  | ||||||
|     await cacheCleaner.forceCleanup() |  | ||||||
|  |  | ||||||
|     expect(fs.existsSync(gradle802)).toBe(false) |  | ||||||
|     expect(fs.existsSync(wrapper802)).toBe(false) |  | ||||||
|     expect(fs.existsSync(gradleCurrent)).toBe(true) |  | ||||||
| }) |  | ||||||
|  |  | ||||||
| async function runGradleBuild(projectRoot: string, args: string, version: string = '3.1'): Promise<void> { |  | ||||||
|     const status31 = await exec.exec(`gradle -g HOME --no-daemon --build-cache -Dcommons_math3_version="${version}" ${args}`, [], { |  | ||||||
|         cwd: projectRoot |  | ||||||
|     }) |  | ||||||
|     console.log(`Gradle User Home initialized with commons_math3_version=${version} ${args}`) |  | ||||||
| } |  | ||||||
|  |  | ||||||
| async function runGradleWrapperBuild(projectRoot: string, args: string, version: string = '3.1'): Promise<void> { |  | ||||||
|     const status31 = await exec.exec(`./gradlew -g HOME --no-daemon --build-cache -Dcommons_math3_version="${version}" ${args}`, [], { |  | ||||||
|         cwd: projectRoot |  | ||||||
|     }) |  | ||||||
|     console.log(`Gradle User Home initialized with commons_math3_version="${version}" ${args}`) |  | ||||||
| } |  | ||||||
|  |  | ||||||
| function prepareTestProject(): string { |  | ||||||
|     const projectRoot = 'test/jest/resources/cache-cleanup' |  | ||||||
|     fs.rmSync(path.resolve(projectRoot, 'HOME'), { recursive: true, force: true }) |  | ||||||
|     fs.rmSync(path.resolve(projectRoot, 'tmp'), { recursive: true, force: true }) |  | ||||||
|     fs.rmSync(path.resolve(projectRoot, 'build'), { recursive: true, force: true }) |  | ||||||
|     fs.rmSync(path.resolve(projectRoot, '.gradle'), { recursive: true, force: true }) |  | ||||||
|     return projectRoot |  | ||||||
| } |  | ||||||
|  |  | ||||||
| @@ -1,95 +0,0 @@ | |||||||
| import {CacheEntryListener, CacheListener} from '../../src/cache-reporting' |  | ||||||
|  |  | ||||||
| describe('caching report', () => { |  | ||||||
|     describe('reports not fully restored', () => { |  | ||||||
|         it('with one requested entry report', async () => { |  | ||||||
|             const report = new CacheListener() |  | ||||||
|             report.entry('foo').markRequested('1', ['2']) |  | ||||||
|             report.entry('bar').markRequested('3').markRestored('4', 500) |  | ||||||
|             expect(report.fullyRestored).toBe(false) |  | ||||||
|         }) |  | ||||||
|     }) |  | ||||||
|     describe('reports fully restored', () => { |  | ||||||
|         it('when empty', async () => { |  | ||||||
|             const report = new CacheListener() |  | ||||||
|             expect(report.fullyRestored).toBe(true) |  | ||||||
|         }) |  | ||||||
|         it('with empty entry reports', async () => { |  | ||||||
|             const report = new CacheListener() |  | ||||||
|             report.entry('foo') |  | ||||||
|             report.entry('bar') |  | ||||||
|             expect(report.fullyRestored).toBe(true) |  | ||||||
|         }) |  | ||||||
|         it('with restored entry report', async () => { |  | ||||||
|             const report = new CacheListener() |  | ||||||
|             report.entry('bar').markRequested('3').markRestored('4', 300) |  | ||||||
|             expect(report.fullyRestored).toBe(true) |  | ||||||
|         }) |  | ||||||
|         it('with multiple restored entry reportss', async () => { |  | ||||||
|             const report = new CacheListener() |  | ||||||
|             report.entry('foo').markRestored('4', 3300) |  | ||||||
|             report.entry('bar').markRequested('3').markRestored('4', 333) |  | ||||||
|             expect(report.fullyRestored).toBe(true) |  | ||||||
|         }) |  | ||||||
|     }) |  | ||||||
|     describe('can be stringified and rehydrated', () => { |  | ||||||
|         it('when empty', async () => { |  | ||||||
|             const report = new CacheListener() |  | ||||||
|  |  | ||||||
|             const stringRep = report.stringify() |  | ||||||
|             const reportClone: CacheListener = CacheListener.rehydrate(stringRep) |  | ||||||
|  |  | ||||||
|             expect(reportClone.cacheEntries).toEqual([]) |  | ||||||
|  |  | ||||||
|             // Can call methods on rehydrated |  | ||||||
|             expect(reportClone.entry('foo')).toBeInstanceOf(CacheEntryListener) |  | ||||||
|         }) |  | ||||||
|         it('with entry reports', async () => { |  | ||||||
|             const report = new CacheListener() |  | ||||||
|             report.entry('foo') |  | ||||||
|             report.entry('bar') |  | ||||||
|             report.entry('baz') |  | ||||||
|  |  | ||||||
|             const stringRep = report.stringify() |  | ||||||
|             const reportClone: CacheListener = CacheListener.rehydrate(stringRep) |  | ||||||
|  |  | ||||||
|             expect(reportClone.cacheEntries.length).toBe(3) |  | ||||||
|             expect(reportClone.cacheEntries[0].entryName).toBe('foo') |  | ||||||
|             expect(reportClone.cacheEntries[1].entryName).toBe('bar') |  | ||||||
|             expect(reportClone.cacheEntries[2].entryName).toBe('baz') |  | ||||||
|  |  | ||||||
|             expect(reportClone.entry('foo')).toBe(reportClone.cacheEntries[0]) |  | ||||||
|         }) |  | ||||||
|         it('with rehydrated entry report', async () => { |  | ||||||
|             const report = new CacheListener() |  | ||||||
|             const entryReport = report.entry('foo') |  | ||||||
|             entryReport.markRequested('1', ['2', '3']) |  | ||||||
|             entryReport.markSaved('4', 100) |  | ||||||
|  |  | ||||||
|             const stringRep = report.stringify() |  | ||||||
|             const reportClone: CacheListener = CacheListener.rehydrate(stringRep) |  | ||||||
|             const entryClone = reportClone.entry('foo') |  | ||||||
|  |  | ||||||
|             expect(entryClone.requestedKey).toBe('1') |  | ||||||
|             expect(entryClone.requestedRestoreKeys).toEqual(['2', '3']) |  | ||||||
|             expect(entryClone.savedKey).toBe('4') |  | ||||||
|         }) |  | ||||||
|         it('with live entry report', async () => { |  | ||||||
|             const report = new CacheListener() |  | ||||||
|             const entryReport = report.entry('foo') |  | ||||||
|             entryReport.markRequested('1', ['2', '3']) |  | ||||||
|  |  | ||||||
|             const stringRep = report.stringify() |  | ||||||
|             const reportClone: CacheListener = CacheListener.rehydrate(stringRep) |  | ||||||
|             const entryClone = reportClone.entry('foo') |  | ||||||
|  |  | ||||||
|             // Check type and call method on rehydrated entry report |  | ||||||
|             expect(entryClone).toBeInstanceOf(CacheEntryListener) |  | ||||||
|             entryClone.markSaved('4', 100) |  | ||||||
|  |  | ||||||
|             expect(entryClone.requestedKey).toBe('1') |  | ||||||
|             expect(entryClone.requestedRestoreKeys).toEqual(['2', '3']) |  | ||||||
|             expect(entryClone.savedKey).toBe('4') |  | ||||||
|         }) |  | ||||||
|     }) |  | ||||||
| }) |  | ||||||
| @@ -1,26 +0,0 @@ | |||||||
| import * as cacheUtils from '../../src/cache-utils' |  | ||||||
|  |  | ||||||
| describe('cacheUtils-utils', () => { |  | ||||||
|     describe('can hash', () => { |  | ||||||
|         it('a string', async () => { |  | ||||||
|             const hash = cacheUtils.hashStrings(['foo']) |  | ||||||
|             expect(hash).toBe('acbd18db4cc2f85cedef654fccc4a4d8') |  | ||||||
|         }) |  | ||||||
|         it('multiple strings', async () => { |  | ||||||
|             const hash = cacheUtils.hashStrings(['foo', 'bar', 'baz']) |  | ||||||
|             expect(hash).toBe('6df23dc03f9b54cc38a0fc1483df6e21') |  | ||||||
|         }) |  | ||||||
|         it('normalized filenames', async () => { |  | ||||||
|             const fileNames = ['/foo/bar/baz.zip', '../boo.html'] |  | ||||||
|             const posixHash = cacheUtils.hashFileNames(fileNames) |  | ||||||
|             const windowsHash = cacheUtils.hashFileNames(fileNames) |  | ||||||
|             expect(posixHash).toBe(windowsHash) |  | ||||||
|         }) |  | ||||||
|     }) |  | ||||||
|     describe('sanitizes workflow name in cache key', () => { |  | ||||||
|         it('with comma', () => { |  | ||||||
|             const cacheKey = cacheUtils.getCacheKeyForJob("Workflow, with,commas", "JOB_ID") |  | ||||||
|             expect(cacheKey).toBe('workflow withcommas-JOB_ID') |  | ||||||
|         }) |  | ||||||
|     }) |  | ||||||
| }) |  | ||||||
| @@ -1,34 +0,0 @@ | |||||||
| import * as dependencyGraph from '../../src/dependency-graph' |  | ||||||
|  |  | ||||||
| describe('dependency-graph', () => { |  | ||||||
|     describe('constructs job correlator', () => { |  | ||||||
|         it('removes commas from workflow name', () => { |  | ||||||
|             const id = dependencyGraph.constructJobCorrelator('Workflow, with,commas', 'jobid', '{}') |  | ||||||
|             expect(id).toBe('workflow_withcommas-jobid') |  | ||||||
|         }) |  | ||||||
|         it('removes non word characters', () => { |  | ||||||
|             const id = dependencyGraph.constructJobCorrelator('Workflow!_with()characters', 'job-*id', '{"foo": "bar!@#$%^&*("}') |  | ||||||
|             expect(id).toBe('workflow_withcharacters-job-id-bar') |  | ||||||
|         }) |  | ||||||
|         it('replaces spaces', () => { |  | ||||||
|             const id = dependencyGraph.constructJobCorrelator('Workflow !_ with () characters, and   spaces', 'job-*id', '{"foo": "bar!@#$%^&*("}') |  | ||||||
|             expect(id).toBe('workflow___with_characters_and_spaces-job-id-bar') |  | ||||||
|         }) |  | ||||||
|         it('without matrix', () => { |  | ||||||
|             const id = dependencyGraph.constructJobCorrelator('workflow', 'jobid', 'null') |  | ||||||
|             expect(id).toBe('workflow-jobid') |  | ||||||
|         }) |  | ||||||
|         it('with dashes in values', () => { |  | ||||||
|             const id = dependencyGraph.constructJobCorrelator('workflow-name', 'job-id', '{"os": "ubuntu-latest"}') |  | ||||||
|             expect(id).toBe('workflow-name-job-id-ubuntu-latest') |  | ||||||
|         }) |  | ||||||
|         it('with single matrix value', () => { |  | ||||||
|             const id = dependencyGraph.constructJobCorrelator('workflow', 'jobid', '{"os": "windows"}') |  | ||||||
|             expect(id).toBe('workflow-jobid-windows') |  | ||||||
|         }) |  | ||||||
|         it('with composite matrix value', () => { |  | ||||||
|             const id = dependencyGraph.constructJobCorrelator('workflow', 'jobid', '{"os": "windows", "java-version": "21.1", "other": "Value, with COMMA"}') |  | ||||||
|             expect(id).toBe('workflow-jobid-windows-211-value_with_comma') |  | ||||||
|         }) |  | ||||||
|     }) |  | ||||||
| }) |  | ||||||
| @@ -1,22 +0,0 @@ | |||||||
| import * as inputParams from '../../src/input-params' |  | ||||||
|  |  | ||||||
| describe('input params', () => { |  | ||||||
|     describe('parses numeric input', () => { |  | ||||||
|         it('uses default value', () => { |  | ||||||
|             const val = inputParams.parseNumericInput('param-name', '', 88) |  | ||||||
|             expect(val).toBe(88) |  | ||||||
|         }) |  | ||||||
|         it('parses numeric input', () => { |  | ||||||
|             const val = inputParams.parseNumericInput('param-name', '34', 88) |  | ||||||
|             expect(val).toBe(34) |  | ||||||
|         }) |  | ||||||
|         it('fails on non-numeric input', () => { |  | ||||||
|             const t = () => { |  | ||||||
|                 inputParams.parseNumericInput('param-name', 'xyz', 88) |  | ||||||
|             }; |  | ||||||
|  |  | ||||||
|             expect(t).toThrow(TypeError) |  | ||||||
|             expect(t).toThrow("The value 'xyz' is not a valid numeric value for 'param-name'.") |  | ||||||
|         }) |  | ||||||
|     }) |  | ||||||
| }) |  | ||||||
| @@ -1,6 +0,0 @@ | |||||||
| # |  | ||||||
| # https://help.github.com/articles/dealing-with-line-endings/ |  | ||||||
| # |  | ||||||
| # These are explicitly windows files and should use crlf |  | ||||||
| *.bat           text eol=crlf |  | ||||||
|  |  | ||||||
							
								
								
									
										8
									
								
								test/jest/resources/cache-cleanup/.gitignore
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										8
									
								
								test/jest/resources/cache-cleanup/.gitignore
									
									
									
									
										vendored
									
									
								
							| @@ -1,8 +0,0 @@ | |||||||
| # Ignore Gradle project-specific cache directory |  | ||||||
| .gradle |  | ||||||
|  |  | ||||||
| # Ignore Gradle build output directory |  | ||||||
| build |  | ||||||
|  |  | ||||||
| HOME |  | ||||||
| tmp |  | ||||||
| @@ -1,11 +0,0 @@ | |||||||
| plugins { |  | ||||||
|     id 'java-library' |  | ||||||
| } |  | ||||||
|  |  | ||||||
| repositories { |  | ||||||
|     mavenCentral() |  | ||||||
| } |  | ||||||
|  |  | ||||||
| dependencies { |  | ||||||
|     api "org.apache.commons:commons-math3:${System.properties['commons_math3_version']}" |  | ||||||
| } |  | ||||||
										
											Binary file not shown.
										
									
								
							| @@ -1,6 +0,0 @@ | |||||||
| distributionBase=GRADLE_USER_HOME |  | ||||||
| distributionPath=wrapper/dists |  | ||||||
| # Deliberately not using the latest Gradle version for cache cleanup testing |  | ||||||
| distributionUrl=https\://services.gradle.org/distributions/gradle-8.0.2-bin.zip |  | ||||||
| zipStoreBase=GRADLE_USER_HOME |  | ||||||
| zipStorePath=wrapper/dists |  | ||||||
							
								
								
									
										245
									
								
								test/jest/resources/cache-cleanup/gradlew
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										245
									
								
								test/jest/resources/cache-cleanup/gradlew
									
									
									
									
										vendored
									
									
								
							| @@ -1,245 +0,0 @@ | |||||||
| #!/bin/sh |  | ||||||
|  |  | ||||||
| # |  | ||||||
| # Copyright © 2015-2021 the original authors. |  | ||||||
| # |  | ||||||
| # Licensed under the Apache License, Version 2.0 (the "License"); |  | ||||||
| # you may not use this file except in compliance with the License. |  | ||||||
| # You may obtain a copy of the License at |  | ||||||
| # |  | ||||||
| #      https://www.apache.org/licenses/LICENSE-2.0 |  | ||||||
| # |  | ||||||
| # Unless required by applicable law or agreed to in writing, software |  | ||||||
| # distributed under the License is distributed on an "AS IS" BASIS, |  | ||||||
| # WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. |  | ||||||
| # See the License for the specific language governing permissions and |  | ||||||
| # limitations under the License. |  | ||||||
| # |  | ||||||
|  |  | ||||||
| ############################################################################## |  | ||||||
| # |  | ||||||
| #   Gradle start up script for POSIX generated by Gradle. |  | ||||||
| # |  | ||||||
| #   Important for running: |  | ||||||
| # |  | ||||||
| #   (1) You need a POSIX-compliant shell to run this script. If your /bin/sh is |  | ||||||
| #       noncompliant, but you have some other compliant shell such as ksh or |  | ||||||
| #       bash, then to run this script, type that shell name before the whole |  | ||||||
| #       command line, like: |  | ||||||
| # |  | ||||||
| #           ksh Gradle |  | ||||||
| # |  | ||||||
| #       Busybox and similar reduced shells will NOT work, because this script |  | ||||||
| #       requires all of these POSIX shell features: |  | ||||||
| #         * functions; |  | ||||||
| #         * expansions «$var», «${var}», «${var:-default}», «${var+SET}», |  | ||||||
| #           «${var#prefix}», «${var%suffix}», and «$( cmd )»; |  | ||||||
| #         * compound commands having a testable exit status, especially «case»; |  | ||||||
| #         * various built-in commands including «command», «set», and «ulimit». |  | ||||||
| # |  | ||||||
| #   Important for patching: |  | ||||||
| # |  | ||||||
| #   (2) This script targets any POSIX shell, so it avoids extensions provided |  | ||||||
| #       by Bash, Ksh, etc; in particular arrays are avoided. |  | ||||||
| # |  | ||||||
| #       The "traditional" practice of packing multiple parameters into a |  | ||||||
| #       space-separated string is a well documented source of bugs and security |  | ||||||
| #       problems, so this is (mostly) avoided, by progressively accumulating |  | ||||||
| #       options in "$@", and eventually passing that to Java. |  | ||||||
| # |  | ||||||
| #       Where the inherited environment variables (DEFAULT_JVM_OPTS, JAVA_OPTS, |  | ||||||
| #       and GRADLE_OPTS) rely on word-splitting, this is performed explicitly; |  | ||||||
| #       see the in-line comments for details. |  | ||||||
| # |  | ||||||
| #       There are tweaks for specific operating systems such as AIX, CygWin, |  | ||||||
| #       Darwin, MinGW, and NonStop. |  | ||||||
| # |  | ||||||
| #   (3) This script is generated from the Groovy template |  | ||||||
| #       https://github.com/gradle/gradle/blob/HEAD/subprojects/plugins/src/main/resources/org/gradle/api/internal/plugins/unixStartScript.txt |  | ||||||
| #       within the Gradle project. |  | ||||||
| # |  | ||||||
| #       You can find Gradle at https://github.com/gradle/gradle/. |  | ||||||
| # |  | ||||||
| ############################################################################## |  | ||||||
|  |  | ||||||
| # Attempt to set APP_HOME |  | ||||||
|  |  | ||||||
| # Resolve links: $0 may be a link |  | ||||||
| app_path=$0 |  | ||||||
|  |  | ||||||
| # Need this for daisy-chained symlinks. |  | ||||||
| while |  | ||||||
|     APP_HOME=${app_path%"${app_path##*/}"}  # leaves a trailing /; empty if no leading path |  | ||||||
|     [ -h "$app_path" ] |  | ||||||
| do |  | ||||||
|     ls=$( ls -ld "$app_path" ) |  | ||||||
|     link=${ls#*' -> '} |  | ||||||
|     case $link in             #( |  | ||||||
|       /*)   app_path=$link ;; #( |  | ||||||
|       *)    app_path=$APP_HOME$link ;; |  | ||||||
|     esac |  | ||||||
| done |  | ||||||
|  |  | ||||||
| # This is normally unused |  | ||||||
| # shellcheck disable=SC2034 |  | ||||||
| APP_BASE_NAME=${0##*/} |  | ||||||
| APP_HOME=$( cd "${APP_HOME:-./}" && pwd -P ) || exit |  | ||||||
|  |  | ||||||
| # Use the maximum available, or set MAX_FD != -1 to use that value. |  | ||||||
| MAX_FD=maximum |  | ||||||
|  |  | ||||||
| warn () { |  | ||||||
|     echo "$*" |  | ||||||
| } >&2 |  | ||||||
|  |  | ||||||
| die () { |  | ||||||
|     echo |  | ||||||
|     echo "$*" |  | ||||||
|     echo |  | ||||||
|     exit 1 |  | ||||||
| } >&2 |  | ||||||
|  |  | ||||||
| # OS specific support (must be 'true' or 'false'). |  | ||||||
| cygwin=false |  | ||||||
| msys=false |  | ||||||
| darwin=false |  | ||||||
| nonstop=false |  | ||||||
| case "$( uname )" in                #( |  | ||||||
|   CYGWIN* )         cygwin=true  ;; #( |  | ||||||
|   Darwin* )         darwin=true  ;; #( |  | ||||||
|   MSYS* | MINGW* )  msys=true    ;; #( |  | ||||||
|   NONSTOP* )        nonstop=true ;; |  | ||||||
| esac |  | ||||||
|  |  | ||||||
| CLASSPATH=$APP_HOME/gradle/wrapper/gradle-wrapper.jar |  | ||||||
|  |  | ||||||
|  |  | ||||||
| # Determine the Java command to use to start the JVM. |  | ||||||
| if [ -n "$JAVA_HOME" ] ; then |  | ||||||
|     if [ -x "$JAVA_HOME/jre/sh/java" ] ; then |  | ||||||
|         # IBM's JDK on AIX uses strange locations for the executables |  | ||||||
|         JAVACMD=$JAVA_HOME/jre/sh/java |  | ||||||
|     else |  | ||||||
|         JAVACMD=$JAVA_HOME/bin/java |  | ||||||
|     fi |  | ||||||
|     if [ ! -x "$JAVACMD" ] ; then |  | ||||||
|         die "ERROR: JAVA_HOME is set to an invalid directory: $JAVA_HOME |  | ||||||
|  |  | ||||||
| Please set the JAVA_HOME variable in your environment to match the |  | ||||||
| location of your Java installation." |  | ||||||
|     fi |  | ||||||
| else |  | ||||||
|     JAVACMD=java |  | ||||||
|     which java >/dev/null 2>&1 || die "ERROR: JAVA_HOME is not set and no 'java' command could be found in your PATH. |  | ||||||
|  |  | ||||||
| Please set the JAVA_HOME variable in your environment to match the |  | ||||||
| location of your Java installation." |  | ||||||
| fi |  | ||||||
|  |  | ||||||
| # Increase the maximum file descriptors if we can. |  | ||||||
| if ! "$cygwin" && ! "$darwin" && ! "$nonstop" ; then |  | ||||||
|     case $MAX_FD in #( |  | ||||||
|       max*) |  | ||||||
|         # In POSIX sh, ulimit -H is undefined. That's why the result is checked to see if it worked. |  | ||||||
|         # shellcheck disable=SC3045 |  | ||||||
|         MAX_FD=$( ulimit -H -n ) || |  | ||||||
|             warn "Could not query maximum file descriptor limit" |  | ||||||
|     esac |  | ||||||
|     case $MAX_FD in  #( |  | ||||||
|       '' | soft) :;; #( |  | ||||||
|       *) |  | ||||||
|         # In POSIX sh, ulimit -n is undefined. That's why the result is checked to see if it worked. |  | ||||||
|         # shellcheck disable=SC3045 |  | ||||||
|         ulimit -n "$MAX_FD" || |  | ||||||
|             warn "Could not set maximum file descriptor limit to $MAX_FD" |  | ||||||
|     esac |  | ||||||
| fi |  | ||||||
|  |  | ||||||
| # Collect all arguments for the java command, stacking in reverse order: |  | ||||||
| #   * args from the command line |  | ||||||
| #   * the main class name |  | ||||||
| #   * -classpath |  | ||||||
| #   * -D...appname settings |  | ||||||
| #   * --module-path (only if needed) |  | ||||||
| #   * DEFAULT_JVM_OPTS, JAVA_OPTS, and GRADLE_OPTS environment variables. |  | ||||||
|  |  | ||||||
| # For Cygwin or MSYS, switch paths to Windows format before running java |  | ||||||
| if "$cygwin" || "$msys" ; then |  | ||||||
|     APP_HOME=$( cygpath --path --mixed "$APP_HOME" ) |  | ||||||
|     CLASSPATH=$( cygpath --path --mixed "$CLASSPATH" ) |  | ||||||
|  |  | ||||||
|     JAVACMD=$( cygpath --unix "$JAVACMD" ) |  | ||||||
|  |  | ||||||
|     # Now convert the arguments - kludge to limit ourselves to /bin/sh |  | ||||||
|     for arg do |  | ||||||
|         if |  | ||||||
|             case $arg in                                #( |  | ||||||
|               -*)   false ;;                            # don't mess with options #( |  | ||||||
|               /?*)  t=${arg#/} t=/${t%%/*}              # looks like a POSIX filepath |  | ||||||
|                     [ -e "$t" ] ;;                      #( |  | ||||||
|               *)    false ;; |  | ||||||
|             esac |  | ||||||
|         then |  | ||||||
|             arg=$( cygpath --path --ignore --mixed "$arg" ) |  | ||||||
|         fi |  | ||||||
|         # Roll the args list around exactly as many times as the number of |  | ||||||
|         # args, so each arg winds up back in the position where it started, but |  | ||||||
|         # possibly modified. |  | ||||||
|         # |  | ||||||
|         # NB: a `for` loop captures its iteration list before it begins, so |  | ||||||
|         # changing the positional parameters here affects neither the number of |  | ||||||
|         # iterations, nor the values presented in `arg`. |  | ||||||
|         shift                   # remove old arg |  | ||||||
|         set -- "$@" "$arg"      # push replacement arg |  | ||||||
|     done |  | ||||||
| fi |  | ||||||
|  |  | ||||||
|  |  | ||||||
| # Add default JVM options here. You can also use JAVA_OPTS and GRADLE_OPTS to pass JVM options to this script. |  | ||||||
| DEFAULT_JVM_OPTS='"-Xmx64m" "-Xms64m"' |  | ||||||
|  |  | ||||||
| # Collect all arguments for the java command; |  | ||||||
| #   * $DEFAULT_JVM_OPTS, $JAVA_OPTS, and $GRADLE_OPTS can contain fragments of |  | ||||||
| #     shell script including quotes and variable substitutions, so put them in |  | ||||||
| #     double quotes to make sure that they get re-expanded; and |  | ||||||
| #   * put everything else in single quotes, so that it's not re-expanded. |  | ||||||
|  |  | ||||||
| set -- \ |  | ||||||
|         "-Dorg.gradle.appname=$APP_BASE_NAME" \ |  | ||||||
|         -classpath "$CLASSPATH" \ |  | ||||||
|         org.gradle.wrapper.GradleWrapperMain \ |  | ||||||
|         "$@" |  | ||||||
|  |  | ||||||
| # Stop when "xargs" is not available. |  | ||||||
| if ! command -v xargs >/dev/null 2>&1 |  | ||||||
| then |  | ||||||
|     die "xargs is not available" |  | ||||||
| fi |  | ||||||
|  |  | ||||||
| # Use "xargs" to parse quoted args. |  | ||||||
| # |  | ||||||
| # With -n1 it outputs one arg per line, with the quotes and backslashes removed. |  | ||||||
| # |  | ||||||
| # In Bash we could simply go: |  | ||||||
| # |  | ||||||
| #   readarray ARGS < <( xargs -n1 <<<"$var" ) && |  | ||||||
| #   set -- "${ARGS[@]}" "$@" |  | ||||||
| # |  | ||||||
| # but POSIX shell has neither arrays nor command substitution, so instead we |  | ||||||
| # post-process each arg (as a line of input to sed) to backslash-escape any |  | ||||||
| # character that might be a shell metacharacter, then use eval to reverse |  | ||||||
| # that process (while maintaining the separation between arguments), and wrap |  | ||||||
| # the whole thing up as a single "set" statement. |  | ||||||
| # |  | ||||||
| # This will of course break if any of these variables contains a newline or |  | ||||||
| # an unmatched quote. |  | ||||||
| # |  | ||||||
|  |  | ||||||
| eval "set -- $( |  | ||||||
|         printf '%s\n' "$DEFAULT_JVM_OPTS $JAVA_OPTS $GRADLE_OPTS" | |  | ||||||
|         xargs -n1 | |  | ||||||
|         sed ' s~[^-[:alnum:]+,./:=@_]~\\&~g; ' | |  | ||||||
|         tr '\n' ' ' |  | ||||||
|     )" '"$@"' |  | ||||||
|  |  | ||||||
| exec "$JAVACMD" "$@" |  | ||||||
Some files were not shown because too many files have changed in this diff Show More
		Reference in New Issue
	
	Block a user