mirror of
				https://github.com/gradle/gradle-build-action.git
				synced 2025-10-23 02:18:55 +08:00 
			
		
		
		
	Consolidate logic for separate artifact file caching
Making this functionality more general will make it easier to add other artifacts to the set that we cache separately.
This commit is contained in:
		
							
								
								
									
										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/post/index.js
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										2
									
								
								dist/post/index.js
									
									
									
									
										vendored
									
									
								
							
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							| @@ -10,11 +10,16 @@ import {AbstractCache} from './cache-utils' | |||||||
|  |  | ||||||
| const CACHE_PATH = [ | const CACHE_PATH = [ | ||||||
|     '~/.gradle/caches/*', // All directories in 'caches' |     '~/.gradle/caches/*', // All directories in 'caches' | ||||||
|     '!~/.gradle/caches/*/generated-gradle-jars', // Exclude generated-gradle-jars |  | ||||||
|     '~/.gradle/notifications/*', // Prevent the re-rendering of first-use message for version |     '~/.gradle/notifications/*', // Prevent the re-rendering of first-use message for version | ||||||
|     '~/.gradle/wrapper/dists/*/*/*.zip.txt' // Only wrapper zips are required : Gradle will expand these on demand |     '~/.gradle/wrapper/dists/*/*/*.zip.txt' // Only wrapper zips are required : We do not want to cache the exploded distributions | ||||||
| ] | ] | ||||||
|  |  | ||||||
|  | const DEDUPLCIATED_PATHS = [ | ||||||
|  |     '~/.gradle/wrapper/dists/*/*/*.zip', | ||||||
|  |     '~/.gradle/caches/*/generated-gradle-jars/*.jar' | ||||||
|  | ] | ||||||
|  | const MARKER_FILE_EXTENSION = '.txt' | ||||||
|  |  | ||||||
| export class GradleUserHomeCache extends AbstractCache { | export class GradleUserHomeCache extends AbstractCache { | ||||||
|     constructor() { |     constructor() { | ||||||
|         super('gradle', 'Gradle User Home') |         super('gradle', 'Gradle User Home') | ||||||
| @@ -23,90 +28,48 @@ export class GradleUserHomeCache extends AbstractCache { | |||||||
|     async restore(): Promise<void> { |     async restore(): Promise<void> { | ||||||
|         await super.restore() |         await super.restore() | ||||||
|         await this.reportCacheEntrySize() |         await this.reportCacheEntrySize() | ||||||
|         await this.restoreWrapperZips() |         await this.restoreDeduplicatedPaths() | ||||||
|         await this.restoreGeneratedJars() |         await this.reportCacheEntrySize() | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     private async restoreWrapperZips(): Promise<void> { |     private async restoreDeduplicatedPaths(): Promise<void> { | ||||||
|         const globber = await glob.create( |         const markerFilePatterns = DEDUPLCIATED_PATHS.map(targetPath => { | ||||||
|             '~/.gradle/wrapper/dists/*/*/*.zip.txt' |             return targetPath + MARKER_FILE_EXTENSION | ||||||
|         ) |         }).join('\n') | ||||||
|         const wrapperMarkers = await globber.glob() |  | ||||||
|  |  | ||||||
|         core.info('Found the following wrapper zips') |         core.info(`Using marker file patterns: ${markerFilePatterns}`) | ||||||
|         for (const wrapperMarker of wrapperMarkers) { |         const globber = await glob.create(markerFilePatterns) | ||||||
|             const wrapperZip = wrapperMarker.substring( |         const markerFiles = await globber.glob() | ||||||
|  |  | ||||||
|  |         for (const markerFile of markerFiles) { | ||||||
|  |             const targetFile = markerFile.substring( | ||||||
|                 0, |                 0, | ||||||
|                 wrapperMarker.length - '.txt'.length |                 markerFile.length - MARKER_FILE_EXTENSION.length | ||||||
|             ) |             ) | ||||||
|             core.info( |             core.info( | ||||||
|                 `Wrapper marker: ${wrapperMarker}. Looking for zip ${wrapperZip}` |                 `Found marker file: ${markerFile}. Looking for ${targetFile}` | ||||||
|             ) |             ) | ||||||
|  |  | ||||||
|             if (!fs.existsSync(wrapperZip)) { |             if (!fs.existsSync(targetFile)) { | ||||||
|                 // Extract the wrapper URL hash |                 const key = path.relative(this.getGradleUserHome(), targetFile) | ||||||
|                 const wrapperKey = path.basename(path.dirname(wrapperMarker)) |                 const cacheKey = `gradle-dedup-${key}` | ||||||
|                 core.info(`Wrapper key: ${wrapperKey}`) |                 core.info(`Cache key: ${cacheKey}. Cache path: ${targetFile}`) | ||||||
|  |  | ||||||
|                 const cacheKey = `gradle-wrapper-${wrapperKey}` |  | ||||||
|                 core.info(`Cache key: ${cacheKey}. Cache path: ${wrapperZip}`) |  | ||||||
|  |  | ||||||
|                 const restoreKey = await cache.restoreCache( |                 const restoreKey = await cache.restoreCache( | ||||||
|                     [wrapperZip], |                     [targetFile], | ||||||
|                     cacheKey |                     cacheKey | ||||||
|                 ) |                 ) | ||||||
|                 if (restoreKey) { |                 if (restoreKey) { | ||||||
|                     core.info( |                     core.info( | ||||||
|                         `Restored wrapper zip ${cacheKey} from cache to ${wrapperZip}` |                         `Restored ${cacheKey} from cache to ${targetFile}` | ||||||
|                     ) |                     ) | ||||||
|                 } else { |                 } else { | ||||||
|                     core.info( |                     core.info( | ||||||
|                         `Did NOT restore wrapper zip from ${cacheKey} to ${wrapperZip}` |                         `Did NOT restore from ${cacheKey} to ${targetFile}` | ||||||
|                     ) |                     ) | ||||||
|                 } |                 } | ||||||
|             } else { |             } else { | ||||||
|                 core.info(`Wrapper zip file already exists: ${wrapperZip}`) |                 core.info(`Target file already exists: ${targetFile}`) | ||||||
|             } |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     private async restoreGeneratedJars(): Promise<void> { |  | ||||||
|         const globber = await glob.create( |  | ||||||
|             '~/.gradle/caches/*/generated-gradle-jars/*.jar.txt' |  | ||||||
|         ) |  | ||||||
|         const generatedJarMarkers = await globber.glob() |  | ||||||
|  |  | ||||||
|         core.info('Found the following generated jars') |  | ||||||
|         for (const jarMarker of generatedJarMarkers) { |  | ||||||
|             const generatedJar = jarMarker.substring( |  | ||||||
|                 0, |  | ||||||
|                 jarMarker.length - '.txt'.length |  | ||||||
|             ) |  | ||||||
|             core.info( |  | ||||||
|                 `Jar marker: ${jarMarker}. Looking for jar ${generatedJar}` |  | ||||||
|             ) |  | ||||||
|  |  | ||||||
|             if (!fs.existsSync(generatedJar)) { |  | ||||||
|                 // Extract the wrapper URL hash |  | ||||||
|                 const jarKey = path.basename(generatedJar) |  | ||||||
|                 const cacheKey = `gradle-generated-jar-${jarKey}` |  | ||||||
|                 core.info(`Cache key: ${cacheKey}. Cache path: ${generatedJar}`) |  | ||||||
|  |  | ||||||
|                 const restoreKey = await cache.restoreCache( |  | ||||||
|                     [generatedJar], |  | ||||||
|                     cacheKey |  | ||||||
|                 ) |  | ||||||
|                 if (restoreKey) { |  | ||||||
|                     core.info( |  | ||||||
|                         `Restored generated jar ${cacheKey} from cache to ${generatedJar}` |  | ||||||
|                     ) |  | ||||||
|                 } else { |  | ||||||
|                     core.info( |  | ||||||
|                         `Did NOT restore generated jar from ${cacheKey} to ${generatedJar}` |  | ||||||
|                     ) |  | ||||||
|                 } |  | ||||||
|             } else { |  | ||||||
|                 core.info(`Generated jar file already exists: ${generatedJar}`) |  | ||||||
|             } |             } | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
| @@ -125,30 +88,28 @@ export class GradleUserHomeCache extends AbstractCache { | |||||||
|     } |     } | ||||||
|  |  | ||||||
|     async save(): Promise<void> { |     async save(): Promise<void> { | ||||||
|         await this.cacheWrapperZips() |         await this.cacheDeduplicatedPaths() | ||||||
|         await this.cacheGeneratedJars() |  | ||||||
|         await super.save() |         await super.save() | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     private async cacheWrapperZips(): Promise<void> { |     private async cacheDeduplicatedPaths(): Promise<void> { | ||||||
|         const globber = await glob.create('~/.gradle/wrapper/dists/*/*/*.zip') |         const targetFilePatterns = DEDUPLCIATED_PATHS.join('\n') | ||||||
|         const wrapperZips = await globber.glob() |         core.info(`Using target file patterns: ${targetFilePatterns}`) | ||||||
|  |         const globber = await glob.create(targetFilePatterns) | ||||||
|  |         const targetFiles = await globber.glob() | ||||||
|  |  | ||||||
|         for (const wrapperZip of wrapperZips) { |         for (const targetFile of targetFiles) { | ||||||
|             core.info(`Wrapper zip: ${wrapperZip}`) |             core.info(`Deduplicate caching: ${targetFile}`) | ||||||
|  |  | ||||||
|             const wrapperMarkerFile = `${wrapperZip}.txt` |             const markerFile = `${targetFile}${MARKER_FILE_EXTENSION}` | ||||||
|  |  | ||||||
|             if (!fs.existsSync(wrapperMarkerFile)) { |             if (!fs.existsSync(markerFile)) { | ||||||
|                 // Extract the wrapper URL hash |                 const key = path.relative(this.getGradleUserHome(), targetFile) | ||||||
|                 const wrapperKey = path.basename(path.dirname(wrapperZip)) |                 const cacheKey = `gradle-dedup-${key}` | ||||||
|                 core.info(`Wrapper key: ${wrapperKey}`) |                 core.info(`Cache key: ${cacheKey}. Cache path: ${targetFile}`) | ||||||
|  |  | ||||||
|                 const cacheKey = `gradle-wrapper-${wrapperKey}` |  | ||||||
|  |  | ||||||
|                 core.info(`Caching wrapper with cache key: ${cacheKey}`) |  | ||||||
|                 try { |                 try { | ||||||
|                     await cache.saveCache([wrapperZip], cacheKey) |                     await cache.saveCache([targetFile], 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 ( | ||||||
| @@ -162,71 +123,28 @@ export class GradleUserHomeCache extends AbstractCache { | |||||||
|                 } |                 } | ||||||
|  |  | ||||||
|                 // Write the marker file and delete the original |                 // Write the marker file and delete the original | ||||||
|                 fs.writeFileSync(wrapperMarkerFile, 'dummy') |                 fs.writeFileSync(markerFile, 'dummy') | ||||||
|             } else { |             } else { | ||||||
|                 core.info( |                 core.info(`Marker file already exists: ${markerFile}`) | ||||||
|                     `Wrapper marker file already exists: ${wrapperMarkerFile}` |  | ||||||
|                 ) |  | ||||||
|             } |             } | ||||||
|  |  | ||||||
|             // TODO : Should not need to delete. Just exclude from cache path. |             // TODO : Should not need to delete. Just exclude from cache path. | ||||||
|             // Delete the wrapper |             // Delete the target file | ||||||
|             fs.unlinkSync(wrapperZip) |             fs.unlinkSync(targetFile) | ||||||
|         } |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     private async cacheGeneratedJars(): Promise<void> { |  | ||||||
|         const globber = await glob.create( |  | ||||||
|             '~/.gradle/caches/*/generated-gradle-jars/*.jar' |  | ||||||
|         ) |  | ||||||
|         const generatedJars = await globber.glob() |  | ||||||
|  |  | ||||||
|         for (const generatedJar of generatedJars) { |  | ||||||
|             core.info(`Generated jar: ${generatedJar}`) |  | ||||||
|  |  | ||||||
|             const generatedJarMarkerFile = `${generatedJar}.txt` |  | ||||||
|  |  | ||||||
|             if (!fs.existsSync(generatedJarMarkerFile)) { |  | ||||||
|                 // Key by jar file name: this includes Gradle version |  | ||||||
|                 const jarKey = path.basename(generatedJar) |  | ||||||
|                 const cacheKey = `gradle-generated-jar-${jarKey}` |  | ||||||
|  |  | ||||||
|                 core.info(`Caching generated jar with cache key: ${cacheKey}`) |  | ||||||
|                 try { |  | ||||||
|                     await cache.saveCache([generatedJar], cacheKey) |  | ||||||
|                 } catch (error) { |  | ||||||
|                     // Fail on validation errors or non-errors (the latter to keep Typescript happy) |  | ||||||
|                     if ( |  | ||||||
|                         error instanceof cache.ValidationError || |  | ||||||
|                         !(error instanceof Error) |  | ||||||
|                     ) { |  | ||||||
|                         throw error |  | ||||||
|                     } |  | ||||||
|                     // TODO : Avoid warning for reserve cache error: this is expected |  | ||||||
|                     core.warning(error.message) |  | ||||||
|                 } |  | ||||||
|  |  | ||||||
|                 // Write the marker file and delete the original |  | ||||||
|                 fs.writeFileSync(generatedJarMarkerFile, 'dummy') |  | ||||||
|             } else { |  | ||||||
|                 core.info( |  | ||||||
|                     `Wrapper marker file already exists: ${generatedJarMarkerFile}` |  | ||||||
|                 ) |  | ||||||
|             } |  | ||||||
|  |  | ||||||
|             // TODO : Should not need to delete. Just exclude from cache path. |  | ||||||
|             // Delete the jar |  | ||||||
|             fs.unlinkSync(generatedJar) |  | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     protected cacheOutputExists(): boolean { |     protected cacheOutputExists(): boolean { | ||||||
|         // Need to check for 'caches' directory to avoid incorrect detection on MacOS agents |         // Need to check for 'caches' directory to avoid incorrect detection on MacOS agents | ||||||
|         const dir = path.resolve(os.homedir(), '.gradle/caches') |         const dir = path.resolve(this.getGradleUserHome(), 'caches') | ||||||
|         return fs.existsSync(dir) |         return fs.existsSync(dir) | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     protected getCachePath(): string[] { |     protected getCachePath(): string[] { | ||||||
|         return CACHE_PATH |         return CACHE_PATH | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     protected getGradleUserHome(): string { | ||||||
|  |         return path.resolve(os.homedir(), '.gradle') | ||||||
|  |     } | ||||||
| } | } | ||||||
|   | |||||||
		Reference in New Issue
	
	Block a user