Cache artifacts with single entry per type

When caching is too fine-grained, an excessive number of cache
requests can result in HTTP 429 errors due to rate limiting.
By caching all artifacts of a particular type in a single entry
we hope to mitigate this, at the expense of some reduction in
cache space optimization.

This change also adds caching for all dependency jars, as well as
instrumented jars in the 'caches/jars-X' directory.
This commit is contained in:
Daz DeBoer 2021-09-15 06:36:54 -06:00
parent dbb485d80d
commit 6084a4eb65
No known key found for this signature in database
GPG key ID: DD6B9F0B06683D5D
5 changed files with 76 additions and 69 deletions

2
dist/main/index.js vendored

File diff suppressed because one or more lines are too long

File diff suppressed because one or more lines are too long

2
dist/post/index.js vendored

File diff suppressed because one or more lines are too long

File diff suppressed because one or more lines are too long

View file

@ -5,27 +5,19 @@ 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} from './cache-utils' import {AbstractCache, hashStrings} from './cache-utils'
// When a common artifact is cached separately, it is replaced by a marker file to allow for restore.
const MARKER_FILE_EXTENSION = '.cached'
// Which paths under Gradle User Home should be cached // Which paths under Gradle User Home should be cached
// TODO: This should adapt for the `GRADLE_USER_HOME` environment variable // TODO: This should adapt for the `GRADLE_USER_HOME` environment variable
// TODO: Allow the user to override / tweak this set // TODO: Allow the user to override / tweak this set
const CACHE_PATH = [ const CACHE_PATH = ['~/.gradle/caches', '~/.gradle/notifications']
'~/.gradle/caches',
'~/.gradle/notifications', // Prevent the re-rendering of first-use message for version
`~/.gradle/wrapper/dists/*/*/*.zip${MARKER_FILE_EXTENSION}` // Only cache/restore wrapper zips: Gradle will automatically expand these on startup if required
]
// Paths to artifacts that are common to all/many Gradle User Home caches const COMMON_ARTIFACT_CACHES = new Map([
// These artifacts are cached separately to avoid blowing out the size of each GUH cache ['generated-gradle-jars', '~/.gradle/caches/*/generated-gradle-jars/*.jar'],
// TODO: Allow the user to override / tweak this set ['wrapper-zips', '~/.gradle/wrapper/dists/*/*/*.zip'],
const COMMON_ARTIFACT_PATHS = [ ['dependency-jars', '~/.gradle/caches/modules-*/files-*/**/*.jar'],
'~/.gradle/caches/*/generated-gradle-jars/*.jar', ['instrumented-jars', '~/.gradle/caches/jars-*/*/*.jar']
'~/.gradle/wrapper/dists/*/*/*.zip' ])
]
export class GradleUserHomeCache extends AbstractCache { export class GradleUserHomeCache extends AbstractCache {
constructor() { constructor() {
@ -39,16 +31,9 @@ export class GradleUserHomeCache extends AbstractCache {
} }
private async restoreCommonArtifacts(): Promise<void> { private async restoreCommonArtifacts(): Promise<void> {
const markerFilePatterns = COMMON_ARTIFACT_PATHS.map(targetPath => {
return targetPath + MARKER_FILE_EXTENSION
}).join('\n')
const globber = await glob.create(markerFilePatterns)
const markerFiles = await globber.glob()
const processes: Promise<void>[] = [] const processes: Promise<void>[] = []
for (const markerFile of markerFiles) { for (const [bundle, pattern] of COMMON_ARTIFACT_CACHES) {
const p = this.restoreCommonArtifact(markerFile) const p = this.restoreCommonArtifactBundle(bundle, pattern)
// Run sequentially when debugging enabled // Run sequentially when debugging enabled
if (this.cacheDebuggingEnabled) { if (this.cacheDebuggingEnabled) {
await p await p
@ -59,31 +44,38 @@ export class GradleUserHomeCache extends AbstractCache {
await Promise.all(processes) await Promise.all(processes)
} }
private async restoreCommonArtifact(markerFile: string): Promise<void> { private async restoreCommonArtifactBundle(
const artifactFile = markerFile.substring( bundle: string,
0, pattern: string
markerFile.length - MARKER_FILE_EXTENSION.length ): Promise<void> {
) const cacheMetaFile = this.getCacheMetaFile(bundle)
if (fs.existsSync(cacheMetaFile)) {
if (!fs.existsSync(artifactFile)) { const cacheKey = fs.readFileSync(cacheMetaFile, 'utf-8').trim()
const key = path.relative(this.getGradleUserHome(), artifactFile) const restoreKey = await this.restoreCache([pattern], cacheKey)
const cacheKey = `gradle-artifact-${key}`
const restoreKey = await this.restoreCache([artifactFile], cacheKey)
if (restoreKey) { if (restoreKey) {
this.debug(`Restored ${cacheKey} from cache to ${artifactFile}`) this.debug(
`Restored ${bundle} with key ${cacheKey} to ${pattern}`
)
} else { } else {
this.debug( this.debug(
`Failed to restore from ${cacheKey} to ${artifactFile}` `Failed to restore ${bundle} with key ${cacheKey} to ${pattern}`
) )
} }
} else { } else {
this.debug( this.debug(
`Artifact file already exists, not restoring: ${artifactFile}` `No metafile found to restore ${bundle}: ${cacheMetaFile}`
) )
} }
} }
private getCacheMetaFile(name: string): string {
return path.resolve(
this.getGradleUserHome(),
'caches',
`.gradle-build-action.${name}.cache`
)
}
private async reportCacheEntrySize(label: string): Promise<void> { private async reportCacheEntrySize(label: string): Promise<void> {
if (!this.cacheDebuggingEnabled) { if (!this.cacheDebuggingEnabled) {
return return
@ -123,13 +115,9 @@ export class GradleUserHomeCache extends AbstractCache {
} }
private async saveCommonArtifacts(): Promise<void> { private async saveCommonArtifacts(): Promise<void> {
const artifactFilePatterns = COMMON_ARTIFACT_PATHS.join('\n')
const globber = await glob.create(artifactFilePatterns)
const commonArtifactFiles = await globber.glob()
const processes: Promise<void>[] = [] const processes: Promise<void>[] = []
for (const artifactFile of commonArtifactFiles) { for (const [bundle, pattern] of COMMON_ARTIFACT_CACHES) {
const p = this.saveCommonArtifact(artifactFile) const p = this.saveCommonArtifactBundle(bundle, pattern)
// Run sequentially when debugging enabled // Run sequentially when debugging enabled
if (this.cacheDebuggingEnabled) { if (this.cacheDebuggingEnabled) {
await p await p
@ -140,30 +128,49 @@ export class GradleUserHomeCache extends AbstractCache {
await Promise.all(processes) await Promise.all(processes)
} }
private async saveCommonArtifact(artifactFile: string): Promise<void> { private async saveCommonArtifactBundle(
const markerFile = `${artifactFile}${MARKER_FILE_EXTENSION}` bundle: string,
pattern: string
): Promise<void> {
const cacheMetaFile = this.getCacheMetaFile(bundle)
if (!fs.existsSync(markerFile)) { const globber = await glob.create(pattern)
const filePath = path.relative( const commonArtifactFiles = await globber.glob()
this.getGradleUserHome(),
artifactFile
)
const cacheKey = `gradle-artifact-${filePath}`
this.debug(`Caching ${artifactFile} with cache key: ${cacheKey}`) // Handle no matching files
await this.saveCache([artifactFile], cacheKey) if (commonArtifactFiles.length === 0) {
this.debug(`No files found to cache for ${bundle}`)
// Write the marker file that will stand in place of the original if (fs.existsSync(cacheMetaFile)) {
fs.writeFileSync(markerFile, 'cached') fs.unlinkSync(cacheMetaFile)
} else { }
this.debug( return
`Marker file already exists: ${markerFile}. Not caching ${artifactFile}`
)
} }
// TODO : Should not need to delete. Just exclude from cache path. const previouslyRestoredKey = fs.existsSync(cacheMetaFile)
// Delete the original artifact file ? fs.readFileSync(cacheMetaFile, 'utf-8').trim()
fs.unlinkSync(artifactFile) : ''
const cacheKey = this.createCacheKey(hashStrings(commonArtifactFiles))
if (previouslyRestoredKey === cacheKey) {
this.debug(
`No change to previously restored ${bundle}. Not caching.`
)
} else {
this.debug(`Caching ${bundle} with cache key: ${cacheKey}`)
await this.saveCache([pattern], cacheKey)
this.debug(`Writing cache metafile: ${cacheMetaFile}`)
fs.writeFileSync(cacheMetaFile, cacheKey)
}
for (const file of commonArtifactFiles) {
fs.unlinkSync(file)
}
}
protected createCacheKey(key: string): string {
const cacheKeyPrefix = process.env['CACHE_KEY_PREFIX'] || ''
return `${cacheKeyPrefix}${key}`
} }
protected getGradleUserHome(): string { protected getGradleUserHome(): string {