mirror of
https://github.com/gradle/gradle-build-action.git
synced 2025-01-12 17:01:23 -05:00
Cache wrapper zips and generated jars individually
Using a single cache entry for all files of a type is necessary to avoid overloading the cache service. However, this mechanism is not very efficient for certain artifacts like wrapper zips and generated-gradle-jars, where the same individual files are often shared between different jobs. With this change, any configured file patterns that do not end in '*' will be cached as individual files. At this time this includes downloaded wrapper zips and generated-gradle-jars. Fixes #78
This commit is contained in:
parent
b88c4086b9
commit
37f2880a8a
5 changed files with 176 additions and 133 deletions
|
@ -25,7 +25,7 @@ jobs:
|
||||||
with:
|
with:
|
||||||
build-root-directory: __tests__/samples/groovy-dsl
|
build-root-directory: __tests__/samples/groovy-dsl
|
||||||
arguments: test
|
arguments: test
|
||||||
# Add "wrapper" to main cache entry and remove 'wrapper-zips' bundle
|
# Add "wrapper" to main cache entry and remove 'wrapper-zips' cache entry
|
||||||
# Exclude build-cache from main cache entry
|
# Exclude build-cache from main cache entry
|
||||||
gradle-home-cache-includes: |
|
gradle-home-cache-includes: |
|
||||||
caches
|
caches
|
||||||
|
@ -33,7 +33,7 @@ jobs:
|
||||||
wrapper
|
wrapper
|
||||||
gradle-home-cache-excludes: |
|
gradle-home-cache-excludes: |
|
||||||
caches/build-cache-1
|
caches/build-cache-1
|
||||||
gradle-home-cache-artifact-bundles: |
|
gradle-home-extracted-cache-entries: |
|
||||||
[
|
[
|
||||||
["generated-gradle-jars", "caches/*/generated-gradle-jars/*.jar"],
|
["generated-gradle-jars", "caches/*/generated-gradle-jars/*.jar"],
|
||||||
["dependencies", "caches/modules-*/files-*/*/*/*/*/"],
|
["dependencies", "caches/modules-*/files-*/*/*/*/*/"],
|
||||||
|
@ -64,7 +64,7 @@ jobs:
|
||||||
wrapper
|
wrapper
|
||||||
gradle-home-cache-excludes: |
|
gradle-home-cache-excludes: |
|
||||||
caches/build-cache-1
|
caches/build-cache-1
|
||||||
gradle-home-cache-artifact-bundles: |
|
gradle-home-extracted-cache-entries: |
|
||||||
[
|
[
|
||||||
["generated-gradle-jars", "caches/*/generated-gradle-jars/*.jar"],
|
["generated-gradle-jars", "caches/*/generated-gradle-jars/*.jar"],
|
||||||
["dependencies", "caches/modules-*/files-*/*/*/*/*/"],
|
["dependencies", "caches/modules-*/files-*/*/*/*/*/"],
|
||||||
|
|
|
@ -45,8 +45,8 @@ jobs:
|
||||||
arguments: test --configuration-cache
|
arguments: test --configuration-cache
|
||||||
cache-read-only: true
|
cache-read-only: true
|
||||||
|
|
||||||
# Check that the build can run when no bundles are restored
|
# Check that the build can run when no extracted cache entries are restored
|
||||||
no-bundles-restored:
|
no-extracted-cache-entries-restored:
|
||||||
needs: seed-build
|
needs: seed-build
|
||||||
strategy:
|
strategy:
|
||||||
matrix:
|
matrix:
|
||||||
|
@ -55,11 +55,11 @@ jobs:
|
||||||
steps:
|
steps:
|
||||||
- name: Checkout sources
|
- name: Checkout sources
|
||||||
uses: actions/checkout@v2
|
uses: actions/checkout@v2
|
||||||
- name: Execute Gradle build with no cache artifact bundles restored
|
- name: Execute Gradle build with no cache extracted cache entries restored
|
||||||
uses: ./
|
uses: ./
|
||||||
with:
|
with:
|
||||||
build-root-directory: __tests__/samples/groovy-dsl
|
build-root-directory: __tests__/samples/groovy-dsl
|
||||||
arguments: test --configuration-cache
|
arguments: test --configuration-cache
|
||||||
cache-read-only: true
|
cache-read-only: true
|
||||||
gradle-home-cache-artifact-bundles: '[]'
|
gradle-home-extracted-cache-entries: '[]'
|
||||||
|
|
||||||
|
|
|
@ -59,8 +59,8 @@ jobs:
|
||||||
arguments: test -DverifyCachedBuild=true
|
arguments: test -DverifyCachedBuild=true
|
||||||
cache-read-only: true
|
cache-read-only: true
|
||||||
|
|
||||||
# Check that the build can run when no bundles are restored
|
# Check that the build can run when Gradle User Home is not fully restored
|
||||||
no-bundles-restored:
|
no-extracted-cache-entries-restored:
|
||||||
needs: seed-build
|
needs: seed-build
|
||||||
strategy:
|
strategy:
|
||||||
matrix:
|
matrix:
|
||||||
|
@ -69,11 +69,11 @@ jobs:
|
||||||
steps:
|
steps:
|
||||||
- name: Checkout sources
|
- name: Checkout sources
|
||||||
uses: actions/checkout@v2
|
uses: actions/checkout@v2
|
||||||
- name: Execute Gradle build with no cache artifact bundles restored
|
- name: Execute Gradle build with no extracted cache entries restored
|
||||||
uses: ./
|
uses: ./
|
||||||
with:
|
with:
|
||||||
build-root-directory: __tests__/samples/groovy-dsl
|
build-root-directory: __tests__/samples/groovy-dsl
|
||||||
arguments: test
|
arguments: test
|
||||||
cache-read-only: true
|
cache-read-only: true
|
||||||
gradle-home-cache-artifact-bundles: '[]'
|
gradle-home-extracted-cache-entries: '[]'
|
||||||
|
|
||||||
|
|
|
@ -55,8 +55,8 @@ 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 (INTERNAL).
|
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) }}
|
||||||
gradle-home-cache-artifact-bundles:
|
gradle-home-extracted-cache-entries:
|
||||||
description: Names and patterns of artifact bundles to cache separately. (EXPERIMENTAL - may be changed/removed without notice)
|
description: Names and patterns of artifacts in Gradle User Home to cache separately. (EXPERIMENTAL - may be changed/removed without notice)
|
||||||
required: false
|
required: false
|
||||||
default: |
|
default: |
|
||||||
[
|
[
|
||||||
|
|
|
@ -13,25 +13,36 @@ const META_FILE = 'cache-metadata.json'
|
||||||
|
|
||||||
const INCLUDE_PATHS_PARAMETER = 'gradle-home-cache-includes'
|
const INCLUDE_PATHS_PARAMETER = 'gradle-home-cache-includes'
|
||||||
const EXCLUDE_PATHS_PARAMETER = 'gradle-home-cache-excludes'
|
const EXCLUDE_PATHS_PARAMETER = 'gradle-home-cache-excludes'
|
||||||
const ARTIFACT_BUNDLES_PARAMETER = 'gradle-home-cache-artifact-bundles'
|
const EXTRACTED_CACHE_ENTRIES_PARAMETER = 'gradle-home-extracted-cache-entries'
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Represents the result of attempting to load or store a cache bundle entry.
|
* Represents the result of attempting to load or store an extracted cache entry.
|
||||||
* An undefined cacheKey indicates that the operation did not succeed.
|
* An undefined cacheKey indicates that the operation did not succeed.
|
||||||
* The collected results are then used to populate the `cache-metadata.json` file for later use.
|
* The collected results are then used to populate the `cache-metadata.json` file for later use.
|
||||||
*/
|
*/
|
||||||
class CacheBundleResult {
|
class ExtractedCacheEntry {
|
||||||
readonly bundle: string
|
artifactType: string
|
||||||
readonly cacheKey: string | undefined
|
pattern: string
|
||||||
|
cacheKey: string | undefined
|
||||||
|
|
||||||
constructor(bundle: string, cacheKey: string | undefined) {
|
constructor(artifactType: string, pattern: string, cacheKey: string | undefined) {
|
||||||
this.bundle = bundle
|
this.artifactType = artifactType
|
||||||
|
this.pattern = pattern
|
||||||
this.cacheKey = cacheKey
|
this.cacheKey = cacheKey
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Caches and restores the entire Gradle User Home directory, extracting bundles of common artifacts
|
* Representation of all of the extracted cache entries for this Gradle User Home.
|
||||||
|
* This object is persisted to JSON file in the Gradle User Home directory for storing,
|
||||||
|
* and subsequently used to restore the Gradle User Home.
|
||||||
|
*/
|
||||||
|
class ExtractedCacheEntryMetadata {
|
||||||
|
entries: ExtractedCacheEntry[] = []
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Caches and restores the entire Gradle User Home directory, extracting entries containing common artifacts
|
||||||
* for more efficient storage.
|
* for more efficient storage.
|
||||||
*/
|
*/
|
||||||
export class GradleUserHomeCache extends AbstractCache {
|
export class GradleUserHomeCache extends AbstractCache {
|
||||||
|
@ -48,73 +59,73 @@ export class GradleUserHomeCache extends AbstractCache {
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Restore any artifact bundles after the main Gradle User Home entry is restored.
|
* Restore any extracted cache entries after the main Gradle User Home entry is restored.
|
||||||
*/
|
*/
|
||||||
async afterRestore(listener: CacheListener): Promise<void> {
|
async afterRestore(listener: CacheListener): Promise<void> {
|
||||||
await this.debugReportGradleUserHomeSize('as restored from cache')
|
await this.debugReportGradleUserHomeSize('as restored from cache')
|
||||||
await this.restoreArtifactBundles(listener)
|
await this.restoreExtractedCacheEntries(listener)
|
||||||
await this.debugReportGradleUserHomeSize('after restoring common artifacts')
|
await this.debugReportGradleUserHomeSize('after restoring common artifacts')
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Restores any artifacts that were cached separately, based on the information in the `cache-metadata.json` file.
|
* Restores any artifacts that were cached separately, based on the information in the `cache-metadata.json` file.
|
||||||
* Each artifact bundle is restored in parallel, except when debugging is enabled.
|
* Each extracted cache entry is restored in parallel, except when debugging is enabled.
|
||||||
*/
|
*/
|
||||||
private async restoreArtifactBundles(listener: CacheListener): Promise<void> {
|
private async restoreExtractedCacheEntries(listener: CacheListener): Promise<void> {
|
||||||
const bundleMetadata = this.loadBundleMetadata()
|
const extractedCacheEntryDefinitions = this.getExtractedCacheEntryDefinitions()
|
||||||
const bundlePatterns = this.getArtifactBundleDefinitions()
|
const previouslyExtractedCacheEntries = this.loadExtractedCacheEntries()
|
||||||
|
|
||||||
const processes: Promise<CacheBundleResult>[] = []
|
const processes: Promise<ExtractedCacheEntry>[] = []
|
||||||
|
|
||||||
for (const [bundle, cacheKey] of bundleMetadata) {
|
for (const cacheEntry of previouslyExtractedCacheEntries) {
|
||||||
const entryListener = listener.entry(bundle)
|
const artifactType = cacheEntry.artifactType
|
||||||
const bundlePattern = bundlePatterns.get(bundle)
|
const entryListener = listener.entry(cacheEntry.pattern)
|
||||||
|
|
||||||
// Handle case where the 'artifactBundlePatterns' have been changed
|
// Handle case where the extracted-cache-entry definitions have been changed
|
||||||
if (bundlePattern === undefined) {
|
if (extractedCacheEntryDefinitions.get(artifactType) === undefined) {
|
||||||
core.info(`Found bundle metadata for ${bundle} but no such bundle defined`)
|
core.info(`Found extracted cache entry for ${artifactType} but no such entry defined`)
|
||||||
entryListener.markRequested('BUNDLE_NOT_CONFIGURED')
|
entryListener.markRequested('EXTRACTED_ENTRY_NOT_DEFINED')
|
||||||
} else {
|
} else {
|
||||||
const p = this.restoreArtifactBundle(bundle, cacheKey, bundlePattern, entryListener)
|
processes.push(
|
||||||
// Run sequentially when debugging enabled
|
this.restoreExtractedCacheEntry(
|
||||||
if (this.cacheDebuggingEnabled) {
|
artifactType,
|
||||||
await p
|
cacheEntry.cacheKey!,
|
||||||
}
|
cacheEntry.pattern,
|
||||||
processes.push(p)
|
entryListener
|
||||||
|
)
|
||||||
|
)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
const results = await Promise.all(processes)
|
this.saveMetadataForCacheResults(await this.collectCacheResults(processes))
|
||||||
|
|
||||||
this.saveMetadataForCacheResults(results)
|
|
||||||
}
|
}
|
||||||
|
|
||||||
private async restoreArtifactBundle(
|
private async restoreExtractedCacheEntry(
|
||||||
bundle: string,
|
artifactType: string,
|
||||||
cacheKey: string,
|
cacheKey: string,
|
||||||
bundlePattern: string,
|
pattern: string,
|
||||||
listener: CacheEntryListener
|
listener: CacheEntryListener
|
||||||
): Promise<CacheBundleResult> {
|
): Promise<ExtractedCacheEntry> {
|
||||||
listener.markRequested(cacheKey)
|
listener.markRequested(cacheKey)
|
||||||
|
|
||||||
const restoredEntry = await this.restoreCache([bundlePattern], cacheKey)
|
const restoredEntry = await this.restoreCache([pattern], cacheKey)
|
||||||
if (restoredEntry) {
|
if (restoredEntry) {
|
||||||
core.info(`Restored ${bundle} with key ${cacheKey} to ${bundlePattern}`)
|
core.info(`Restored ${artifactType} with key ${cacheKey} to ${pattern}`)
|
||||||
listener.markRestored(restoredEntry.key, restoredEntry.size)
|
listener.markRestored(restoredEntry.key, restoredEntry.size)
|
||||||
return new CacheBundleResult(bundle, cacheKey)
|
return new ExtractedCacheEntry(artifactType, pattern, cacheKey)
|
||||||
} else {
|
} else {
|
||||||
core.info(`Did not restore ${bundle} with key ${cacheKey} to ${bundlePattern}`)
|
core.info(`Did not restore ${artifactType} with key ${cacheKey} to ${pattern}`)
|
||||||
return new CacheBundleResult(bundle, undefined)
|
return new ExtractedCacheEntry(artifactType, pattern, undefined)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Save and delete any artifact bundles prior to the main Gradle User Home entry being saved.
|
* Extract and save any defined extracted cache entries prior to the main Gradle User Home entry being saved.
|
||||||
*/
|
*/
|
||||||
async beforeSave(listener: CacheListener): Promise<void> {
|
async beforeSave(listener: CacheListener): Promise<void> {
|
||||||
await this.debugReportGradleUserHomeSize('before saving common artifacts')
|
await this.debugReportGradleUserHomeSize('before saving common artifacts')
|
||||||
this.removeExcludedPaths()
|
this.removeExcludedPaths()
|
||||||
await this.saveArtifactBundles(listener)
|
await this.saveExtractedCacheEntries(listener)
|
||||||
await this.debugReportGradleUserHomeSize(
|
await this.debugReportGradleUserHomeSize(
|
||||||
"after saving common artifacts (only 'caches' and 'notifications' will be stored)"
|
"after saving common artifacts (only 'caches' and 'notifications' will be stored)"
|
||||||
)
|
)
|
||||||
|
@ -134,110 +145,143 @@ export class GradleUserHomeCache extends AbstractCache {
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Saves any artifacts that are configured to be cached separately, based on the artifact bundle definitions.
|
* Saves any artifacts that are configured to be cached separately, based on the extracted cache entry definitions.
|
||||||
* These definitions are normally fixed, but can be overridden by the `gradle-home-cache-artifact-bundles` parameter.
|
* These definitions are normally fixed, but can be overridden by the `gradle-home-extracted-cache-entries` parameter.
|
||||||
* Each artifact bundle is saved in parallel, except when debugging is enabled.
|
* Each entry is extracted and saved in parallel, except when debugging is enabled.
|
||||||
*/
|
*/
|
||||||
private async saveArtifactBundles(listener: CacheListener): Promise<void> {
|
private async saveExtractedCacheEntries(listener: CacheListener): Promise<void> {
|
||||||
const bundleMetadata = this.loadBundleMetadata()
|
// Load the cache entry definitions (from config) and the previously restored entries (from filesystem)
|
||||||
|
const cacheEntryDefinitions = this.getExtractedCacheEntryDefinitions()
|
||||||
|
const previouslyRestoredEntries = this.loadExtractedCacheEntries()
|
||||||
|
const cacheActions: Promise<ExtractedCacheEntry>[] = []
|
||||||
|
|
||||||
const processes: Promise<CacheBundleResult>[] = []
|
for (const [artifactType, pattern] of cacheEntryDefinitions) {
|
||||||
for (const [bundle, pattern] of this.getArtifactBundleDefinitions()) {
|
// Find all matching files for this cache entry definition
|
||||||
const entryListener = listener.entry(bundle)
|
const globber = await glob.create(pattern, {
|
||||||
const previouslyRestoredKey = bundleMetadata.get(bundle)
|
|
||||||
const p = this.saveArtifactBundle(bundle, pattern, previouslyRestoredKey, entryListener)
|
|
||||||
// Run sequentially when debugging enabled
|
|
||||||
if (this.cacheDebuggingEnabled) {
|
|
||||||
await p
|
|
||||||
}
|
|
||||||
processes.push(p)
|
|
||||||
}
|
|
||||||
|
|
||||||
const results = await Promise.all(processes)
|
|
||||||
|
|
||||||
this.saveMetadataForCacheResults(results)
|
|
||||||
}
|
|
||||||
|
|
||||||
private async saveArtifactBundle(
|
|
||||||
bundle: string,
|
|
||||||
artifactPath: string,
|
|
||||||
previouslyRestoredKey: string | undefined,
|
|
||||||
listener: CacheEntryListener
|
|
||||||
): Promise<CacheBundleResult> {
|
|
||||||
const globber = await glob.create(artifactPath, {
|
|
||||||
implicitDescendants: false,
|
implicitDescendants: false,
|
||||||
followSymbolicLinks: false
|
followSymbolicLinks: false
|
||||||
})
|
})
|
||||||
const bundleFiles = await globber.glob()
|
const matchingFiles = await globber.glob()
|
||||||
|
|
||||||
// Handle no matching files
|
if (matchingFiles.length === 0) {
|
||||||
if (bundleFiles.length === 0) {
|
this.debug(`No files found to cache for ${artifactType}`)
|
||||||
this.debug(`No files found to cache for ${bundle}`)
|
continue
|
||||||
return new CacheBundleResult(bundle, undefined)
|
|
||||||
}
|
}
|
||||||
|
|
||||||
const cacheKey = this.createCacheKeyForArtifacts(bundle, bundleFiles)
|
if (this.isBundlePattern(pattern)) {
|
||||||
|
// For an extracted "bundle", use the defined pattern and cache all matching files in a single entry.
|
||||||
|
cacheActions.push(
|
||||||
|
this.saveExtractedCacheEntry(
|
||||||
|
matchingFiles,
|
||||||
|
artifactType,
|
||||||
|
pattern,
|
||||||
|
previouslyRestoredEntries,
|
||||||
|
listener.entry(pattern)
|
||||||
|
)
|
||||||
|
)
|
||||||
|
} else {
|
||||||
|
// Otherwise cache each matching file in a separate entry, using the complete file path as the cache pattern.
|
||||||
|
for (const cacheFile of matchingFiles) {
|
||||||
|
cacheActions.push(
|
||||||
|
this.saveExtractedCacheEntry(
|
||||||
|
[cacheFile],
|
||||||
|
artifactType,
|
||||||
|
cacheFile,
|
||||||
|
previouslyRestoredEntries,
|
||||||
|
listener.entry(cacheFile)
|
||||||
|
)
|
||||||
|
)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
this.saveMetadataForCacheResults(await this.collectCacheResults(cacheActions))
|
||||||
|
}
|
||||||
|
|
||||||
|
private async saveExtractedCacheEntry(
|
||||||
|
matchingFiles: string[],
|
||||||
|
artifactType: string,
|
||||||
|
pattern: string,
|
||||||
|
previouslyRestoredEntries: ExtractedCacheEntry[],
|
||||||
|
entryListener: CacheEntryListener
|
||||||
|
): Promise<ExtractedCacheEntry> {
|
||||||
|
const cacheKey = this.createCacheKeyForArtifacts(artifactType, matchingFiles)
|
||||||
|
const previouslyRestoredKey = previouslyRestoredEntries.find(
|
||||||
|
x => x.artifactType === artifactType && x.pattern === pattern
|
||||||
|
)?.cacheKey
|
||||||
|
|
||||||
if (previouslyRestoredKey === cacheKey) {
|
if (previouslyRestoredKey === cacheKey) {
|
||||||
this.debug(`No change to previously restored ${bundle}. Not caching.`)
|
this.debug(`No change to previously restored ${artifactType}. Not saving.`)
|
||||||
} else {
|
} else {
|
||||||
core.info(`Caching ${bundle} with cache key: ${cacheKey}`)
|
core.info(`Caching ${artifactType} with path '${pattern}' and cache key: ${cacheKey}`)
|
||||||
const savedEntry = await this.saveCache([artifactPath], cacheKey)
|
const savedEntry = await this.saveCache([pattern], cacheKey)
|
||||||
if (savedEntry !== undefined) {
|
if (savedEntry !== undefined) {
|
||||||
listener.markSaved(savedEntry.key, savedEntry.size)
|
entryListener.markSaved(savedEntry.key, savedEntry.size)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
for (const file of bundleFiles) {
|
for (const file of matchingFiles) {
|
||||||
tryDelete(file)
|
tryDelete(file)
|
||||||
}
|
}
|
||||||
|
|
||||||
return new CacheBundleResult(bundle, cacheKey)
|
return new ExtractedCacheEntry(artifactType, pattern, cacheKey)
|
||||||
}
|
}
|
||||||
|
|
||||||
protected createCacheKeyForArtifacts(bundle: string, files: string[]): string {
|
protected createCacheKeyForArtifacts(artifactType: string, files: string[]): string {
|
||||||
const cacheKeyPrefix = getCacheKeyPrefix()
|
const cacheKeyPrefix = getCacheKeyPrefix()
|
||||||
const relativeFiles = files.map(x => path.relative(this.gradleUserHome, x))
|
const relativeFiles = files.map(x => path.relative(this.gradleUserHome, x))
|
||||||
const key = hashFileNames(relativeFiles)
|
const key = hashFileNames(relativeFiles)
|
||||||
|
|
||||||
this.debug(`Generating cache key for ${bundle} from files: ${relativeFiles}`)
|
this.debug(`Generating cache key for ${artifactType} from files: ${relativeFiles}`)
|
||||||
|
|
||||||
return `${cacheKeyPrefix}${bundle}-${key}`
|
return `${cacheKeyPrefix}${artifactType}-${key}`
|
||||||
|
}
|
||||||
|
|
||||||
|
private isBundlePattern(pattern: string): boolean {
|
||||||
|
return pattern.endsWith('*')
|
||||||
|
}
|
||||||
|
|
||||||
|
private async collectCacheResults(processes: Promise<ExtractedCacheEntry>[]): Promise<ExtractedCacheEntry[]> {
|
||||||
|
// Run cache actions sequentially when debugging enabled
|
||||||
|
if (this.cacheDebuggingEnabled) {
|
||||||
|
for (const p of processes) {
|
||||||
|
await p
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return await Promise.all(processes)
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Load information about the previously restored/saved artifact bundles from the 'cache-metadata.json' file.
|
* Load information about the extracted cache entries previously restored/saved. This is loaded from the 'cache-metadata.json' file.
|
||||||
*/
|
*/
|
||||||
private loadBundleMetadata(): Map<string, string> {
|
private loadExtractedCacheEntries(): ExtractedCacheEntry[] {
|
||||||
const bundleMetaFile = path.resolve(this.gradleUserHome, META_FILE_DIR, META_FILE)
|
const cacheMetadataFile = path.resolve(this.gradleUserHome, META_FILE_DIR, META_FILE)
|
||||||
if (!fs.existsSync(bundleMetaFile)) {
|
if (!fs.existsSync(cacheMetadataFile)) {
|
||||||
return new Map<string, string>()
|
return []
|
||||||
}
|
}
|
||||||
const filedata = fs.readFileSync(bundleMetaFile, 'utf-8')
|
|
||||||
core.debug(`Loaded bundle metadata: ${filedata}`)
|
const filedata = fs.readFileSync(cacheMetadataFile, 'utf-8')
|
||||||
return new Map(JSON.parse(filedata))
|
core.debug(`Loaded cache metadata: ${filedata}`)
|
||||||
|
const extractedCacheEntryMetadata = JSON.parse(filedata) as ExtractedCacheEntryMetadata
|
||||||
|
return extractedCacheEntryMetadata.entries
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Saves information about the artifact bundle restore/save into the 'cache-metadata.json' file.
|
* Saves information about the extracted cache entries into the 'cache-metadata.json' file.
|
||||||
*/
|
*/
|
||||||
private saveMetadataForCacheResults(results: CacheBundleResult[]): void {
|
private saveMetadataForCacheResults(results: ExtractedCacheEntry[]): void {
|
||||||
const metadata = new Map<string, string>()
|
const extractedCacheEntryMetadata = new ExtractedCacheEntryMetadata()
|
||||||
for (const result of results) {
|
extractedCacheEntryMetadata.entries = results.filter(x => x.cacheKey !== undefined)
|
||||||
if (result.cacheKey !== undefined) {
|
|
||||||
metadata.set(result.bundle, result.cacheKey)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
const filedata = JSON.stringify(Array.from(metadata))
|
|
||||||
core.debug(`Saving bundle metadata: ${filedata}`)
|
|
||||||
|
|
||||||
const bundleMetaDir = path.resolve(this.gradleUserHome, META_FILE_DIR)
|
const filedata = JSON.stringify(extractedCacheEntryMetadata)
|
||||||
const bundleMetaFile = path.resolve(bundleMetaDir, META_FILE)
|
core.debug(`Saving cache metadata: ${filedata}`)
|
||||||
|
|
||||||
if (!fs.existsSync(bundleMetaDir)) {
|
const actionMetadataDirectory = path.resolve(this.gradleUserHome, META_FILE_DIR)
|
||||||
fs.mkdirSync(bundleMetaDir, {recursive: true})
|
const cacheMetadataFile = path.resolve(actionMetadataDirectory, META_FILE)
|
||||||
}
|
|
||||||
fs.writeFileSync(bundleMetaFile, filedata, 'utf-8')
|
fs.mkdirSync(actionMetadataDirectory, {recursive: true})
|
||||||
|
fs.writeFileSync(cacheMetadataFile, filedata, 'utf-8')
|
||||||
}
|
}
|
||||||
|
|
||||||
protected determineGradleUserHome(rootDir: string): string {
|
protected determineGradleUserHome(rootDir: string): string {
|
||||||
|
@ -277,15 +321,14 @@ export class GradleUserHomeCache extends AbstractCache {
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Return the artifact bundle definitions, which determine which artifacts will be cached
|
* Return the extracted cache entry definitions, which determine which artifacts will be cached
|
||||||
* separately from the rest of the Gradle User Home cache entry.
|
* separately from the rest of the Gradle User Home cache entry.
|
||||||
* This is normally a fixed set, but can be overridden by the `gradle-home-cache-artifact-bundles` parameter.
|
* This is normally a fixed set, but can be overridden by the `gradle-home-extracted-cache-entries` parameter.
|
||||||
*/
|
*/
|
||||||
private getArtifactBundleDefinitions(): Map<string, string> {
|
private getExtractedCacheEntryDefinitions(): Map<string, string> {
|
||||||
const artifactBundleDefinition = core.getInput(ARTIFACT_BUNDLES_PARAMETER)
|
const rawDefinitions = core.getInput(EXTRACTED_CACHE_ENTRIES_PARAMETER)
|
||||||
this.debug(`Using artifact bundle definition: ${artifactBundleDefinition}`)
|
const parsedDefinitions = JSON.parse(rawDefinitions)
|
||||||
const artifactBundles = JSON.parse(artifactBundleDefinition)
|
return new Map(Array.from(parsedDefinitions, ([key, value]) => [key, path.resolve(this.gradleUserHome, value)]))
|
||||||
return new Map(Array.from(artifactBundles, ([key, value]) => [key, path.resolve(this.gradleUserHome, value)]))
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
|
Loading…
Reference in a new issue