Merge pull request #93 from gradle/dd/instrumented-jars-fix
Ensure all-or-nothing restore of cached instrumented-jars Leaving the .lock and .receipt files lying around was causing issues when the actual jar files were not restored. Now the entire directory will either be missing, or completely restored. Fixes #91
This commit is contained in:
		
						commit
						0eb5996567
					
				
							
								
								
									
										16
									
								
								.github/workflows/integTest-caching.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										16
									
								
								.github/workflows/integTest-caching.yml
									
									
									
									
										vendored
									
									
								
							| @ -82,3 +82,19 @@ jobs: | ||||
|         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 | ||||
|     runs-on: ubuntu-latest | ||||
|     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 | ||||
|         cache-artifact-bundles: '[]' | ||||
|         cache-read-only: true | ||||
| 
 | ||||
|  | ||||
							
								
								
									
										10
									
								
								action.yml
									
									
									
									
									
								
							
							
						
						
									
										10
									
								
								action.yml
									
									
									
									
									
								
							| @ -30,6 +30,16 @@ inputs: | ||||
|     description: Used to uniquely identify the current job invocation. Defaults to the matrix values for this job; this should not be overridden by users. | ||||
|     required: false | ||||
|     default: ${{ toJSON(matrix) }} | ||||
|   cache-artifact-bundles: | ||||
|     description: Names and patterns of artifact bundles to cache separately. For internal use only. | ||||
|     required: false | ||||
|     default: | | ||||
|         [ | ||||
|           ["generated-gradle-jars", "caches/*/generated-gradle-jars/*.jar"], | ||||
|           ["wrapper-zips", "wrapper/dists/*/*/*.zip"], | ||||
|           ["dependency-jars", "caches/modules-*/files-*/**/*.jar"], | ||||
|           ["instrumented-jars", "caches/jars-*/*"] | ||||
|         ] | ||||
| 
 | ||||
| outputs: | ||||
|   build-scan-url: | ||||
|  | ||||
							
								
								
									
										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
											
										
									
								
							| @ -10,13 +10,6 @@ import {AbstractCache, hashFileNames, tryDelete} from './cache-utils' | ||||
| // Which paths under Gradle User Home should be cached
 | ||||
| const CACHE_PATH = ['caches', 'notifications'] | ||||
| 
 | ||||
| const COMMON_ARTIFACT_CACHES = new Map([ | ||||
|     ['generated-gradle-jars', 'caches/*/generated-gradle-jars/*.jar'], | ||||
|     ['wrapper-zips', 'wrapper/dists/*/*/*.zip'], | ||||
|     ['dependency-jars', 'caches/modules-*/files-*/**/*.jar'], | ||||
|     ['instrumented-jars', 'caches/jars-*/*/*.jar'] | ||||
| ]) | ||||
| 
 | ||||
| export class GradleUserHomeCache extends AbstractCache { | ||||
|     private gradleUserHome: string | ||||
| 
 | ||||
| @ -27,14 +20,14 @@ export class GradleUserHomeCache extends AbstractCache { | ||||
| 
 | ||||
|     async afterRestore(): Promise<void> { | ||||
|         await this.reportGradleUserHomeSize('as restored from cache') | ||||
|         await this.restoreCommonArtifacts() | ||||
|         await this.restoreArtifactBundles() | ||||
|         await this.reportGradleUserHomeSize('after restoring common artifacts') | ||||
|     } | ||||
| 
 | ||||
|     private async restoreCommonArtifacts(): Promise<void> { | ||||
|     private async restoreArtifactBundles(): Promise<void> { | ||||
|         const processes: Promise<void>[] = [] | ||||
|         for (const [bundle, pattern] of this.getCommonArtifactPaths()) { | ||||
|             const p = this.restoreCommonArtifactBundle(bundle, pattern) | ||||
|         for (const [bundle, pattern] of this.getArtifactBundles()) { | ||||
|             const p = this.restoreArtifactBundle(bundle, pattern) | ||||
|             // Run sequentially when debugging enabled
 | ||||
|             if (this.cacheDebuggingEnabled) { | ||||
|                 await p | ||||
| @ -45,13 +38,13 @@ export class GradleUserHomeCache extends AbstractCache { | ||||
|         await Promise.all(processes) | ||||
|     } | ||||
| 
 | ||||
|     private async restoreCommonArtifactBundle( | ||||
|     private async restoreArtifactBundle( | ||||
|         bundle: string, | ||||
|         artifactPath: string | ||||
|     ): Promise<void> { | ||||
|         const cacheMetaFile = this.getCacheMetaFile(bundle) | ||||
|         if (fs.existsSync(cacheMetaFile)) { | ||||
|             const cacheKey = fs.readFileSync(cacheMetaFile, 'utf-8').trim() | ||||
|         const bundleMetaFile = this.getBundleMetaFile(bundle) | ||||
|         if (fs.existsSync(bundleMetaFile)) { | ||||
|             const cacheKey = fs.readFileSync(bundleMetaFile, 'utf-8').trim() | ||||
|             const restoreKey = await this.restoreCache([artifactPath], cacheKey) | ||||
|             if (restoreKey) { | ||||
|                 core.info( | ||||
| @ -64,12 +57,12 @@ export class GradleUserHomeCache extends AbstractCache { | ||||
|             } | ||||
|         } else { | ||||
|             this.debug( | ||||
|                 `No metafile found to restore ${bundle}: ${cacheMetaFile}` | ||||
|                 `No metafile found to restore ${bundle}: ${bundleMetaFile}` | ||||
|             ) | ||||
|         } | ||||
|     } | ||||
| 
 | ||||
|     private getCacheMetaFile(name: string): string { | ||||
|     private getBundleMetaFile(name: string): string { | ||||
|         return path.resolve( | ||||
|             this.gradleUserHome, | ||||
|             'caches', | ||||
| @ -79,14 +72,14 @@ export class GradleUserHomeCache extends AbstractCache { | ||||
| 
 | ||||
|     async beforeSave(): Promise<void> { | ||||
|         await this.reportGradleUserHomeSize('before saving common artifacts') | ||||
|         await this.saveCommonArtifacts() | ||||
|         await this.saveArtifactBundles() | ||||
|         await this.reportGradleUserHomeSize('after saving common artifacts') | ||||
|     } | ||||
| 
 | ||||
|     private async saveCommonArtifacts(): Promise<void> { | ||||
|     private async saveArtifactBundles(): Promise<void> { | ||||
|         const processes: Promise<void>[] = [] | ||||
|         for (const [bundle, pattern] of this.getCommonArtifactPaths()) { | ||||
|             const p = this.saveCommonArtifactBundle(bundle, pattern) | ||||
|         for (const [bundle, pattern] of this.getArtifactBundles()) { | ||||
|             const p = this.saveArtifactBundle(bundle, pattern) | ||||
|             // Run sequentially when debugging enabled
 | ||||
|             if (this.cacheDebuggingEnabled) { | ||||
|                 await p | ||||
| @ -97,28 +90,28 @@ export class GradleUserHomeCache extends AbstractCache { | ||||
|         await Promise.all(processes) | ||||
|     } | ||||
| 
 | ||||
|     private async saveCommonArtifactBundle( | ||||
|     private async saveArtifactBundle( | ||||
|         bundle: string, | ||||
|         artifactPath: string | ||||
|     ): Promise<void> { | ||||
|         const cacheMetaFile = this.getCacheMetaFile(bundle) | ||||
|         const bundleMetaFile = this.getBundleMetaFile(bundle) | ||||
| 
 | ||||
|         const globber = await glob.create(artifactPath) | ||||
|         const commonArtifactFiles = await globber.glob() | ||||
|         const bundleFiles = await globber.glob() | ||||
| 
 | ||||
|         // Handle no matching files
 | ||||
|         if (commonArtifactFiles.length === 0) { | ||||
|         if (bundleFiles.length === 0) { | ||||
|             this.debug(`No files found to cache for ${bundle}`) | ||||
|             if (fs.existsSync(cacheMetaFile)) { | ||||
|                 tryDelete(cacheMetaFile) | ||||
|             if (fs.existsSync(bundleMetaFile)) { | ||||
|                 tryDelete(bundleMetaFile) | ||||
|             } | ||||
|             return | ||||
|         } | ||||
| 
 | ||||
|         const previouslyRestoredKey = fs.existsSync(cacheMetaFile) | ||||
|             ? fs.readFileSync(cacheMetaFile, 'utf-8').trim() | ||||
|         const previouslyRestoredKey = fs.existsSync(bundleMetaFile) | ||||
|             ? fs.readFileSync(bundleMetaFile, 'utf-8').trim() | ||||
|             : '' | ||||
|         const cacheKey = this.createCacheKey(bundle, commonArtifactFiles) | ||||
|         const cacheKey = this.createCacheKey(bundle, bundleFiles) | ||||
| 
 | ||||
|         if (previouslyRestoredKey === cacheKey) { | ||||
|             this.debug( | ||||
| @ -128,11 +121,11 @@ export class GradleUserHomeCache extends AbstractCache { | ||||
|             core.info(`Caching ${bundle} with cache key: ${cacheKey}`) | ||||
|             await this.saveCache([artifactPath], cacheKey) | ||||
| 
 | ||||
|             this.debug(`Writing cache metafile: ${cacheMetaFile}`) | ||||
|             fs.writeFileSync(cacheMetaFile, cacheKey) | ||||
|             this.debug(`Writing cache metafile: ${bundleMetaFile}`) | ||||
|             fs.writeFileSync(bundleMetaFile, cacheKey) | ||||
|         } | ||||
| 
 | ||||
|         for (const file of commonArtifactFiles) { | ||||
|         for (const file of bundleFiles) { | ||||
|             tryDelete(file) | ||||
|         } | ||||
|     } | ||||
| @ -170,9 +163,14 @@ export class GradleUserHomeCache extends AbstractCache { | ||||
|         return CACHE_PATH.map(x => path.resolve(this.gradleUserHome, x)) | ||||
|     } | ||||
| 
 | ||||
|     private getCommonArtifactPaths(): Map<string, string> { | ||||
|     private getArtifactBundles(): Map<string, string> { | ||||
|         const artifactBundleDefinition = core.getInput('cache-artifact-bundles') | ||||
|         this.debug( | ||||
|             `Using artifact bundle definition: ${artifactBundleDefinition}` | ||||
|         ) | ||||
|         const artifactBundles = JSON.parse(artifactBundleDefinition) | ||||
|         return new Map( | ||||
|             Array.from(COMMON_ARTIFACT_CACHES, ([key, value]) => [ | ||||
|             Array.from(artifactBundles, ([key, value]) => [ | ||||
|                 key, | ||||
|                 path.resolve(this.gradleUserHome, value) | ||||
|             ]) | ||||
|  | ||||
| @ -62,12 +62,17 @@ export function hashFileNames(fileNames: string[]): string { | ||||
| } | ||||
| 
 | ||||
| /** | ||||
|  * Attempt to delete a file, waiting to allow locks to be released | ||||
|  * Attempt to delete a file or directory, waiting to allow locks to be released | ||||
|  */ | ||||
| export async function tryDelete(file: string): Promise<void> { | ||||
|     const stat = fs.lstatSync(file) | ||||
|     for (let count = 0; count < 3; count++) { | ||||
|         try { | ||||
|             fs.unlinkSync(file) | ||||
|             if (stat.isDirectory()) { | ||||
|                 fs.rmdirSync(file, {recursive: true}) | ||||
|             } else { | ||||
|                 fs.unlinkSync(file) | ||||
|             } | ||||
|             return | ||||
|         } catch (error) { | ||||
|             if (count === 2) { | ||||
|  | ||||
		Loading…
	
	
			
			x
			
			
		
	
		Reference in New Issue
	
	Block a user