mirror of
				https://github.com/gradle/gradle-build-action.git
				synced 2025-10-25 03:58:56 +08:00 
			
		
		
		
	Compare commits
	
		
			39 Commits
		
	
	
		
			v2.0-beta.
			...
			v2.0-rc.2
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
|  | 2a57ddf74a | ||
|  | 230fd6b47f | ||
|  | 472ac8a356 | ||
|  | 3ba05ede1f | ||
|  | d785346c8c | ||
|  | 6ca4d4ade2 | ||
|  | 75cec40e58 | ||
|  | c317ccac62 | ||
|  | a74bb0fad6 | ||
|  | 6ff2065a12 | ||
|  | 727b4612ba | ||
|  | 613f4ec588 | ||
|  | db6202adcd | ||
|  | f0f68e07c3 | ||
|  | 8ba5a0033b | ||
|  | 9edc2a11bd | ||
|  | 079e4844d6 | ||
|  | 4ebd000afd | ||
|  | 063fc6a872 | ||
|  | e3ada7e5c2 | ||
|  | d61e5be06a | ||
|  | db2b34260f | ||
|  | c031dc946b | ||
|  | 0eb881f067 | ||
|  | 27f2dc276c | ||
|  | cba1833dde | ||
|  | 39db90e99b | ||
|  | 947a893558 | ||
|  | b99e9f0bc3 | ||
|  | 4cf255df10 | ||
|  | 614d8770a4 | ||
|  | 69453dbfc5 | ||
|  | 1113cb87cb | ||
|  | 9c95294209 | ||
|  | f901ec9c20 | ||
|  | a94b9252d5 | ||
|  | 25672bf196 | ||
|  | cb6a0acca4 | ||
|  | aa2ed2e033 | 
| @@ -11,7 +11,7 @@ | |||||||
|       "eslint-comments/no-use": "off", |       "eslint-comments/no-use": "off", | ||||||
|       "import/no-namespace": "off", |       "import/no-namespace": "off", | ||||||
|       "no-unused-vars": "off", |       "no-unused-vars": "off", | ||||||
|       "@typescript-eslint/no-unused-vars": "error", |       "@typescript-eslint/no-unused-vars": ["error", { "argsIgnorePattern": "^_" }], | ||||||
|       "@typescript-eslint/explicit-member-accessibility": ["error", {"accessibility": "no-public"}], |       "@typescript-eslint/explicit-member-accessibility": ["error", {"accessibility": "no-public"}], | ||||||
|       "@typescript-eslint/no-require-imports": "error", |       "@typescript-eslint/no-require-imports": "error", | ||||||
|       "@typescript-eslint/array-type": "error", |       "@typescript-eslint/array-type": "error", | ||||||
|   | |||||||
							
								
								
									
										2
									
								
								.github/workflows/failure-cases.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										2
									
								
								.github/workflows/failure-cases.yml
									
									
									
									
										vendored
									
									
								
							| @@ -4,7 +4,7 @@ on: | |||||||
|   workflow_dispatch: |   workflow_dispatch: | ||||||
|  |  | ||||||
| env: | env: | ||||||
|   CACHE_KEY_PREFIX: ${{github.workflow}}#${{github.run_number}}- |   GRADLE_BUILD_ACTION_CACHE_KEY_PREFIX: ${{github.workflow}}#${{github.run_number}}- | ||||||
|  |  | ||||||
| jobs: | jobs: | ||||||
|  |  | ||||||
|   | |||||||
| @@ -6,11 +6,14 @@ on: | |||||||
|   workflow_dispatch: |   workflow_dispatch: | ||||||
|  |  | ||||||
| env: | env: | ||||||
|   CACHE_KEY_PREFIX: ${{github.workflow}}#${{github.run_number}}- |   GRADLE_BUILD_ACTION_CACHE_KEY_PREFIX: ${{github.workflow}}#${{github.run_number}}- | ||||||
|  |  | ||||||
| jobs: | jobs: | ||||||
|   action-inputs: |   action-inputs: | ||||||
|     runs-on: ubuntu-latest |     strategy: | ||||||
|  |       matrix: | ||||||
|  |         os: [ubuntu-latest, windows-latest] | ||||||
|  |     runs-on: ${{ matrix.os }} | ||||||
|     steps: |     steps: | ||||||
|     - name: Checkout sources |     - name: Checkout sources | ||||||
|       uses: actions/checkout@v2 |       uses: actions/checkout@v2 | ||||||
|   | |||||||
							
								
								
									
										75
									
								
								.github/workflows/integTest-caching-config.yml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										75
									
								
								.github/workflows/integTest-caching-config.yml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,75 @@ | |||||||
|  | name: Test caching configuration | ||||||
|  |  | ||||||
|  | on: | ||||||
|  |   pull_request: | ||||||
|  |   push: | ||||||
|  |   workflow_dispatch: | ||||||
|  |  | ||||||
|  | env: | ||||||
|  |   GRADLE_BUILD_ACTION_CACHE_KEY_PREFIX: ${{github.workflow}}#${{github.run_number}}- | ||||||
|  |   GRADLE_BUILD_ACTION_CACHE_DEBUG_ENABLED: true | ||||||
|  |  | ||||||
|  | jobs: | ||||||
|  |   # Run initial Gradle builds to push initial cache entries | ||||||
|  |   # These builds should start fresh without cache hits, due to the seed injected into the cache key above. | ||||||
|  |   seed-build: | ||||||
|  |     strategy: | ||||||
|  |       matrix: | ||||||
|  |         os: [ubuntu-latest, windows-latest] | ||||||
|  |     runs-on: ${{ matrix.os }} | ||||||
|  |     steps: | ||||||
|  |     - name: Checkout sources | ||||||
|  |       uses: actions/checkout@v2 | ||||||
|  |     - name: Build using Gradle wrapper | ||||||
|  |       uses: ./ | ||||||
|  |       with: | ||||||
|  |         build-root-directory: __tests__/samples/groovy-dsl | ||||||
|  |         arguments: test | ||||||
|  |         # Add "wrapper" to main cache entry and remove 'wrapper-zips' bundle | ||||||
|  |         # Exclude build-cache from main cache entry | ||||||
|  |         gradle-home-cache-includes: | | ||||||
|  |             caches | ||||||
|  |             notifications | ||||||
|  |             wrapper | ||||||
|  |         gradle-home-cache-excludes: | | ||||||
|  |             caches/build-cache-1 | ||||||
|  |         gradle-home-cache-artifact-bundles: | | ||||||
|  |             [ | ||||||
|  |               ["generated-gradle-jars", "caches/*/generated-gradle-jars/*.jar"], | ||||||
|  |               ["dependencies", "caches/modules-*/files-*/*/*/*/*/"], | ||||||
|  |               ["instrumented-jars", "caches/jars-*/*/"], | ||||||
|  |               ["kotlin-dsl", "caches/*/kotlin-dsl/*/*/"] | ||||||
|  |             ] | ||||||
|  |  | ||||||
|  |   # Test that the gradle-user-home cache will cache dependencies, by running build with --offline | ||||||
|  |   verify-build: | ||||||
|  |     needs: seed-build | ||||||
|  |     strategy: | ||||||
|  |       matrix: | ||||||
|  |         os: [ubuntu-latest, windows-latest] | ||||||
|  |     runs-on: ${{ matrix.os }} | ||||||
|  |     steps: | ||||||
|  |     - name: Checkout sources | ||||||
|  |       uses: actions/checkout@v2 | ||||||
|  |     - name: Execute Gradle build with --offline | ||||||
|  |       uses: ./ | ||||||
|  |       with: | ||||||
|  |         build-root-directory: __tests__/samples/groovy-dsl | ||||||
|  |         arguments: test --offline | ||||||
|  |         cache-read-only: true | ||||||
|  |         # Need the same configuration when restoring state from cache | ||||||
|  |         gradle-home-cache-includes: | | ||||||
|  |             caches | ||||||
|  |             notifications | ||||||
|  |             wrapper | ||||||
|  |         gradle-home-cache-excludes: | | ||||||
|  |             caches/build-cache-1 | ||||||
|  |         gradle-home-cache-artifact-bundles: | | ||||||
|  |             [ | ||||||
|  |               ["generated-gradle-jars", "caches/*/generated-gradle-jars/*.jar"], | ||||||
|  |               ["dependencies", "caches/modules-*/files-*/*/*/*/*/"], | ||||||
|  |               ["instrumented-jars", "caches/jars-*/*/"], | ||||||
|  |               ["kotlin-dsl", "caches/*/kotlin-dsl/*/*/"] | ||||||
|  |             ] | ||||||
|  |  | ||||||
|  |  | ||||||
							
								
								
									
										65
									
								
								.github/workflows/integTest-caching-configuration-cache.yml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										65
									
								
								.github/workflows/integTest-caching-configuration-cache.yml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,65 @@ | |||||||
|  | name: Test save/restore configuration-cache state | ||||||
|  |  | ||||||
|  | on: | ||||||
|  |   pull_request: | ||||||
|  |   push: | ||||||
|  |   workflow_dispatch: | ||||||
|  |  | ||||||
|  | env: | ||||||
|  |   GRADLE_BUILD_ACTION_CACHE_KEY_PREFIX: ${{github.workflow}}#${{github.run_number}}- | ||||||
|  |   GRADLE_BUILD_ACTION_CACHE_DEBUG_ENABLED: true | ||||||
|  |  | ||||||
|  | jobs: | ||||||
|  |   # Run initial Gradle builds to push initial cache entries | ||||||
|  |   # These builds should start fresh without cache hits, due to the seed injected into the cache key above. | ||||||
|  |   seed-build: | ||||||
|  |     strategy: | ||||||
|  |       matrix: | ||||||
|  |         os: [ubuntu-latest, windows-latest] | ||||||
|  |     runs-on: ${{ matrix.os }} | ||||||
|  |     steps: | ||||||
|  |     - name: Checkout sources | ||||||
|  |       uses: actions/checkout@v2 | ||||||
|  |     - name: Build with configuration-cache enabled | ||||||
|  |       uses: ./ | ||||||
|  |       with: | ||||||
|  |         build-root-directory: __tests__/samples/groovy-dsl | ||||||
|  |         arguments: test --configuration-cache | ||||||
|  |  | ||||||
|  |   # Test that the project-dot-gradle cache will cache and restore configuration-cache | ||||||
|  |   configuration-cache: | ||||||
|  |     needs: seed-build | ||||||
|  |     strategy: | ||||||
|  |       matrix: | ||||||
|  |         os: [ubuntu-latest, windows-latest] | ||||||
|  |     runs-on: ${{ matrix.os }} | ||||||
|  |     steps: | ||||||
|  |     - name: Checkout sources | ||||||
|  |       uses: actions/checkout@v2 | ||||||
|  |     - name: Execute Gradle build and verify cached configuration | ||||||
|  |       uses: ./ | ||||||
|  |       env:  | ||||||
|  |         VERIFY_CACHED_CONFIGURATION: true | ||||||
|  |       with: | ||||||
|  |         build-root-directory: __tests__/samples/groovy-dsl | ||||||
|  |         arguments: test --configuration-cache | ||||||
|  |         cache-read-only: true | ||||||
|  |  | ||||||
|  |   # Check that the build can run when no bundles are restored | ||||||
|  |   no-bundles-restored: | ||||||
|  |     needs: seed-build | ||||||
|  |     strategy: | ||||||
|  |       matrix: | ||||||
|  |         os: [ubuntu-latest, windows-latest] | ||||||
|  |     runs-on: ${{ matrix.os }} | ||||||
|  |     steps: | ||||||
|  |     - name: Checkout sources | ||||||
|  |       uses: actions/checkout@v2 | ||||||
|  |     - name: Execute Gradle build with no cache artifact bundles restored | ||||||
|  |       uses: ./ | ||||||
|  |       with: | ||||||
|  |         build-root-directory: __tests__/samples/groovy-dsl | ||||||
|  |         arguments: test --configuration-cache | ||||||
|  |         cache-read-only: true | ||||||
|  |         gradle-home-cache-artifact-bundles: '[]' | ||||||
|  |  | ||||||
| @@ -1,4 +1,4 @@ | |||||||
| name: Test caching | name: Test save/restore Gradle Home directory | ||||||
| 
 | 
 | ||||||
| on: | on: | ||||||
|   pull_request: |   pull_request: | ||||||
| @@ -6,7 +6,7 @@ on: | |||||||
|   workflow_dispatch: |   workflow_dispatch: | ||||||
| 
 | 
 | ||||||
| env: | env: | ||||||
|   CACHE_KEY_PREFIX: ${{github.workflow}}#${{github.run_number}}- |   GRADLE_BUILD_ACTION_CACHE_KEY_PREFIX: ${{github.workflow}}#${{github.run_number}}- | ||||||
| 
 | 
 | ||||||
| jobs: | jobs: | ||||||
|   # Run initial Gradle builds to push initial cache entries |   # Run initial Gradle builds to push initial cache entries | ||||||
| @@ -14,7 +14,7 @@ jobs: | |||||||
|   seed-build: |   seed-build: | ||||||
|     strategy: |     strategy: | ||||||
|       matrix: |       matrix: | ||||||
|         os: [ubuntu-latest, macos-latest, windows-latest] |         os: [ubuntu-latest, windows-latest] | ||||||
|     runs-on: ${{ matrix.os }} |     runs-on: ${{ matrix.os }} | ||||||
|     steps: |     steps: | ||||||
|     - name: Checkout sources |     - name: Checkout sources | ||||||
| @@ -24,18 +24,13 @@ jobs: | |||||||
|       with: |       with: | ||||||
|         build-root-directory: __tests__/samples/groovy-dsl |         build-root-directory: __tests__/samples/groovy-dsl | ||||||
|         arguments: test |         arguments: test | ||||||
|     - name: Build with configuration-cache enabled |  | ||||||
|       uses: ./ |  | ||||||
|       with: |  | ||||||
|         build-root-directory: __tests__/samples/groovy-dsl |  | ||||||
|         arguments: test --configuration-cache |  | ||||||
| 
 | 
 | ||||||
|   # Test that the gradle-user-home cache will cache dependencies, by running build with --offline |   # Test that the gradle-user-home cache will cache dependencies, by running build with --offline | ||||||
|   dependencies-cache: |   dependencies-cache: | ||||||
|     needs: seed-build |     needs: seed-build | ||||||
|     strategy: |     strategy: | ||||||
|       matrix: |       matrix: | ||||||
|         os: [ubuntu-latest, macos-latest, windows-latest] |         os: [ubuntu-latest, windows-latest] | ||||||
|     runs-on: ${{ matrix.os }} |     runs-on: ${{ matrix.os }} | ||||||
|     steps: |     steps: | ||||||
|     - name: Checkout sources |     - name: Checkout sources | ||||||
| @@ -52,7 +47,7 @@ jobs: | |||||||
|     needs: seed-build |     needs: seed-build | ||||||
|     strategy: |     strategy: | ||||||
|       matrix: |       matrix: | ||||||
|         os: [ubuntu-latest, macos-latest, windows-latest] |         os: [ubuntu-latest, windows-latest] | ||||||
|     runs-on: ${{ matrix.os }} |     runs-on: ${{ matrix.os }} | ||||||
|     steps: |     steps: | ||||||
|     - name: Checkout sources |     - name: Checkout sources | ||||||
| @@ -64,29 +59,13 @@ jobs: | |||||||
|         arguments: test -DverifyCachedBuild=true |         arguments: test -DverifyCachedBuild=true | ||||||
|         cache-read-only: true |         cache-read-only: true | ||||||
| 
 | 
 | ||||||
|   # Test that the project-dot-gradle cache will cache and restore configuration-cache |  | ||||||
|   configuration-cache: |  | ||||||
|     needs: seed-build |  | ||||||
|     strategy: |  | ||||||
|       matrix: |  | ||||||
|         os: [ubuntu-latest, macos-latest, windows-latest] |  | ||||||
|     runs-on: ${{ matrix.os }} |  | ||||||
|     steps: |  | ||||||
|     - name: Checkout sources |  | ||||||
|       uses: actions/checkout@v2 |  | ||||||
|     - name: Execute Gradle build and verify cached configuration |  | ||||||
|       uses: ./ |  | ||||||
|       env:  |  | ||||||
|         VERIFY_CACHED_CONFIGURATION: true |  | ||||||
|       with: |  | ||||||
|         build-root-directory: __tests__/samples/groovy-dsl |  | ||||||
|         arguments: test --configuration-cache |  | ||||||
|         cache-read-only: true |  | ||||||
| 
 |  | ||||||
|   # Check that the build can run when no bundles are restored |   # Check that the build can run when no bundles are restored | ||||||
|   no-bundles-restored: |   no-bundles-restored: | ||||||
|     needs: seed-build |     needs: seed-build | ||||||
|     runs-on: ubuntu-latest |     strategy: | ||||||
|  |       matrix: | ||||||
|  |         os: [ubuntu-latest, windows-latest] | ||||||
|  |     runs-on: ${{ matrix.os }} | ||||||
|     steps: |     steps: | ||||||
|     - name: Checkout sources |     - name: Checkout sources | ||||||
|       uses: actions/checkout@v2 |       uses: actions/checkout@v2 | ||||||
| @@ -95,6 +74,6 @@ jobs: | |||||||
|       with: |       with: | ||||||
|         build-root-directory: __tests__/samples/groovy-dsl |         build-root-directory: __tests__/samples/groovy-dsl | ||||||
|         arguments: test |         arguments: test | ||||||
|         cache-artifact-bundles: '[]' |  | ||||||
|         cache-read-only: true |         cache-read-only: true | ||||||
|  |         gradle-home-cache-artifact-bundles: '[]' | ||||||
| 
 | 
 | ||||||
							
								
								
									
										6
									
								
								.github/workflows/integTest-execution.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										6
									
								
								.github/workflows/integTest-execution.yml
									
									
									
									
										vendored
									
									
								
							| @@ -6,7 +6,7 @@ on: | |||||||
|   workflow_dispatch: |   workflow_dispatch: | ||||||
|  |  | ||||||
| env: | env: | ||||||
|   CACHE_KEY_PREFIX: ${{github.workflow}}#${{github.run_number}}- |   GRADLE_BUILD_ACTION_CACHE_KEY_PREFIX: ${{github.workflow}}#${{github.run_number}}- | ||||||
|  |  | ||||||
| jobs:    | jobs:    | ||||||
|   # Tests for executing with different Gradle versions.  |   # Tests for executing with different Gradle versions.  | ||||||
| @@ -14,7 +14,7 @@ jobs: | |||||||
|   gradle-execution: |   gradle-execution: | ||||||
|     strategy: |     strategy: | ||||||
|       matrix: |       matrix: | ||||||
|         os: [ubuntu-latest, macos-latest, windows-latest] |         os: [ubuntu-latest, windows-latest] | ||||||
|         include: |         include: | ||||||
|           - os: windows-latest |           - os: windows-latest | ||||||
|             script-suffix: '.bat' |             script-suffix: '.bat' | ||||||
| @@ -44,7 +44,7 @@ jobs: | |||||||
|   gradle-versions: |   gradle-versions: | ||||||
|     strategy: |     strategy: | ||||||
|       matrix: |       matrix: | ||||||
|         os: [ubuntu-latest, macos-latest, windows-latest] |         os: [ubuntu-latest, windows-latest] | ||||||
|         include: |         include: | ||||||
|           - os: windows-latest |           - os: windows-latest | ||||||
|             script-suffix: '.bat' |             script-suffix: '.bat' | ||||||
|   | |||||||
							
								
								
									
										17
									
								
								.github/workflows/integTest-gradle-user-home.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										17
									
								
								.github/workflows/integTest-gradle-user-home.yml
									
									
									
									
										vendored
									
									
								
							| @@ -6,14 +6,17 @@ on: | |||||||
|   workflow_dispatch: |   workflow_dispatch: | ||||||
|  |  | ||||||
| env: | env: | ||||||
|   CACHE_KEY_PREFIX: ${{github.workflow}}#${{github.run_number}}- |   GRADLE_BUILD_ACTION_CACHE_KEY_PREFIX: ${{github.workflow}}#${{github.run_number}}- | ||||||
|   GRADLE_USER_HOME: custom/gradle/home |   GRADLE_USER_HOME: custom/gradle/home | ||||||
|  |  | ||||||
| jobs: | jobs: | ||||||
|   # Run initial Gradle builds to push initial cache entries |   # Run initial Gradle builds to push initial cache entries | ||||||
|   # These builds should start fresh without cache hits, due to the seed injected into the cache key above. |   # These builds should start fresh without cache hits, due to the seed injected into the cache key above. | ||||||
|   seed-build: |   seed-build: | ||||||
|     runs-on: ubuntu-latest |     strategy: | ||||||
|  |       matrix: | ||||||
|  |         os: [ubuntu-latest, windows-latest] | ||||||
|  |     runs-on: ${{ matrix.os }} | ||||||
|     steps: |     steps: | ||||||
|     - name: Checkout sources |     - name: Checkout sources | ||||||
|       uses: actions/checkout@v2 |       uses: actions/checkout@v2 | ||||||
| @@ -26,7 +29,10 @@ jobs: | |||||||
|   # Test that the gradle-user-home cache will cache dependencies, by running build with --offline |   # Test that the gradle-user-home cache will cache dependencies, by running build with --offline | ||||||
|   dependencies-cache: |   dependencies-cache: | ||||||
|     needs: seed-build |     needs: seed-build | ||||||
|     runs-on: ubuntu-latest |     strategy: | ||||||
|  |       matrix: | ||||||
|  |         os: [ubuntu-latest, windows-latest] | ||||||
|  |     runs-on: ${{ matrix.os }} | ||||||
|     steps: |     steps: | ||||||
|     - name: Checkout sources |     - name: Checkout sources | ||||||
|       uses: actions/checkout@v2 |       uses: actions/checkout@v2 | ||||||
| @@ -40,7 +46,10 @@ jobs: | |||||||
|   # Test that the gradle-user-home cache will cache and restore local build-cache |   # Test that the gradle-user-home cache will cache and restore local build-cache | ||||||
|   build-cache: |   build-cache: | ||||||
|     needs: seed-build |     needs: seed-build | ||||||
|     runs-on: ubuntu-latest |     strategy: | ||||||
|  |       matrix: | ||||||
|  |         os: [ubuntu-latest, windows-latest] | ||||||
|  |     runs-on: ${{ matrix.os }} | ||||||
|     steps: |     steps: | ||||||
|     - name: Checkout sources |     - name: Checkout sources | ||||||
|       uses: actions/checkout@v2 |       uses: actions/checkout@v2 | ||||||
|   | |||||||
							
								
								
									
										14
									
								
								.github/workflows/integTest-kotlin-dsl.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										14
									
								
								.github/workflows/integTest-kotlin-dsl.yml
									
									
									
									
										vendored
									
									
								
							| @@ -6,13 +6,16 @@ on: | |||||||
|   workflow_dispatch: |   workflow_dispatch: | ||||||
|  |  | ||||||
| env: | env: | ||||||
|   CACHE_KEY_PREFIX: ${{github.workflow}}#${{github.run_number}}- |   GRADLE_BUILD_ACTION_CACHE_KEY_PREFIX: ${{github.workflow}}#${{github.run_number}}- | ||||||
|   CACHE_DEBUG_ENABLED: true |   GRADLE_BUILD_ACTION_CACHE_DEBUG_ENABLED: true | ||||||
|  |  | ||||||
| jobs: | jobs: | ||||||
|   # Use kotlin-dsl project to verify caching of generated jars and compiled scripts |   # Use kotlin-dsl project to verify caching of generated jars and compiled scripts | ||||||
|   seed-build: |   seed-build: | ||||||
|     runs-on: ubuntu-latest |     strategy: | ||||||
|  |       matrix: | ||||||
|  |         os: [ubuntu-latest, windows-latest] | ||||||
|  |     runs-on: ${{ matrix.os }} | ||||||
|     steps: |     steps: | ||||||
|     - name: Checkout sources |     - name: Checkout sources | ||||||
|       uses: actions/checkout@v2 |       uses: actions/checkout@v2 | ||||||
| @@ -25,7 +28,10 @@ jobs: | |||||||
|   # Check that the build can run --offline |   # Check that the build can run --offline | ||||||
|   verify-build: |   verify-build: | ||||||
|     needs: seed-build |     needs: seed-build | ||||||
|     runs-on: ubuntu-latest |     strategy: | ||||||
|  |       matrix: | ||||||
|  |         os: [ubuntu-latest, windows-latest] | ||||||
|  |     runs-on: ${{ matrix.os }} | ||||||
|     steps: |     steps: | ||||||
|     - name: Checkout sources |     - name: Checkout sources | ||||||
|       uses: actions/checkout@v2 |       uses: actions/checkout@v2 | ||||||
|   | |||||||
| @@ -1,5 +1,5 @@ | |||||||
| { | { | ||||||
|     "printWidth": 80, |     "printWidth": 120, | ||||||
|     "tabWidth": 4, |     "tabWidth": 4, | ||||||
|     "useTabs": false, |     "useTabs": false, | ||||||
|     "semi": false, |     "semi": false, | ||||||
|   | |||||||
							
								
								
									
										36
									
								
								README.md
									
									
									
									
									
								
							
							
						
						
									
										36
									
								
								README.md
									
									
									
									
									
								
							| @@ -2,9 +2,6 @@ | |||||||
|  |  | ||||||
| This GitHub Action can be used to execute a Gradle build on any platform supported by GitHub Actions. | This GitHub Action can be used to execute a Gradle build on any platform supported by GitHub Actions. | ||||||
|  |  | ||||||
| **Note:** The following documentation is for `gradle/gradle-build-action@v2`, currently in Beta release. |  | ||||||
| You can view the documentation for the latest stable release (v1.5.1) [on the GitHub Marketplace](https://github.com/marketplace/actions/gradle-build-action?version=v1.5.1).  |  | ||||||
|  |  | ||||||
| ## Usage | ## Usage | ||||||
|  |  | ||||||
| The following workflow will run `./gradlew build` on ubuntu, macos and windows.  | The following workflow will run `./gradlew build` on ubuntu, macos and windows.  | ||||||
| @@ -159,12 +156,9 @@ Caching is enabled by default. You can disable caching for the action as follows | |||||||
| cache-disabled: true | cache-disabled: true | ||||||
| ``` | ``` | ||||||
|  |  | ||||||
| At this time it is not possible to fine-tune the caching performed by this action.  |  | ||||||
| If you have a legitimate use case for fine-grained caching or restricting which files are cached, please raise an issue. |  | ||||||
|  |  | ||||||
| ### Cache keys | ### Cache keys | ||||||
|  |  | ||||||
| For cached distributions, the cache key is unique to the downloaded distribution. This will not change over time. | For cached distributions outside of Gradle User Home, the cache key is unique to the downloaded distribution. This will not change over time. | ||||||
|  |  | ||||||
| The state of the Gradle User Home and configuration-cache are highly dependent on the Gradle execution, so the cache key is composed of the current commit hash and the GitHub actions job id. | The state of the Gradle User Home and configuration-cache are highly dependent on the Gradle execution, so the cache key is composed of the current commit hash and the GitHub actions job id. | ||||||
| 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.  | ||||||
| @@ -183,16 +177,38 @@ For example, you may want to write cache entries for builds on your `main` branc | |||||||
| You can enable read-only caching for any of the caches as follows: | You can enable read-only caching for any of the caches as follows: | ||||||
|  |  | ||||||
| ```yaml | ```yaml | ||||||
| cache-read-only: true | # Only write to the cache for builds on the 'main' branch. | ||||||
|  | # Builds on other branches will only read existing entries from the cache. | ||||||
|  | cache-read-only: ${{ github.ref != 'refs/heads/main' }} | ||||||
| ``` | ``` | ||||||
|  |  | ||||||
|  | ### Gradle User Home cache tuning | ||||||
|  |  | ||||||
|  | As well as any wrapper distributions, the action will attempt to save and restore the `caches` and `notifications` directories from Gradle User Home. | ||||||
|  |  | ||||||
|  | The contents to be cached can be fine tuned by including and excluding certain paths with Gradle User Home. | ||||||
|  |  | ||||||
|  | ```yaml | ||||||
|  | # Cache downloaded JDKs in addition to the default directories. | ||||||
|  | gradle-home-cache-includes: | | ||||||
|  |     caches | ||||||
|  |     notifications | ||||||
|  |     jdks | ||||||
|  | # Exclude the local build-cache from the directories cached. | ||||||
|  | gradle-home-cache-excludes: | | ||||||
|  |     caches/build-cache-1 | ||||||
|  | ``` | ||||||
|  |  | ||||||
|  | You can specify any number of fixed paths or patterns to include or exclude.  | ||||||
|  | File pattern support is documented at https://docs.github.com/en/actions/learn-github-actions/workflow-syntax-for-github-actions#patterns-to-match-file-paths. | ||||||
|  |  | ||||||
| ### Cache debugging | ### Cache debugging | ||||||
|  |  | ||||||
| It is possible to enable additional debug logging for cache operations. You do via the `CACHE_DEBUG_ENABLED` environment variable: | It is possible to enable additional debug logging for cache operations. You do via the `GRADLE_BUILD_ACTION_CACHE_DEBUG_ENABLED` environment variable: | ||||||
|  |  | ||||||
| ```yaml | ```yaml | ||||||
| env: | env: | ||||||
|   CACHE_DEBUG_ENABLED: true |   GRADLE_BUILD_ACTION_CACHE_DEBUG_ENABLED: true | ||||||
| ``` | ``` | ||||||
|  |  | ||||||
| Note that this setting will also prevent certain cache operations from running in parallel, further assisting with debugging. | Note that this setting will also prevent certain cache operations from running in parallel, further assisting with debugging. | ||||||
|   | |||||||
							
								
								
									
										95
									
								
								__tests__/cache-base.test.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										95
									
								
								__tests__/cache-base.test.ts
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,95 @@ | |||||||
|  | import {CacheEntryListener, CacheListener} from '../src/cache-base' | ||||||
|  |  | ||||||
|  | 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,5 +1,4 @@ | |||||||
| import * as cacheUtils from '../src/cache-utils' | import * as cacheUtils from '../src/cache-utils' | ||||||
| import * as path from 'path' |  | ||||||
|  |  | ||||||
| describe('cacheUtils-utils', () => { | describe('cacheUtils-utils', () => { | ||||||
|     describe('can hash', () => { |     describe('can hash', () => { | ||||||
|   | |||||||
							
								
								
									
										46
									
								
								action.yml
									
									
									
									
									
								
							
							
						
						
									
										46
									
								
								action.yml
									
									
									
									
									
								
							| @@ -7,38 +7,64 @@ inputs: | |||||||
|   gradle-executable: |   gradle-executable: | ||||||
|     description: Path to the Gradle executable |     description: Path to the Gradle executable | ||||||
|     required: false |     required: false | ||||||
|  |  | ||||||
|   gradle-version: |   gradle-version: | ||||||
|     description: Gradle version to use |     description: Gradle version to use | ||||||
|     required: false |     required: false | ||||||
|  |  | ||||||
|   build-root-directory: |   build-root-directory: | ||||||
|     description: Path to the root directory of the build |     description: Path to the root directory of the build | ||||||
|     required: false |     required: false | ||||||
|  |  | ||||||
|   arguments: |   arguments: | ||||||
|     description: Gradle command line arguments (supports multi-line input) |     description: Gradle command line arguments (supports multi-line input) | ||||||
|     required: false |     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 | ||||||
|     default: false |     default: false | ||||||
|   cache-read-only: |  | ||||||
|     description: When 'true', existing entries will be read from the cache but no entries will be written |  | ||||||
|     required: false |  | ||||||
|     # TODO: It might be useful to default to read-only for PRs, or non-main branch. |  | ||||||
|     default: false  |  | ||||||
|  |  | ||||||
|  |   cache-read-only: | ||||||
|  |     description: When 'true', existing entries will be read from the cache but no entries will be written. | ||||||
|  |     required: false | ||||||
|  |     default: false  | ||||||
|  |   # e.g. Use the following setting to only write cache entries from your 'main' branch | ||||||
|  |   #     cache-read-only: ${{ github.ref != 'refs/heads/main' }} | ||||||
|  |  | ||||||
|  |   gradle-home-cache-includes: | ||||||
|  |     description: Paths within Gradle User Home to cache. | ||||||
|  |     required: false | ||||||
|  |     default: | | ||||||
|  |         caches | ||||||
|  |         notifications | ||||||
|  |  | ||||||
|  |   gradle-home-cache-excludes: | ||||||
|  |     description: Paths within Gradle User Home to exclude from cache. | ||||||
|  |     required: false | ||||||
|  |    | ||||||
|  |   # e.g. Use the following setting to prevent the local build cache from being saved/restored | ||||||
|  |   #      gradle-home-cache-excludes: | | ||||||
|  |   #           ["caches/build-cache-1"] | ||||||
|  |  | ||||||
|  |   # EXPERIMENTAL & INTERNAL CONFIGURATION PROPERTIES | ||||||
|  |   # The following action properties allow fine-grained tweaking of the action caching behaviour. | ||||||
|  |   # These properties are not designed for production use, and may change without notice in a subsequent release of `gradle-build-action`. | ||||||
|  |   # Use at your own risk! | ||||||
|   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. |     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 | ||||||
|     default: ${{ toJSON(matrix) }} |     default: ${{ toJSON(matrix) }} | ||||||
|   cache-artifact-bundles: |   gradle-home-cache-artifact-bundles: | ||||||
|     description: Names and patterns of artifact bundles to cache separately. For internal use only. |     description: Names and patterns of artifact bundles to cache separately. (EXPERIMENTAL - may be changed/removed without notice) | ||||||
|     required: false |     required: false | ||||||
|     default: | |     default: | | ||||||
|         [ |         [ | ||||||
|           ["generated-gradle-jars", "caches/*/generated-gradle-jars/*.jar"], |           ["generated-gradle-jars", "caches/*/generated-gradle-jars/*.jar"], | ||||||
|           ["wrapper-zips", "wrapper/dists/*/*/*.zip"], |           ["wrapper-zips", "wrapper/dists/*/*/*.zip"], | ||||||
|           ["dependency-jars", "caches/modules-*/files-*/**/*.jar"], |           ["dependencies", "caches/modules-*/files-*/*/*/*/*/"], | ||||||
|           ["instrumented-jars", "caches/jars-*/*"] |           ["instrumented-jars", "caches/jars-*/*/"], | ||||||
|  |           ["kotlin-dsl", "caches/*/kotlin-dsl/*/*/"] | ||||||
|         ] |         ] | ||||||
|  |  | ||||||
| outputs: | outputs: | ||||||
|   | |||||||
							
								
								
									
										2
									
								
								dist/main/index.js
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										2
									
								
								dist/main/index.js
									
									
									
									
										vendored
									
									
								
							
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							
							
								
								
									
										2
									
								
								dist/main/index.js.map
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										2
									
								
								dist/main/index.js.map
									
									
									
									
										vendored
									
									
								
							
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							
							
								
								
									
										2
									
								
								dist/post/index.js
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										2
									
								
								dist/post/index.js
									
									
									
									
										vendored
									
									
								
							
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							
							
								
								
									
										2
									
								
								dist/post/index.js.map
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										2
									
								
								dist/post/index.js.map
									
									
									
									
										vendored
									
									
								
							
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							
							
								
								
									
										7352
									
								
								package-lock.json
									
									
									
										generated
									
									
									
								
							
							
						
						
									
										7352
									
								
								package-lock.json
									
									
									
										generated
									
									
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							
							
								
								
									
										11
									
								
								package.json
									
									
									
									
									
								
							
							
						
						
									
										11
									
								
								package.json
									
									
									
									
									
								
							| @@ -4,6 +4,7 @@ | |||||||
|   "private": true, |   "private": true, | ||||||
|   "description": "Execute Gradle Build", |   "description": "Execute Gradle Build", | ||||||
|   "scripts": { |   "scripts": { | ||||||
|  |     "postinstall": "patch-package", | ||||||
|     "format": "prettier --write **/*.ts", |     "format": "prettier --write **/*.ts", | ||||||
|     "format-check": "prettier --check **/*.ts", |     "format-check": "prettier --check **/*.ts", | ||||||
|     "lint": "eslint src/**/*.ts", |     "lint": "eslint src/**/*.ts", | ||||||
| @@ -21,7 +22,6 @@ | |||||||
|     "github-actions", |     "github-actions", | ||||||
|     "gradle" |     "gradle" | ||||||
|   ], |   ], | ||||||
|   "author": "Paul Merlin <paul@nosphere.org>", |  | ||||||
|   "license": "MIT", |   "license": "MIT", | ||||||
|   "dependencies": { |   "dependencies": { | ||||||
|     "@actions/cache": "1.0.7", |     "@actions/cache": "1.0.7", | ||||||
| @@ -31,10 +31,11 @@ | |||||||
|     "@actions/glob": "0.2.0", |     "@actions/glob": "0.2.0", | ||||||
|     "@actions/http-client": "1.0.11", |     "@actions/http-client": "1.0.11", | ||||||
|     "@actions/tool-cache": "1.7.1", |     "@actions/tool-cache": "1.7.1", | ||||||
|  |     "patch-package": "6.4.7", | ||||||
|     "string-argv": "0.3.1" |     "string-argv": "0.3.1" | ||||||
|   }, |   }, | ||||||
|   "devDependencies": { |   "devDependencies": { | ||||||
|     "@types/jest": "26.0.23", |     "@types/jest": "27.0.2", | ||||||
|     "@types/node": "14.17.3", |     "@types/node": "14.17.3", | ||||||
|     "@types/unzipper": "0.10.4", |     "@types/unzipper": "0.10.4", | ||||||
|     "@typescript-eslint/parser": "4.28.2", |     "@typescript-eslint/parser": "4.28.2", | ||||||
| @@ -42,11 +43,11 @@ | |||||||
|     "eslint": "7.30.0", |     "eslint": "7.30.0", | ||||||
|     "eslint-plugin-github": "4.1.3", |     "eslint-plugin-github": "4.1.3", | ||||||
|     "eslint-plugin-jest": "24.3.6", |     "eslint-plugin-jest": "24.3.6", | ||||||
|     "jest": "26.6.3", |     "jest": "27.3.1", | ||||||
|     "jest-circus": "26.6.3", |     "jest-circus": "27.3.1", | ||||||
|     "js-yaml": "3.14.1", |     "js-yaml": "3.14.1", | ||||||
|     "prettier": "2.3.2", |     "prettier": "2.3.2", | ||||||
|     "ts-jest": "26.5.6", |     "ts-jest": "27.0.7", | ||||||
|     "typescript": "4.3.5" |     "typescript": "4.3.5" | ||||||
|   } |   } | ||||||
| } | } | ||||||
|   | |||||||
							
								
								
									
										92
									
								
								patches/@actions+cache+1.0.7.patch
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										92
									
								
								patches/@actions+cache+1.0.7.patch
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,92 @@ | |||||||
|  | diff --git a/node_modules/@actions/cache/lib/cache.d.ts b/node_modules/@actions/cache/lib/cache.d.ts | ||||||
|  | index 805a8e5..d3ab419 100644 | ||||||
|  | --- a/node_modules/@actions/cache/lib/cache.d.ts | ||||||
|  | +++ b/node_modules/@actions/cache/lib/cache.d.ts | ||||||
|  | @@ -5,6 +5,11 @@ export declare class ValidationError extends Error { | ||||||
|  |  export declare class ReserveCacheError extends Error { | ||||||
|  |      constructor(message: string); | ||||||
|  |  } | ||||||
|  | +export declare class CacheEntry { | ||||||
|  | +    key: string; | ||||||
|  | +    size?: number; | ||||||
|  | +    constructor(key: string, size?: number); | ||||||
|  | +} | ||||||
|  |  /** | ||||||
|  |   * Restores cache from keys | ||||||
|  |   * | ||||||
|  | @@ -14,7 +19,7 @@ export declare class ReserveCacheError extends Error { | ||||||
|  |   * @param downloadOptions cache download options | ||||||
|  |   * @returns string returns the key for the cache hit, otherwise returns undefined | ||||||
|  |   */ | ||||||
|  | -export declare function restoreCache(paths: string[], primaryKey: string, restoreKeys?: string[], options?: DownloadOptions): Promise<string | undefined>; | ||||||
|  | +export declare function restoreCache(paths: string[], primaryKey: string, restoreKeys?: string[], options?: DownloadOptions): Promise<CacheEntry | undefined>; | ||||||
|  |  /** | ||||||
|  |   * Saves a list of files with the specified key | ||||||
|  |   * | ||||||
|  | @@ -23,4 +28,4 @@ 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): Promise<number>; | ||||||
|  | +export declare function saveCache(paths: string[], key: string, options?: UploadOptions): Promise<CacheEntry>; | ||||||
|  | diff --git a/node_modules/@actions/cache/lib/cache.js b/node_modules/@actions/cache/lib/cache.js | ||||||
|  | index df78fe0..540114f 100644 | ||||||
|  | --- a/node_modules/@actions/cache/lib/cache.js | ||||||
|  | +++ b/node_modules/@actions/cache/lib/cache.js | ||||||
|  | @@ -37,6 +37,13 @@ class ReserveCacheError extends Error { | ||||||
|  |      } | ||||||
|  |  } | ||||||
|  |  exports.ReserveCacheError = ReserveCacheError; | ||||||
|  | +class CacheEntry { | ||||||
|  | +    constructor(key, size) { | ||||||
|  | +        this.key = key; | ||||||
|  | +        this.size = size; | ||||||
|  | +    } | ||||||
|  | +} | ||||||
|  | +exports.CacheEntry = CacheEntry; | ||||||
|  |  function checkPaths(paths) { | ||||||
|  |      if (!paths || paths.length === 0) { | ||||||
|  |          throw new ValidationError(`Path Validation Error: At least one directory or file path is required`); | ||||||
|  | @@ -84,6 +91,7 @@ function restoreCache(paths, primaryKey, restoreKeys, options) { | ||||||
|  |          } | ||||||
|  |          const archivePath = path.join(yield utils.createTempDirectory(), utils.getCacheFileName(compressionMethod)); | ||||||
|  |          core.debug(`Archive Path: ${archivePath}`); | ||||||
|  | +        const restoredEntry = new CacheEntry(cacheEntry.cacheKey); | ||||||
|  |          try { | ||||||
|  |              // Download the cache from the cache entry | ||||||
|  |              yield cacheHttpClient.downloadCache(cacheEntry.archiveLocation, archivePath, options); | ||||||
|  | @@ -91,6 +99,7 @@ function restoreCache(paths, primaryKey, restoreKeys, options) { | ||||||
|  |                  yield tar_1.listTar(archivePath, compressionMethod); | ||||||
|  |              } | ||||||
|  |              const archiveFileSize = utils.getArchiveFileSizeIsBytes(archivePath); | ||||||
|  | +            restoredEntry.size = archiveFileSize; | ||||||
|  |              core.info(`Cache Size: ~${Math.round(archiveFileSize / (1024 * 1024))} MB (${archiveFileSize} B)`); | ||||||
|  |              yield tar_1.extractTar(archivePath, compressionMethod); | ||||||
|  |              core.info('Cache restored successfully'); | ||||||
|  | @@ -104,7 +113,7 @@ function restoreCache(paths, primaryKey, restoreKeys, options) { | ||||||
|  |                  core.debug(`Failed to delete archive: ${error}`); | ||||||
|  |              } | ||||||
|  |          } | ||||||
|  | -        return cacheEntry.cacheKey; | ||||||
|  | +        return restoredEntry; | ||||||
|  |      }); | ||||||
|  |  } | ||||||
|  |  exports.restoreCache = restoreCache; | ||||||
|  | @@ -147,7 +156,7 @@ function saveCache(paths, key, options) { | ||||||
|  |          } | ||||||
|  |          core.debug(`Saving Cache (ID: ${cacheId})`); | ||||||
|  |          yield cacheHttpClient.saveCache(cacheId, archivePath, options); | ||||||
|  | -        return cacheId; | ||||||
|  | +        return new CacheEntry(key, archiveFileSize); | ||||||
|  |      }); | ||||||
|  |  } | ||||||
|  |  exports.saveCache = saveCache; | ||||||
|  | diff --git a/node_modules/@actions/cache/lib/cache.js.map b/node_modules/@actions/cache/lib/cache.js.map | ||||||
|  | index 05fc369..41b9189 100644 | ||||||
|  | --- a/node_modules/@actions/cache/lib/cache.js.map | ||||||
|  | +++ b/node_modules/@actions/cache/lib/cache.js.map | ||||||
|  | @@ -1 +1 @@ | ||||||
|  | -{"version":3,"file":"cache.js","sourceRoot":"","sources":["../src/cache.ts"],"names":[],"mappings":"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"} | ||||||
|  | \ No newline at end of file | ||||||
|  | +{"version":3,"file":"cache.js","sourceRoot":"","sources":["../src/cache.ts"],"names":[],"mappings":"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"} | ||||||
|  | \ No newline at end of file | ||||||
							
								
								
									
										258
									
								
								src/cache-base.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										258
									
								
								src/cache-base.ts
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,258 @@ | |||||||
|  | import * as core from '@actions/core' | ||||||
|  | import * as cache from '@actions/cache' | ||||||
|  | import * as github from '@actions/github' | ||||||
|  | import {isCacheDebuggingEnabled, getCacheKeyPrefix, hashStrings} from './cache-utils' | ||||||
|  |  | ||||||
|  | const JOB_CONTEXT_PARAMETER = 'workflow-job-context' | ||||||
|  |  | ||||||
|  | function generateCacheKey(cacheName: string): CacheKey { | ||||||
|  |     const cacheKeyPrefix = getCacheKeyPrefix() | ||||||
|  |  | ||||||
|  |     // At the most general level, share caches for all executions on the same OS | ||||||
|  |     const runnerOs = process.env['RUNNER_OS'] || '' | ||||||
|  |     const cacheKeyForOs = `${cacheKeyPrefix}${cacheName}|${runnerOs}` | ||||||
|  |  | ||||||
|  |     // Prefer caches that run this job | ||||||
|  |     const cacheKeyForJob = `${cacheKeyForOs}|${github.context.job}` | ||||||
|  |  | ||||||
|  |     // Prefer (even more) jobs that run this job with the same context (matrix) | ||||||
|  |     const cacheKeyForJobContext = `${cacheKeyForJob}[${determineJobContext()}]` | ||||||
|  |  | ||||||
|  |     // Exact match on Git SHA | ||||||
|  |     const cacheKey = `${cacheKeyForJobContext}-${github.context.sha}` | ||||||
|  |  | ||||||
|  |     return new CacheKey(cacheKey, [cacheKeyForJobContext, cacheKeyForJob, cacheKeyForOs]) | ||||||
|  | } | ||||||
|  |  | ||||||
|  | function determineJobContext(): string { | ||||||
|  |     // By default, we hash the full `matrix` data for the run, to uniquely identify this job invocation | ||||||
|  |     const workflowJobContext = core.getInput(JOB_CONTEXT_PARAMETER) | ||||||
|  |     return hashStrings([workflowJobContext]) | ||||||
|  | } | ||||||
|  |  | ||||||
|  | class CacheKey { | ||||||
|  |     key: string | ||||||
|  |     restoreKeys: string[] | ||||||
|  |  | ||||||
|  |     constructor(key: string, restoreKeys: string[]) { | ||||||
|  |         this.key = key | ||||||
|  |         this.restoreKeys = restoreKeys | ||||||
|  |     } | ||||||
|  | } | ||||||
|  |  | ||||||
|  | export class CacheListener { | ||||||
|  |     cacheEntries: CacheEntryListener[] = [] | ||||||
|  |  | ||||||
|  |     get fullyRestored(): boolean { | ||||||
|  |         return this.cacheEntries.every(x => !x.wasRequestedButNotRestored()) | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     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 { | ||||||
|  |         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 | ||||||
|  |     } | ||||||
|  | } | ||||||
|  |  | ||||||
|  | export class CacheEntryListener { | ||||||
|  |     entryName: string | ||||||
|  |     requestedKey: string | undefined | ||||||
|  |     requestedRestoreKeys: string[] | undefined | ||||||
|  |     restoredKey: string | undefined | ||||||
|  |     restoredSize: number | undefined | ||||||
|  |  | ||||||
|  |     savedKey: string | undefined | ||||||
|  |     savedSize: number | 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 | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     markSaved(key: string, size: number | undefined): CacheEntryListener { | ||||||
|  |         this.savedKey = key | ||||||
|  |         this.savedSize = size | ||||||
|  |         return this | ||||||
|  |     } | ||||||
|  | } | ||||||
|  |  | ||||||
|  | export abstract class AbstractCache { | ||||||
|  |     private cacheName: string | ||||||
|  |     private cacheDescription: string | ||||||
|  |     private cacheKeyStateKey: string | ||||||
|  |     private cacheResultStateKey: string | ||||||
|  |  | ||||||
|  |     protected readonly cacheDebuggingEnabled: boolean | ||||||
|  |  | ||||||
|  |     constructor(cacheName: string, cacheDescription: string) { | ||||||
|  |         this.cacheName = cacheName | ||||||
|  |         this.cacheDescription = cacheDescription | ||||||
|  |         this.cacheKeyStateKey = `CACHE_KEY_${cacheName}` | ||||||
|  |         this.cacheResultStateKey = `CACHE_RESULT_${cacheName}` | ||||||
|  |         this.cacheDebuggingEnabled = isCacheDebuggingEnabled() | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     async restore(listener: CacheListener): Promise<void> { | ||||||
|  |         if (this.cacheOutputExists()) { | ||||||
|  |             core.info(`${this.cacheDescription} already exists. Not restoring from cache.`) | ||||||
|  |             return | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         const cacheKey = this.prepareCacheKey() | ||||||
|  |         const entryReport = listener.entry(this.cacheDescription) | ||||||
|  |         entryReport.markRequested(cacheKey.key, cacheKey.restoreKeys) | ||||||
|  |  | ||||||
|  |         this.debug( | ||||||
|  |             `Requesting ${this.cacheDescription} with | ||||||
|  |                 key:${cacheKey.key} | ||||||
|  |                 restoreKeys:[${cacheKey.restoreKeys}]` | ||||||
|  |         ) | ||||||
|  |  | ||||||
|  |         const cacheResult = await this.restoreCache(this.getCachePath(), cacheKey.key, cacheKey.restoreKeys) | ||||||
|  |  | ||||||
|  |         if (!cacheResult) { | ||||||
|  |             core.info(`${this.cacheDescription} cache not found. Will start with empty.`) | ||||||
|  |             return | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         core.saveState(this.cacheResultStateKey, cacheResult) | ||||||
|  |         entryReport.markRestored(cacheResult.key, cacheResult.size) | ||||||
|  |         core.info(`Restored ${this.cacheDescription} from cache key: ${cacheResult}`) | ||||||
|  |  | ||||||
|  |         try { | ||||||
|  |             await this.afterRestore(listener) | ||||||
|  |         } catch (error) { | ||||||
|  |             core.warning(`Restore ${this.cacheDescription} failed in 'afterRestore': ${error}`) | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     prepareCacheKey(): CacheKey { | ||||||
|  |         const cacheKey = generateCacheKey(this.cacheName) | ||||||
|  |  | ||||||
|  |         core.saveState(this.cacheKeyStateKey, cacheKey.key) | ||||||
|  |         return cacheKey | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     protected async restoreCache( | ||||||
|  |         cachePath: string[], | ||||||
|  |         cacheKey: string, | ||||||
|  |         cacheRestoreKeys: string[] = [] | ||||||
|  |     ): Promise<cache.CacheEntry | undefined> { | ||||||
|  |         try { | ||||||
|  |             return await cache.restoreCache(cachePath, cacheKey, cacheRestoreKeys) | ||||||
|  |         } catch (error) { | ||||||
|  |             if (error instanceof cache.ValidationError) { | ||||||
|  |                 // Validation errors should fail the build action | ||||||
|  |                 throw error | ||||||
|  |             } | ||||||
|  |             // Warn about any other error and continue | ||||||
|  |             core.warning(`Failed to restore ${cacheKey}: ${error}`) | ||||||
|  |             return undefined | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     protected async afterRestore(_listener: CacheListener): Promise<void> {} | ||||||
|  |  | ||||||
|  |     async save(listener: CacheListener): Promise<void> { | ||||||
|  |         if (!this.cacheOutputExists()) { | ||||||
|  |             this.debug(`No ${this.cacheDescription} to cache.`) | ||||||
|  |             return | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         const cacheKey = core.getState(this.cacheKeyStateKey) | ||||||
|  |         const cacheResult = core.getState(this.cacheResultStateKey) | ||||||
|  |  | ||||||
|  |         if (!cacheKey) { | ||||||
|  |             this.debug(`${this.cacheDescription} existed prior to cache restore. Not saving.`) | ||||||
|  |             return | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         if (cacheResult && cacheKey === cacheResult) { | ||||||
|  |             core.info(`Cache hit occurred on the cache key ${cacheKey}, not saving cache.`) | ||||||
|  |             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() | ||||||
|  |         const savedEntry = await this.saveCache(cachePath, cacheKey) | ||||||
|  |  | ||||||
|  |         if (savedEntry) { | ||||||
|  |             listener.entry(this.cacheDescription).markSaved(savedEntry.key, savedEntry.size) | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         return | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     protected async beforeSave(_listener: CacheListener): Promise<void> {} | ||||||
|  |  | ||||||
|  |     protected async saveCache(cachePath: string[], cacheKey: string): Promise<cache.CacheEntry | undefined> { | ||||||
|  |         try { | ||||||
|  |             return await cache.saveCache(cachePath, cacheKey) | ||||||
|  |         } catch (error) { | ||||||
|  |             if (error instanceof cache.ValidationError) { | ||||||
|  |                 // Validation errors should fail the build action | ||||||
|  |                 throw error | ||||||
|  |             } else if (error instanceof cache.ReserveCacheError) { | ||||||
|  |                 // Reserve cache errors are expected if the artifact has been previously cached | ||||||
|  |                 this.debug(error.message) | ||||||
|  |             } else { | ||||||
|  |                 // Warn about any other error and continue | ||||||
|  |                 core.warning(String(error)) | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |         return undefined | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     protected debug(message: string): void { | ||||||
|  |         if (this.cacheDebuggingEnabled) { | ||||||
|  |             core.info(message) | ||||||
|  |         } else { | ||||||
|  |             core.debug(message) | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     protected abstract cacheOutputExists(): boolean | ||||||
|  |     protected abstract getCachePath(): string[] | ||||||
|  | } | ||||||
| @@ -5,10 +5,14 @@ import * as core from '@actions/core' | |||||||
| import * as glob from '@actions/glob' | import * as glob from '@actions/glob' | ||||||
| import * as exec from '@actions/exec' | import * as exec from '@actions/exec' | ||||||
|  |  | ||||||
| import {AbstractCache, hashFileNames, tryDelete} from './cache-utils' | import {AbstractCache, CacheEntryListener, CacheListener} from './cache-base' | ||||||
|  | import {getCacheKeyPrefix, hashFileNames, tryDelete} from './cache-utils' | ||||||
|  |  | ||||||
| // Which paths under Gradle User Home should be cached | const META_FILE_DIR = '.gradle-build-action' | ||||||
| const CACHE_PATH = ['caches', 'notifications'] |  | ||||||
|  | const INCLUDE_PATHS_PARAMETER = 'gradle-home-cache-includes' | ||||||
|  | const EXCLUDE_PATHS_PARAMETER = 'gradle-home-cache-excludes' | ||||||
|  | const ARTIFACT_BUNDLES_PARAMETER = 'gradle-home-cache-artifact-bundles' | ||||||
|  |  | ||||||
| export class GradleUserHomeCache extends AbstractCache { | export class GradleUserHomeCache extends AbstractCache { | ||||||
|     private gradleUserHome: string |     private gradleUserHome: string | ||||||
| @@ -18,68 +22,97 @@ export class GradleUserHomeCache extends AbstractCache { | |||||||
|         this.gradleUserHome = this.determineGradleUserHome(rootDir) |         this.gradleUserHome = this.determineGradleUserHome(rootDir) | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     async afterRestore(): Promise<void> { |     async afterRestore(listener: CacheListener): Promise<void> { | ||||||
|         await this.reportGradleUserHomeSize('as restored from cache') |         await this.reportGradleUserHomeSize('as restored from cache') | ||||||
|         await this.restoreArtifactBundles() |         await this.restoreArtifactBundles(listener) | ||||||
|         await this.reportGradleUserHomeSize('after restoring common artifacts') |         await this.reportGradleUserHomeSize('after restoring common artifacts') | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     private async restoreArtifactBundles(): Promise<void> { |     private async restoreArtifactBundles(listener: CacheListener): Promise<void> { | ||||||
|         const processes: Promise<void>[] = [] |         const processes: Promise<void>[] = [] | ||||||
|         for (const [bundle, pattern] of this.getArtifactBundles()) { |  | ||||||
|             const p = this.restoreArtifactBundle(bundle, pattern) |         const bundleMetaFiles = await this.getBundleMetaFiles() | ||||||
|  |         const bundlePatterns = this.getArtifactBundles() | ||||||
|  |  | ||||||
|  |         // Iterate over all bundle meta files and try to restore | ||||||
|  |         for (const bundleMetaFile of bundleMetaFiles) { | ||||||
|  |             const bundle = path.basename(bundleMetaFile, '.cache') | ||||||
|  |             const entryListener = listener.entry(bundle) | ||||||
|  |             const bundlePattern = bundlePatterns.get(bundle) | ||||||
|  |  | ||||||
|  |             // Handle case where the 'artifactBundlePatterns' have been changed | ||||||
|  |             if (bundlePattern === undefined) { | ||||||
|  |                 core.info(`Found bundle metafile for ${bundle} but no such bundle defined`) | ||||||
|  |                 entryListener.markRequested('BUNDLE_NOT_CONFIGURED') | ||||||
|  |                 tryDelete(bundleMetaFile) | ||||||
|  |             } else { | ||||||
|  |                 const p = this.restoreArtifactBundle(bundle, bundlePattern, bundleMetaFile, entryListener) | ||||||
|                 // Run sequentially when debugging enabled |                 // Run sequentially when debugging enabled | ||||||
|                 if (this.cacheDebuggingEnabled) { |                 if (this.cacheDebuggingEnabled) { | ||||||
|                     await p |                     await p | ||||||
|                 } |                 } | ||||||
|                 processes.push(p) |                 processes.push(p) | ||||||
|             } |             } | ||||||
|  |         } | ||||||
|  |  | ||||||
|         await Promise.all(processes) |         await Promise.all(processes) | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     private async restoreArtifactBundle( |     private async restoreArtifactBundle( | ||||||
|         bundle: string, |         bundle: string, | ||||||
|         artifactPath: string |         bundlePattern: string, | ||||||
|  |         bundleMetaFile: string, | ||||||
|  |         listener: CacheEntryListener | ||||||
|     ): Promise<void> { |     ): Promise<void> { | ||||||
|         const bundleMetaFile = this.getBundleMetaFile(bundle) |  | ||||||
|         if (fs.existsSync(bundleMetaFile)) { |  | ||||||
|         const cacheKey = fs.readFileSync(bundleMetaFile, 'utf-8').trim() |         const cacheKey = fs.readFileSync(bundleMetaFile, 'utf-8').trim() | ||||||
|             const restoreKey = await this.restoreCache([artifactPath], cacheKey) |         listener.markRequested(cacheKey) | ||||||
|             if (restoreKey) { |  | ||||||
|                 core.info( |         const restoredEntry = await this.restoreCache([bundlePattern], cacheKey) | ||||||
|                     `Restored ${bundle} with key ${cacheKey} to ${artifactPath}` |         if (restoredEntry) { | ||||||
|                 ) |             core.info(`Restored ${bundle} with key ${cacheKey} to ${bundlePattern}`) | ||||||
|  |             listener.markRestored(restoredEntry.key, restoredEntry.size) | ||||||
|         } else { |         } else { | ||||||
|                 this.debug( |             core.info(`Did not restore ${bundle} with key ${cacheKey} to ${bundlePattern}`) | ||||||
|                     `Did not restore ${bundle} with key ${cacheKey} to ${artifactPath}` |             tryDelete(bundleMetaFile) | ||||||
|                 ) |  | ||||||
|             } |  | ||||||
|         } else { |  | ||||||
|             this.debug( |  | ||||||
|                 `No metafile found to restore ${bundle}: ${bundleMetaFile}` |  | ||||||
|             ) |  | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     private getBundleMetaFile(name: string): string { |     private getBundleMetaFile(name: string): string { | ||||||
|         return path.resolve( |         return path.resolve(this.gradleUserHome, META_FILE_DIR, `${name}.cache`) | ||||||
|             this.gradleUserHome, |     } | ||||||
|             'caches', |  | ||||||
|             `.gradle-build-action.${name}.cache` |     private async getBundleMetaFiles(): Promise<string[]> { | ||||||
|  |         const metaFiles = path.resolve(this.gradleUserHome, META_FILE_DIR, '*.cache') | ||||||
|  |         const globber = await glob.create(metaFiles) | ||||||
|  |         const bundleFiles = await globber.glob() | ||||||
|  |         return bundleFiles | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     async beforeSave(listener: CacheListener): Promise<void> { | ||||||
|  |         await this.reportGradleUserHomeSize('before saving common artifacts') | ||||||
|  |         this.removeExcludedPaths() | ||||||
|  |         await this.saveArtifactBundles(listener) | ||||||
|  |         await this.reportGradleUserHomeSize( | ||||||
|  |             "after saving common artifacts (only 'caches' and 'notifications' will be stored)" | ||||||
|         ) |         ) | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     async beforeSave(): Promise<void> { |     private removeExcludedPaths(): void { | ||||||
|         await this.reportGradleUserHomeSize('before saving common artifacts') |         const rawPaths: string[] = core.getMultilineInput(EXCLUDE_PATHS_PARAMETER) | ||||||
|         await this.saveArtifactBundles() |         const resolvedPaths = rawPaths.map(x => path.resolve(this.gradleUserHome, x)) | ||||||
|         await this.reportGradleUserHomeSize('after saving common artifacts') |  | ||||||
|  |         for (const p of resolvedPaths) { | ||||||
|  |             this.debug(`Deleting excluded path: ${p}`) | ||||||
|  |             tryDelete(p) | ||||||
|  |         } | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     private async saveArtifactBundles(): Promise<void> { |     private async saveArtifactBundles(listener: CacheListener): Promise<void> { | ||||||
|         const processes: Promise<void>[] = [] |         const processes: Promise<void>[] = [] | ||||||
|         for (const [bundle, pattern] of this.getArtifactBundles()) { |         for (const [bundle, pattern] of this.getArtifactBundles()) { | ||||||
|             const p = this.saveArtifactBundle(bundle, pattern) |             const entryListener = listener.entry(bundle) | ||||||
|  |  | ||||||
|  |             const p = this.saveArtifactBundle(bundle, pattern, entryListener) | ||||||
|             // Run sequentially when debugging enabled |             // Run sequentially when debugging enabled | ||||||
|             if (this.cacheDebuggingEnabled) { |             if (this.cacheDebuggingEnabled) { | ||||||
|                 await p |                 await p | ||||||
| @@ -92,11 +125,15 @@ export class GradleUserHomeCache extends AbstractCache { | |||||||
|  |  | ||||||
|     private async saveArtifactBundle( |     private async saveArtifactBundle( | ||||||
|         bundle: string, |         bundle: string, | ||||||
|         artifactPath: string |         artifactPath: string, | ||||||
|  |         listener: CacheEntryListener | ||||||
|     ): Promise<void> { |     ): Promise<void> { | ||||||
|         const bundleMetaFile = this.getBundleMetaFile(bundle) |         const bundleMetaFile = this.getBundleMetaFile(bundle) | ||||||
|  |  | ||||||
|         const globber = await glob.create(artifactPath) |         const globber = await glob.create(artifactPath, { | ||||||
|  |             implicitDescendants: false, | ||||||
|  |             followSymbolicLinks: false | ||||||
|  |         }) | ||||||
|         const bundleFiles = await globber.glob() |         const bundleFiles = await globber.glob() | ||||||
|  |  | ||||||
|         // Handle no matching files |         // Handle no matching files | ||||||
| @@ -114,15 +151,14 @@ export class GradleUserHomeCache extends AbstractCache { | |||||||
|         const cacheKey = this.createCacheKey(bundle, bundleFiles) |         const cacheKey = this.createCacheKey(bundle, bundleFiles) | ||||||
|  |  | ||||||
|         if (previouslyRestoredKey === cacheKey) { |         if (previouslyRestoredKey === cacheKey) { | ||||||
|             this.debug( |             this.debug(`No change to previously restored ${bundle}. Not caching.`) | ||||||
|                 `No change to previously restored ${bundle}. Not caching.` |  | ||||||
|             ) |  | ||||||
|         } else { |         } else { | ||||||
|             core.info(`Caching ${bundle} with cache key: ${cacheKey}`) |             core.info(`Caching ${bundle} with cache key: ${cacheKey}`) | ||||||
|             await this.saveCache([artifactPath], cacheKey) |             const savedEntry = await this.saveCache([artifactPath], cacheKey) | ||||||
|  |             if (savedEntry !== undefined) { | ||||||
|             this.debug(`Writing cache metafile: ${bundleMetaFile}`) |                 this.writeBundleMetaFile(bundleMetaFile, cacheKey) | ||||||
|             fs.writeFileSync(bundleMetaFile, cacheKey) |                 listener.markSaved(savedEntry.key, savedEntry.size) | ||||||
|  |             } | ||||||
|         } |         } | ||||||
|  |  | ||||||
|         for (const file of bundleFiles) { |         for (const file of bundleFiles) { | ||||||
| @@ -131,19 +167,26 @@ export class GradleUserHomeCache extends AbstractCache { | |||||||
|     } |     } | ||||||
|  |  | ||||||
|     protected createCacheKey(bundle: string, files: string[]): string { |     protected createCacheKey(bundle: string, files: string[]): string { | ||||||
|         const cacheKeyPrefix = process.env['CACHE_KEY_PREFIX'] || '' |         const cacheKeyPrefix = getCacheKeyPrefix() | ||||||
|         const relativeFiles = files.map(x => |         const relativeFiles = files.map(x => path.relative(this.gradleUserHome, x)) | ||||||
|             path.relative(this.gradleUserHome, x) |  | ||||||
|         ) |  | ||||||
|         const key = hashFileNames(relativeFiles) |         const key = hashFileNames(relativeFiles) | ||||||
|  |  | ||||||
|         this.debug( |         this.debug(`Generating cache key for ${bundle} from files: ${relativeFiles}`) | ||||||
|             `Generating cache key for ${bundle} from files: ${relativeFiles}` |  | ||||||
|         ) |  | ||||||
|  |  | ||||||
|         return `${cacheKeyPrefix}${bundle}-${key}` |         return `${cacheKeyPrefix}${bundle}-${key}` | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     private writeBundleMetaFile(metaFile: string, cacheKey: string): void { | ||||||
|  |         this.debug(`Writing bundle metafile: ${metaFile}`) | ||||||
|  |  | ||||||
|  |         const dirName = path.dirname(metaFile) | ||||||
|  |         if (!fs.existsSync(dirName)) { | ||||||
|  |             fs.mkdirSync(dirName) | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         fs.writeFileSync(metaFile, cacheKey) | ||||||
|  |     } | ||||||
|  |  | ||||||
|     protected determineGradleUserHome(rootDir: string): string { |     protected determineGradleUserHome(rootDir: string): string { | ||||||
|         const customGradleUserHome = process.env['GRADLE_USER_HOME'] |         const customGradleUserHome = process.env['GRADLE_USER_HOME'] | ||||||
|         if (customGradleUserHome) { |         if (customGradleUserHome) { | ||||||
| @@ -160,21 +203,26 @@ export class GradleUserHomeCache extends AbstractCache { | |||||||
|     } |     } | ||||||
|  |  | ||||||
|     protected getCachePath(): string[] { |     protected getCachePath(): string[] { | ||||||
|         return CACHE_PATH.map(x => path.resolve(this.gradleUserHome, x)) |         const rawPaths: string[] = core.getMultilineInput(INCLUDE_PATHS_PARAMETER) | ||||||
|  |         rawPaths.push(META_FILE_DIR) | ||||||
|  |         const resolvedPaths = rawPaths.map(x => this.resolveCachePath(x)) | ||||||
|  |         this.debug(`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 getArtifactBundles(): Map<string, string> { |     private getArtifactBundles(): Map<string, string> { | ||||||
|         const artifactBundleDefinition = core.getInput('cache-artifact-bundles') |         const artifactBundleDefinition = core.getInput(ARTIFACT_BUNDLES_PARAMETER) | ||||||
|         this.debug( |         this.debug(`Using artifact bundle definition: ${artifactBundleDefinition}`) | ||||||
|             `Using artifact bundle definition: ${artifactBundleDefinition}` |  | ||||||
|         ) |  | ||||||
|         const artifactBundles = JSON.parse(artifactBundleDefinition) |         const artifactBundles = JSON.parse(artifactBundleDefinition) | ||||||
|         return new Map( |         return new Map(Array.from(artifactBundles, ([key, value]) => [key, path.resolve(this.gradleUserHome, value)])) | ||||||
|             Array.from(artifactBundles, ([key, value]) => [ |  | ||||||
|                 key, |  | ||||||
|                 path.resolve(this.gradleUserHome, value) |  | ||||||
|             ]) |  | ||||||
|         ) |  | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     private async reportGradleUserHomeSize(label: string): Promise<void> { |     private async reportGradleUserHomeSize(label: string): Promise<void> { | ||||||
| @@ -184,17 +232,13 @@ export class GradleUserHomeCache extends AbstractCache { | |||||||
|         if (!fs.existsSync(this.gradleUserHome)) { |         if (!fs.existsSync(this.gradleUserHome)) { | ||||||
|             return |             return | ||||||
|         } |         } | ||||||
|         const result = await exec.getExecOutput( |         const result = await exec.getExecOutput('du', ['-h', '-c', '-t', '5M'], { | ||||||
|             'du', |  | ||||||
|             ['-h', '-c', '-t', '5M'], |  | ||||||
|             { |  | ||||||
|             cwd: this.gradleUserHome, |             cwd: this.gradleUserHome, | ||||||
|             silent: true, |             silent: true, | ||||||
|             ignoreReturnCode: true |             ignoreReturnCode: true | ||||||
|             } |         }) | ||||||
|         ) |  | ||||||
|  |  | ||||||
|         core.info(`Gradle User Home cache entry (directories >5M): ${label}`) |         core.info(`Gradle User Home (directories >5M): ${label}`) | ||||||
|  |  | ||||||
|         core.info( |         core.info( | ||||||
|             result.stdout |             result.stdout | ||||||
|   | |||||||
| @@ -1,6 +1,6 @@ | |||||||
| import path from 'path' | import path from 'path' | ||||||
| import fs from 'fs' | import fs from 'fs' | ||||||
| import {AbstractCache} from './cache-utils' | import {AbstractCache} from './cache-base' | ||||||
|  |  | ||||||
| // TODO: Maybe allow the user to override / tweak this set | // TODO: Maybe allow the user to override / tweak this set | ||||||
| const PATHS_TO_CACHE = [ | const PATHS_TO_CACHE = [ | ||||||
| @@ -10,7 +10,7 @@ const PATHS_TO_CACHE = [ | |||||||
| export class ProjectDotGradleCache extends AbstractCache { | export class ProjectDotGradleCache extends AbstractCache { | ||||||
|     private rootDir: string |     private rootDir: string | ||||||
|     constructor(rootDir: string) { |     constructor(rootDir: string) { | ||||||
|         super('project', 'Project .gradle directory') |         super('project', 'Project configuration cache') | ||||||
|         this.rootDir = rootDir |         this.rootDir = rootDir | ||||||
|     } |     } | ||||||
|  |  | ||||||
|   | |||||||
| @@ -1,50 +1,30 @@ | |||||||
| import * as core from '@actions/core' | import * as core from '@actions/core' | ||||||
| import * as cache from '@actions/cache' |  | ||||||
| import * as github from '@actions/github' |  | ||||||
| import * as crypto from 'crypto' | import * as crypto from 'crypto' | ||||||
| import * as path from 'path' | import * as path from 'path' | ||||||
| import * as fs from 'fs' | import * as fs from 'fs' | ||||||
|  |  | ||||||
|  | const CACHE_PROTOCOL_VERSION = 'v4-' | ||||||
|  |  | ||||||
|  | const CACHE_DISABLED_PARAMETER = 'cache-disabled' | ||||||
|  | const CACHE_READONLY_PARAMETER = 'cache-read-only' | ||||||
|  | const CACHE_DEBUG_VAR = 'GRADLE_BUILD_ACTION_CACHE_DEBUG_ENABLED' | ||||||
|  | const CACHE_PREFIX_VAR = 'GRADLE_BUILD_ACTION_CACHE_KEY_PREFIX' | ||||||
|  |  | ||||||
| export function isCacheDisabled(): boolean { | export function isCacheDisabled(): boolean { | ||||||
|     return core.getBooleanInput('cache-disabled') |     return core.getBooleanInput(CACHE_DISABLED_PARAMETER) | ||||||
| } | } | ||||||
|  |  | ||||||
| export function isCacheReadOnly(): boolean { | export function isCacheReadOnly(): boolean { | ||||||
|     return core.getBooleanInput('cache-read-only') |     return core.getBooleanInput(CACHE_READONLY_PARAMETER) | ||||||
| } | } | ||||||
|  |  | ||||||
| export function isCacheDebuggingEnabled(): boolean { | export function isCacheDebuggingEnabled(): boolean { | ||||||
|     return process.env['CACHE_DEBUG_ENABLED'] ? true : false |     return process.env[CACHE_DEBUG_VAR] ? true : false | ||||||
| } | } | ||||||
|  |  | ||||||
| function generateCacheKey(cacheName: string): CacheKey { | export function getCacheKeyPrefix(): string { | ||||||
|     // Prefix can be used to force change all cache keys (defaults to cache protocol version) |     // Prefix can be used to force change all cache keys (defaults to cache protocol version) | ||||||
|     const cacheKeyPrefix = process.env['CACHE_KEY_PREFIX'] || 'v2-' |     return process.env[CACHE_PREFIX_VAR] || CACHE_PROTOCOL_VERSION | ||||||
|  |  | ||||||
|     // At the most general level, share caches for all executions on the same OS |  | ||||||
|     const runnerOs = process.env['RUNNER_OS'] || '' |  | ||||||
|     const cacheKeyForOs = `${cacheKeyPrefix}${cacheName}|${runnerOs}` |  | ||||||
|  |  | ||||||
|     // Prefer caches that run this job |  | ||||||
|     const cacheKeyForJob = `${cacheKeyForOs}|${github.context.job}` |  | ||||||
|  |  | ||||||
|     // Prefer (even more) jobs that run this job with the same context (matrix) |  | ||||||
|     const cacheKeyForJobContext = `${cacheKeyForJob}[${determineJobContext()}]` |  | ||||||
|  |  | ||||||
|     // Exact match on Git SHA |  | ||||||
|     const cacheKey = `${cacheKeyForJobContext}-${github.context.sha}` |  | ||||||
|  |  | ||||||
|     return new CacheKey(cacheKey, [ |  | ||||||
|         cacheKeyForJobContext, |  | ||||||
|         cacheKeyForJob, |  | ||||||
|         cacheKeyForOs |  | ||||||
|     ]) |  | ||||||
| } |  | ||||||
|  |  | ||||||
| function determineJobContext(): string { |  | ||||||
|     // By default, we hash the full `matrix` data for the run, to uniquely identify this job invocation |  | ||||||
|     const workflowJobContext = core.getInput('workflow-job-context') |  | ||||||
|     return hashStrings([workflowJobContext]) |  | ||||||
| } | } | ||||||
|  |  | ||||||
| export function hashStrings(values: string[]): string { | export function hashStrings(values: string[]): string { | ||||||
| @@ -56,9 +36,7 @@ export function hashStrings(values: string[]): string { | |||||||
| } | } | ||||||
|  |  | ||||||
| export function hashFileNames(fileNames: string[]): string { | export function hashFileNames(fileNames: string[]): string { | ||||||
|     return hashStrings( |     return hashStrings(fileNames.map(x => x.replace(new RegExp(`\\${path.sep}`, 'g'), '/'))) | ||||||
|         fileNames.map(x => x.replace(new RegExp(`\\${path.sep}`, 'g'), '/')) |  | ||||||
|     ) |  | ||||||
| } | } | ||||||
|  |  | ||||||
| /** | /** | ||||||
| @@ -88,176 +66,3 @@ export async function tryDelete(file: string): Promise<void> { | |||||||
| async function delay(ms: number): Promise<void> { | async function delay(ms: number): Promise<void> { | ||||||
|     return new Promise(resolve => setTimeout(resolve, ms)) |     return new Promise(resolve => setTimeout(resolve, ms)) | ||||||
| } | } | ||||||
|  |  | ||||||
| class CacheKey { |  | ||||||
|     key: string |  | ||||||
|     restoreKeys: string[] |  | ||||||
|  |  | ||||||
|     constructor(key: string, restoreKeys: string[]) { |  | ||||||
|         this.key = key |  | ||||||
|         this.restoreKeys = restoreKeys |  | ||||||
|     } |  | ||||||
| } |  | ||||||
|  |  | ||||||
| export abstract class AbstractCache { |  | ||||||
|     private cacheName: string |  | ||||||
|     private cacheDescription: string |  | ||||||
|     private cacheKeyStateKey: string |  | ||||||
|     private cacheResultStateKey: string |  | ||||||
|  |  | ||||||
|     protected readonly cacheDebuggingEnabled: boolean |  | ||||||
|  |  | ||||||
|     constructor(cacheName: string, cacheDescription: string) { |  | ||||||
|         this.cacheName = cacheName |  | ||||||
|         this.cacheDescription = cacheDescription |  | ||||||
|         this.cacheKeyStateKey = `CACHE_KEY_${cacheName}` |  | ||||||
|         this.cacheResultStateKey = `CACHE_RESULT_${cacheName}` |  | ||||||
|         this.cacheDebuggingEnabled = isCacheDebuggingEnabled() |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     async restore(): Promise<void> { |  | ||||||
|         if (this.cacheOutputExists()) { |  | ||||||
|             core.info( |  | ||||||
|                 `${this.cacheDescription} already exists. Not restoring from cache.` |  | ||||||
|             ) |  | ||||||
|             return |  | ||||||
|         } |  | ||||||
|  |  | ||||||
|         const cacheKey = generateCacheKey(this.cacheName) |  | ||||||
|  |  | ||||||
|         core.saveState(this.cacheKeyStateKey, cacheKey.key) |  | ||||||
|  |  | ||||||
|         this.debug( |  | ||||||
|             `Requesting ${this.cacheDescription} with |  | ||||||
|                 key:${cacheKey.key} |  | ||||||
|                 restoreKeys:[${cacheKey.restoreKeys}]` |  | ||||||
|         ) |  | ||||||
|  |  | ||||||
|         const cacheResult = await this.restoreCache( |  | ||||||
|             this.getCachePath(), |  | ||||||
|             cacheKey.key, |  | ||||||
|             cacheKey.restoreKeys |  | ||||||
|         ) |  | ||||||
|  |  | ||||||
|         if (!cacheResult) { |  | ||||||
|             core.info( |  | ||||||
|                 `${this.cacheDescription} cache not found. Will start with empty.` |  | ||||||
|             ) |  | ||||||
|             return |  | ||||||
|         } |  | ||||||
|  |  | ||||||
|         core.saveState(this.cacheResultStateKey, cacheResult) |  | ||||||
|  |  | ||||||
|         core.info( |  | ||||||
|             `Restored ${this.cacheDescription} from cache key: ${cacheResult}` |  | ||||||
|         ) |  | ||||||
|  |  | ||||||
|         try { |  | ||||||
|             await this.afterRestore() |  | ||||||
|         } catch (error) { |  | ||||||
|             core.warning( |  | ||||||
|                 `Restore ${this.cacheDescription} failed in 'afterRestore': ${error}` |  | ||||||
|             ) |  | ||||||
|         } |  | ||||||
|  |  | ||||||
|         return |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     protected async restoreCache( |  | ||||||
|         cachePath: string[], |  | ||||||
|         cacheKey: string, |  | ||||||
|         cacheRestoreKeys: string[] = [] |  | ||||||
|     ): Promise<string | undefined> { |  | ||||||
|         try { |  | ||||||
|             return await cache.restoreCache( |  | ||||||
|                 cachePath, |  | ||||||
|                 cacheKey, |  | ||||||
|                 cacheRestoreKeys |  | ||||||
|             ) |  | ||||||
|         } catch (error) { |  | ||||||
|             if (error instanceof cache.ValidationError) { |  | ||||||
|                 // Validation errors should fail the build action |  | ||||||
|                 throw error |  | ||||||
|             } |  | ||||||
|             // Warn about any other error and continue |  | ||||||
|             core.warning(`Failed to restore ${cacheKey}: ${error}`) |  | ||||||
|             return undefined |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     protected async afterRestore(): Promise<void> {} |  | ||||||
|  |  | ||||||
|     async save(): Promise<void> { |  | ||||||
|         if (!this.cacheOutputExists()) { |  | ||||||
|             this.debug(`No ${this.cacheDescription} to cache.`) |  | ||||||
|             return |  | ||||||
|         } |  | ||||||
|  |  | ||||||
|         const cacheKey = core.getState(this.cacheKeyStateKey) |  | ||||||
|         const cacheResult = core.getState(this.cacheResultStateKey) |  | ||||||
|  |  | ||||||
|         if (!cacheKey) { |  | ||||||
|             this.debug( |  | ||||||
|                 `${this.cacheDescription} existed prior to cache restore. Not saving.` |  | ||||||
|             ) |  | ||||||
|             return |  | ||||||
|         } |  | ||||||
|  |  | ||||||
|         if (cacheResult && cacheKey === cacheResult) { |  | ||||||
|             core.info( |  | ||||||
|                 `Cache hit occurred on the cache key ${cacheKey}, not saving cache.` |  | ||||||
|             ) |  | ||||||
|             return |  | ||||||
|         } |  | ||||||
|  |  | ||||||
|         try { |  | ||||||
|             await this.beforeSave() |  | ||||||
|         } 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 this.saveCache(cachePath, cacheKey) |  | ||||||
|  |  | ||||||
|         return |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     protected async beforeSave(): Promise<void> {} |  | ||||||
|  |  | ||||||
|     protected async saveCache( |  | ||||||
|         cachePath: string[], |  | ||||||
|         cacheKey: string |  | ||||||
|     ): Promise<void> { |  | ||||||
|         try { |  | ||||||
|             await cache.saveCache(cachePath, cacheKey) |  | ||||||
|         } catch (error) { |  | ||||||
|             if (error instanceof cache.ValidationError) { |  | ||||||
|                 // Validation errors should fail the build action |  | ||||||
|                 throw error |  | ||||||
|             } else if (error instanceof cache.ReserveCacheError) { |  | ||||||
|                 // Reserve cache errors are expected if the artifact has been previously cached |  | ||||||
|                 this.debug(error.message) |  | ||||||
|             } else { |  | ||||||
|                 // Warn about any other error and continue |  | ||||||
|                 core.warning(String(error)) |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     protected debug(message: string): void { |  | ||||||
|         if (this.cacheDebuggingEnabled) { |  | ||||||
|             core.info(message) |  | ||||||
|         } else { |  | ||||||
|             core.debug(message) |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     protected abstract cacheOutputExists(): boolean |  | ||||||
|     protected abstract getCachePath(): string[] |  | ||||||
| } |  | ||||||
|   | |||||||
| @@ -2,35 +2,93 @@ import {GradleUserHomeCache} from './cache-gradle-user-home' | |||||||
| import {ProjectDotGradleCache} from './cache-project-dot-gradle' | import {ProjectDotGradleCache} from './cache-project-dot-gradle' | ||||||
| import * as core from '@actions/core' | import * as core from '@actions/core' | ||||||
| import {isCacheDisabled, isCacheReadOnly} from './cache-utils' | import {isCacheDisabled, isCacheReadOnly} from './cache-utils' | ||||||
|  | import {CacheEntryListener, CacheListener} from './cache-base' | ||||||
|  |  | ||||||
| const BUILD_ROOT_DIR = 'BUILD_ROOT_DIR' | const BUILD_ROOT_DIR = 'BUILD_ROOT_DIR' | ||||||
|  | const CACHE_LISTENER = 'CACHE_LISTENER' | ||||||
|  |  | ||||||
| export async function restore(buildRootDirectory: string): Promise<void> { | export async function restore(buildRootDirectory: string): Promise<void> { | ||||||
|     if (isCacheDisabled()) { |     if (isCacheDisabled()) { | ||||||
|         core.debug('Cache read disabled') |         core.info('Cache is disabled: will not restore state from previous builds.') | ||||||
|         return |         return | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     await core.group('Restore Gradle state from cache', async () => { |     await core.group('Restore Gradle state from cache', async () => { | ||||||
|         core.saveState(BUILD_ROOT_DIR, buildRootDirectory) |         core.saveState(BUILD_ROOT_DIR, buildRootDirectory) | ||||||
|         return Promise.all([ |  | ||||||
|             new GradleUserHomeCache(buildRootDirectory).restore(), |         const cacheListener = new CacheListener() | ||||||
|             new ProjectDotGradleCache(buildRootDirectory).restore() |         await new GradleUserHomeCache(buildRootDirectory).restore(cacheListener) | ||||||
|         ]) |  | ||||||
|  |         const projectDotGradleCache = new ProjectDotGradleCache(buildRootDirectory) | ||||||
|  |         if (cacheListener.fullyRestored) { | ||||||
|  |             // Only restore the configuration-cache if the Gradle Home is fully restored | ||||||
|  |             await projectDotGradleCache.restore(cacheListener) | ||||||
|  |         } else { | ||||||
|  |             // Otherwise, prepare the cache key for later save() | ||||||
|  |             core.info('Gradle Home cache not fully restored: not restoring configuration-cache state') | ||||||
|  |             projectDotGradleCache.prepareCacheKey() | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         core.saveState(CACHE_LISTENER, cacheListener.stringify()) | ||||||
|     }) |     }) | ||||||
| } | } | ||||||
|  |  | ||||||
| export async function save(): Promise<void> { | export async function save(): Promise<void> { | ||||||
|  |     const cacheListener: CacheListener = CacheListener.rehydrate(core.getState(CACHE_LISTENER)) | ||||||
|  |  | ||||||
|     if (isCacheReadOnly()) { |     if (isCacheReadOnly()) { | ||||||
|         core.debug('Cache is read-only: not saving cache entry') |         core.info('Cache is read-only: will not save state for use in subsequent builds.') | ||||||
|  |         logCachingReport(cacheListener) | ||||||
|         return |         return | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     await core.group('Caching Gradle state', async () => { |     await core.group('Caching Gradle state', async () => { | ||||||
|         const buildRootDirectory = core.getState(BUILD_ROOT_DIR) |         const buildRootDirectory = core.getState(BUILD_ROOT_DIR) | ||||||
|         return Promise.all([ |         return Promise.all([ | ||||||
|             new GradleUserHomeCache(buildRootDirectory).save(), |             new GradleUserHomeCache(buildRootDirectory).save(cacheListener), | ||||||
|             new ProjectDotGradleCache(buildRootDirectory).save() |             new ProjectDotGradleCache(buildRootDirectory).save(cacheListener) | ||||||
|         ]) |         ]) | ||||||
|     }) |     }) | ||||||
|  |  | ||||||
|  |     logCachingReport(cacheListener) | ||||||
|  | } | ||||||
|  |  | ||||||
|  | function logCachingReport(listener: CacheListener): void { | ||||||
|  |     core.info(`---------- Caching Summary ------------- | ||||||
|  | Restored Entries Count: ${getCount(listener.cacheEntries, e => e.restoredSize)} | ||||||
|  |                   Size: ${getSum(listener.cacheEntries, e => e.restoredSize)} | ||||||
|  | Saved Entries    Count: ${getCount(listener.cacheEntries, e => e.savedSize)} | ||||||
|  |                   Size: ${getSum(listener.cacheEntries, e => e.savedSize)}`) | ||||||
|  |  | ||||||
|  |     core.startGroup('Cache Entry details') | ||||||
|  |     for (const entry of listener.cacheEntries) { | ||||||
|  |         core.info(`Entry: ${entry.entryName} | ||||||
|  |     Requested Key : ${entry.requestedKey ?? ''} | ||||||
|  |     Restored  Key : ${entry.restoredKey ?? ''} | ||||||
|  |               Size: ${formatSize(entry.restoredSize)} | ||||||
|  |     Saved     Key : ${entry.savedKey ?? ''} | ||||||
|  |               Size: ${formatSize(entry.savedSize)}`) | ||||||
|  |     } | ||||||
|  |     core.endGroup() | ||||||
|  | } | ||||||
|  |  | ||||||
|  | function getCount( | ||||||
|  |     cacheEntries: CacheEntryListener[], | ||||||
|  |     predicate: (value: CacheEntryListener) => number | undefined | ||||||
|  | ): number { | ||||||
|  |     return cacheEntries.filter(e => predicate(e) !== undefined).length | ||||||
|  | } | ||||||
|  |  | ||||||
|  | function getSum( | ||||||
|  |     cacheEntries: CacheEntryListener[], | ||||||
|  |     predicate: (value: CacheEntryListener) => number | undefined | ||||||
|  | ): string { | ||||||
|  |     return formatSize(cacheEntries.map(e => predicate(e) ?? 0).reduce((p, v) => p + v, 0)) | ||||||
|  | } | ||||||
|  |  | ||||||
|  | function formatSize(bytes: number | undefined): string { | ||||||
|  |     if (bytes === undefined || bytes === 0) { | ||||||
|  |         return '' | ||||||
|  |     } | ||||||
|  |     return `${Math.round(bytes / (1024 * 1024))} MB (${bytes} B)` | ||||||
| } | } | ||||||
|   | |||||||
| @@ -3,11 +3,7 @@ import fs from 'fs' | |||||||
| import path from 'path' | import path from 'path' | ||||||
| import {writeInitScript} from './build-scan-capture' | import {writeInitScript} from './build-scan-capture' | ||||||
|  |  | ||||||
| export async function execute( | export async function execute(executable: string, root: string, args: string[]): Promise<BuildResult> { | ||||||
|     executable: string, |  | ||||||
|     root: string, |  | ||||||
|     args: string[] |  | ||||||
| ): Promise<BuildResult> { |  | ||||||
|     let buildScanUrl: string | undefined |     let buildScanUrl: string | undefined | ||||||
|  |  | ||||||
|     // TODO: instead of running with no-daemon, run `--stop` in post action. |     // TODO: instead of running with no-daemon, run `--stop` in post action. | ||||||
|   | |||||||
| @@ -17,10 +17,7 @@ export function locateGradleWrapperScript(buildRootDirectory: string): string { | |||||||
| } | } | ||||||
|  |  | ||||||
| function validateGradleWrapper(buildRootDirectory: string): void { | function validateGradleWrapper(buildRootDirectory: string): void { | ||||||
|     const wrapperProperties = path.resolve( |     const wrapperProperties = path.resolve(buildRootDirectory, 'gradle/wrapper/gradle-wrapper.properties') | ||||||
|         buildRootDirectory, |  | ||||||
|         'gradle/wrapper/gradle-wrapper.properties' |  | ||||||
|     ) |  | ||||||
|     if (!fs.existsSync(wrapperProperties)) { |     if (!fs.existsSync(wrapperProperties)) { | ||||||
|         throw new Error( |         throw new Error( | ||||||
|             `Cannot locate a Gradle wrapper properties file at '${wrapperProperties}'. Specify 'gradle-version' or 'gradle-executable' for projects without Gradle wrapper configured.` |             `Cannot locate a Gradle wrapper properties file at '${wrapperProperties}'. Specify 'gradle-version' or 'gradle-executable' for projects without Gradle wrapper configured.` | ||||||
|   | |||||||
							
								
								
									
										27
									
								
								src/main.ts
									
									
									
									
									
								
							
							
						
						
									
										27
									
								
								src/main.ts
									
									
									
									
									
								
							| @@ -18,23 +18,25 @@ export async function run(): Promise<void> { | |||||||
|         const args: string[] = parseCommandLineArguments() |         const args: string[] = parseCommandLineArguments() | ||||||
|  |  | ||||||
|         const result = await execution.execute( |         const result = await execution.execute( | ||||||
|             await resolveGradleExecutable( |             await resolveGradleExecutable(workspaceDirectory, buildRootDirectory), | ||||||
|                 workspaceDirectory, |  | ||||||
|                 buildRootDirectory |  | ||||||
|             ), |  | ||||||
|             buildRootDirectory, |             buildRootDirectory, | ||||||
|             args |             args | ||||||
|         ) |         ) | ||||||
|  |  | ||||||
|         if (result.buildScanUrl) { |         if (result.buildScanUrl) { | ||||||
|             core.setOutput('build-scan-url', result.buildScanUrl) |             core.setOutput('build-scan-url', result.buildScanUrl) | ||||||
|             // TODO Include context about the invocation (eg step name) in this message |  | ||||||
|             // Unfortunately it doesn't seem possible to access the current step name here |  | ||||||
|             core.notice(`Gradle build scan: ${result.buildScanUrl}`) |  | ||||||
|         } |         } | ||||||
|  |  | ||||||
|         if (result.status !== 0) { |         if (result.status !== 0) { | ||||||
|             core.setFailed(`Gradle process exited with status ${result.status}`) |             if (result.buildScanUrl) { | ||||||
|  |                 core.setFailed(`Gradle build failed: ${result.buildScanUrl}`) | ||||||
|  |             } else { | ||||||
|  |                 core.setFailed(`Gradle build failed: process exited with status ${result.status}`) | ||||||
|  |             } | ||||||
|  |         } else { | ||||||
|  |             if (result.buildScanUrl) { | ||||||
|  |                 core.notice(`Gradle build succeeded: ${result.buildScanUrl}`) | ||||||
|  |             } | ||||||
|         } |         } | ||||||
|     } catch (error) { |     } catch (error) { | ||||||
|         core.setFailed(String(error)) |         core.setFailed(String(error)) | ||||||
| @@ -46,10 +48,7 @@ export async function run(): Promise<void> { | |||||||
|  |  | ||||||
| run() | run() | ||||||
|  |  | ||||||
| async function resolveGradleExecutable( | async function resolveGradleExecutable(workspaceDirectory: string, buildRootDirectory: string): Promise<string> { | ||||||
|     workspaceDirectory: string, |  | ||||||
|     buildRootDirectory: string |  | ||||||
| ): Promise<string> { |  | ||||||
|     const gradleVersion = core.getInput('gradle-version') |     const gradleVersion = core.getInput('gradle-version') | ||||||
|     if (gradleVersion !== '' && gradleVersion !== 'wrapper') { |     if (gradleVersion !== '' && gradleVersion !== 'wrapper') { | ||||||
|         return path.resolve(await provision.gradleVersion(gradleVersion)) |         return path.resolve(await provision.gradleVersion(gradleVersion)) | ||||||
| @@ -66,9 +65,7 @@ async function resolveGradleExecutable( | |||||||
| function resolveBuildRootDirectory(baseDirectory: string): string { | function resolveBuildRootDirectory(baseDirectory: string): string { | ||||||
|     const buildRootDirectory = core.getInput('build-root-directory') |     const buildRootDirectory = core.getInput('build-root-directory') | ||||||
|     const resolvedBuildRootDirectory = |     const resolvedBuildRootDirectory = | ||||||
|         buildRootDirectory === '' |         buildRootDirectory === '' ? path.resolve(baseDirectory) : path.resolve(baseDirectory, buildRootDirectory) | ||||||
|             ? path.resolve(baseDirectory) |  | ||||||
|             : path.resolve(baseDirectory, buildRootDirectory) |  | ||||||
|     return resolvedBuildRootDirectory |     return resolvedBuildRootDirectory | ||||||
| } | } | ||||||
|  |  | ||||||
|   | |||||||
| @@ -19,9 +19,7 @@ export async function gradleVersion(version: string): Promise<string> { | |||||||
|         case 'current': |         case 'current': | ||||||
|             return gradleCurrent() |             return gradleCurrent() | ||||||
|         case 'rc': |         case 'rc': | ||||||
|             core.warning( |             core.warning(`Specifying gradle-version 'rc' has been deprecated. Use 'release-candidate' instead.`) | ||||||
|                 `Specifying gradle-version 'rc' has been deprecated. Use 'release-candidate' instead.` |  | ||||||
|             ) |  | ||||||
|             return gradleReleaseCandidate() |             return gradleReleaseCandidate() | ||||||
|         case 'release-candidate': |         case 'release-candidate': | ||||||
|             return gradleReleaseCandidate() |             return gradleReleaseCandidate() | ||||||
| @@ -35,16 +33,12 @@ export async function gradleVersion(version: string): Promise<string> { | |||||||
| } | } | ||||||
|  |  | ||||||
| async function gradleCurrent(): Promise<string> { | async function gradleCurrent(): Promise<string> { | ||||||
|     const versionInfo = await gradleVersionDeclaration( |     const versionInfo = await gradleVersionDeclaration(`${gradleVersionsBaseUrl}/current`) | ||||||
|         `${gradleVersionsBaseUrl}/current` |  | ||||||
|     ) |  | ||||||
|     return provisionGradle(versionInfo) |     return provisionGradle(versionInfo) | ||||||
| } | } | ||||||
|  |  | ||||||
| async function gradleReleaseCandidate(): Promise<string> { | async function gradleReleaseCandidate(): Promise<string> { | ||||||
|     const versionInfo = await gradleVersionDeclaration( |     const versionInfo = await gradleVersionDeclaration(`${gradleVersionsBaseUrl}/release-candidate`) | ||||||
|         `${gradleVersionsBaseUrl}/release-candidate` |  | ||||||
|     ) |  | ||||||
|     if (versionInfo && versionInfo.version && versionInfo.downloadUrl) { |     if (versionInfo && versionInfo.version && versionInfo.downloadUrl) { | ||||||
|         return provisionGradle(versionInfo) |         return provisionGradle(versionInfo) | ||||||
|     } |     } | ||||||
| @@ -53,16 +47,12 @@ async function gradleReleaseCandidate(): Promise<string> { | |||||||
| } | } | ||||||
|  |  | ||||||
| async function gradleNightly(): Promise<string> { | async function gradleNightly(): Promise<string> { | ||||||
|     const versionInfo = await gradleVersionDeclaration( |     const versionInfo = await gradleVersionDeclaration(`${gradleVersionsBaseUrl}/nightly`) | ||||||
|         `${gradleVersionsBaseUrl}/nightly` |  | ||||||
|     ) |  | ||||||
|     return provisionGradle(versionInfo) |     return provisionGradle(versionInfo) | ||||||
| } | } | ||||||
|  |  | ||||||
| async function gradleReleaseNightly(): Promise<string> { | async function gradleReleaseNightly(): Promise<string> { | ||||||
|     const versionInfo = await gradleVersionDeclaration( |     const versionInfo = await gradleVersionDeclaration(`${gradleVersionsBaseUrl}/release-nightly`) | ||||||
|         `${gradleVersionsBaseUrl}/release-nightly` |  | ||||||
|     ) |  | ||||||
|     return provisionGradle(versionInfo) |     return provisionGradle(versionInfo) | ||||||
| } | } | ||||||
|  |  | ||||||
| @@ -74,34 +64,24 @@ async function gradle(version: string): Promise<string> { | |||||||
|     return provisionGradle(versionInfo) |     return provisionGradle(versionInfo) | ||||||
| } | } | ||||||
|  |  | ||||||
| async function gradleVersionDeclaration( | async function gradleVersionDeclaration(url: string): Promise<GradleVersionInfo> { | ||||||
|     url: string |  | ||||||
| ): Promise<GradleVersionInfo> { |  | ||||||
|     return await httpGetGradleVersion(url) |     return await httpGetGradleVersion(url) | ||||||
| } | } | ||||||
|  |  | ||||||
| async function findGradleVersionDeclaration( | async function findGradleVersionDeclaration(version: string): Promise<GradleVersionInfo | undefined> { | ||||||
|     version: string |     const gradleVersions = await httpGetGradleVersions(`${gradleVersionsBaseUrl}/all`) | ||||||
| ): Promise<GradleVersionInfo | undefined> { |  | ||||||
|     const gradleVersions = await httpGetGradleVersions( |  | ||||||
|         `${gradleVersionsBaseUrl}/all` |  | ||||||
|     ) |  | ||||||
|     return gradleVersions.find((entry: GradleVersionInfo) => { |     return gradleVersions.find((entry: GradleVersionInfo) => { | ||||||
|         return entry.version === version |         return entry.version === version | ||||||
|     }) |     }) | ||||||
| } | } | ||||||
|  |  | ||||||
| async function provisionGradle( | async function provisionGradle(versionInfo: GradleVersionInfo): Promise<string> { | ||||||
|     versionInfo: GradleVersionInfo |  | ||||||
| ): Promise<string> { |  | ||||||
|     return core.group(`Provision Gradle ${versionInfo.version}`, async () => { |     return core.group(`Provision Gradle ${versionInfo.version}`, async () => { | ||||||
|         return locateGradleAndDownloadIfRequired(versionInfo) |         return locateGradleAndDownloadIfRequired(versionInfo) | ||||||
|     }) |     }) | ||||||
| } | } | ||||||
|  |  | ||||||
| async function locateGradleAndDownloadIfRequired( | async function locateGradleAndDownloadIfRequired(versionInfo: GradleVersionInfo): Promise<string> { | ||||||
|     versionInfo: GradleVersionInfo |  | ||||||
| ): Promise<string> { |  | ||||||
|     const installsDir = path.join(os.homedir(), 'gradle-installations/installs') |     const installsDir = path.join(os.homedir(), 'gradle-installations/installs') | ||||||
|     const installDir = path.join(installsDir, `gradle-${versionInfo.version}`) |     const installDir = path.join(installsDir, `gradle-${versionInfo.version}`) | ||||||
|     if (fs.existsSync(installDir)) { |     if (fs.existsSync(installDir)) { | ||||||
| @@ -120,13 +100,8 @@ async function locateGradleAndDownloadIfRequired( | |||||||
|     return executable |     return executable | ||||||
| } | } | ||||||
|  |  | ||||||
| async function downloadAndCacheGradleDistribution( | async function downloadAndCacheGradleDistribution(versionInfo: GradleVersionInfo): Promise<string> { | ||||||
|     versionInfo: GradleVersionInfo |     const downloadPath = path.join(os.homedir(), `gradle-installations/downloads/gradle-${versionInfo.version}-bin.zip`) | ||||||
| ): Promise<string> { |  | ||||||
|     const downloadPath = path.join( |  | ||||||
|         os.homedir(), |  | ||||||
|         `gradle-installations/downloads/gradle-${versionInfo.version}-bin.zip` |  | ||||||
|     ) |  | ||||||
|  |  | ||||||
|     if (isCacheDisabled()) { |     if (isCacheDisabled()) { | ||||||
|         await downloadGradleDistribution(versionInfo, downloadPath) |         await downloadGradleDistribution(versionInfo, downloadPath) | ||||||
| @@ -136,14 +111,10 @@ async function downloadAndCacheGradleDistribution( | |||||||
|     const cacheKey = `gradle-${versionInfo.version}` |     const cacheKey = `gradle-${versionInfo.version}` | ||||||
|     const restoreKey = await cache.restoreCache([downloadPath], cacheKey) |     const restoreKey = await cache.restoreCache([downloadPath], cacheKey) | ||||||
|     if (restoreKey) { |     if (restoreKey) { | ||||||
|         core.info( |         core.info(`Restored Gradle distribution ${cacheKey} from cache to ${downloadPath}`) | ||||||
|             `Restored Gradle distribution ${cacheKey} from cache to ${downloadPath}` |  | ||||||
|         ) |  | ||||||
|         return downloadPath |         return downloadPath | ||||||
|     } |     } | ||||||
|     core.info( |     core.info(`Gradle distribution ${versionInfo.version} not found in cache. Will download.`) | ||||||
|         `Gradle distribution ${versionInfo.version} not found in cache. Will download.` |  | ||||||
|     ) |  | ||||||
|     await downloadGradleDistribution(versionInfo, downloadPath) |     await downloadGradleDistribution(versionInfo, downloadPath) | ||||||
|  |  | ||||||
|     if (!isCacheReadOnly()) { |     if (!isCacheReadOnly()) { | ||||||
| @@ -151,10 +122,7 @@ async function downloadAndCacheGradleDistribution( | |||||||
|             await cache.saveCache([downloadPath], cacheKey) |             await cache.saveCache([downloadPath], cacheKey) | ||||||
|         } catch (error) { |         } catch (error) { | ||||||
|             // Fail on validation errors or non-errors (the latter to keep Typescript happy) |             // Fail on validation errors or non-errors (the latter to keep Typescript happy) | ||||||
|             if ( |             if (error instanceof cache.ValidationError || !(error instanceof Error)) { | ||||||
|                 error instanceof cache.ValidationError || |  | ||||||
|                 !(error instanceof Error) |  | ||||||
|             ) { |  | ||||||
|                 throw error |                 throw error | ||||||
|             } |             } | ||||||
|             core.warning(error.message) |             core.warning(error.message) | ||||||
| @@ -163,16 +131,9 @@ async function downloadAndCacheGradleDistribution( | |||||||
|     return downloadPath |     return downloadPath | ||||||
| } | } | ||||||
|  |  | ||||||
| async function downloadGradleDistribution( | async function downloadGradleDistribution(versionInfo: GradleVersionInfo, downloadPath: string): Promise<void> { | ||||||
|     versionInfo: GradleVersionInfo, |  | ||||||
|     downloadPath: string |  | ||||||
| ): Promise<void> { |  | ||||||
|     await toolCache.downloadTool(versionInfo.downloadUrl, downloadPath) |     await toolCache.downloadTool(versionInfo.downloadUrl, downloadPath) | ||||||
|     core.info( |     core.info(`Downloaded ${versionInfo.downloadUrl} to ${downloadPath} (size ${fs.statSync(downloadPath).size})`) | ||||||
|         `Downloaded ${versionInfo.downloadUrl} to ${downloadPath} (size ${ |  | ||||||
|             fs.statSync(downloadPath).size |  | ||||||
|         })` |  | ||||||
|     ) |  | ||||||
| } | } | ||||||
|  |  | ||||||
| function executableFrom(installDir: string): string { | function executableFrom(installDir: string): string { | ||||||
| @@ -183,9 +144,7 @@ async function httpGetGradleVersion(url: string): Promise<GradleVersionInfo> { | |||||||
|     return JSON.parse(await httpGetString(url)) |     return JSON.parse(await httpGetString(url)) | ||||||
| } | } | ||||||
|  |  | ||||||
| async function httpGetGradleVersions( | async function httpGetGradleVersions(url: string): Promise<GradleVersionInfo[]> { | ||||||
|     url: string |  | ||||||
| ): Promise<GradleVersionInfo[]> { |  | ||||||
|     return JSON.parse(await httpGetString(url)) |     return JSON.parse(await httpGetString(url)) | ||||||
| } | } | ||||||
|  |  | ||||||
|   | |||||||
		Reference in New Issue
	
	Block a user