mirror of
https://github.com/gradle/gradle-build-action.git
synced 2024-11-23 01:22:50 +00:00
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:
parent
dbb485d80d
commit
6084a4eb65
5 changed files with 76 additions and 69 deletions
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
|
@ -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 {
|
||||||
|
|
Loading…
Reference in a new issue