Store all cache metadata in a single JSON file

Instead of storing cache metadata in a separate .cache file per
artifact bundle, all of the metadata is now stored in a single `.json` file.
This will make it easier to implement more flexible artifact-caching
strategies, such as caching each wrapper zip separately.
This commit is contained in:
Daz DeBoer 2021-11-10 20:11:55 -07:00
parent 4fab1e5c7f
commit 201b58add4
No known key found for this signature in database
GPG key ID: DD6B9F0B06683D5D

View file

@ -14,6 +14,16 @@ 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 ARTIFACT_BUNDLES_PARAMETER = 'gradle-home-cache-artifact-bundles'
class CacheResult {
readonly bundle: string
readonly cacheKey: string | undefined
constructor(bundle: string, cacheKey: string | undefined) {
this.bundle = bundle
this.cacheKey = cacheKey
}
}
export class GradleUserHomeCache extends AbstractCache { export class GradleUserHomeCache extends AbstractCache {
private gradleUserHome: string private gradleUserHome: string
@ -34,24 +44,21 @@ export class GradleUserHomeCache extends AbstractCache {
} }
private async restoreArtifactBundles(listener: CacheListener): Promise<void> { private async restoreArtifactBundles(listener: CacheListener): Promise<void> {
const processes: Promise<void>[] = [] const bundleMetadata = this.loadBundleMetadata()
const bundleMetaFiles = await this.getBundleMetaFiles()
const bundlePatterns = this.getArtifactBundles() const bundlePatterns = this.getArtifactBundles()
// Iterate over all bundle meta files and try to restore const processes: Promise<CacheResult>[] = []
for (const bundleMetaFile of bundleMetaFiles) {
const bundle = path.basename(bundleMetaFile, '.cache') for (const [bundle, cacheKey] of bundleMetadata) {
const entryListener = listener.entry(bundle) const entryListener = listener.entry(bundle)
const bundlePattern = bundlePatterns.get(bundle) const bundlePattern = bundlePatterns.get(bundle)
// Handle case where the 'artifactBundlePatterns' have been changed // Handle case where the 'artifactBundlePatterns' have been changed
if (bundlePattern === undefined) { if (bundlePattern === undefined) {
core.info(`Found bundle metafile for ${bundle} but no such bundle defined`) core.info(`Found bundle metadata for ${bundle} but no such bundle defined`)
entryListener.markRequested('BUNDLE_NOT_CONFIGURED') entryListener.markRequested('BUNDLE_NOT_CONFIGURED')
tryDelete(bundleMetaFile)
} else { } else {
const p = this.restoreArtifactBundle(bundle, bundlePattern, bundleMetaFile, entryListener) const p = this.restoreArtifactBundle(bundle, cacheKey, bundlePattern, entryListener)
// Run sequentially when debugging enabled // Run sequentially when debugging enabled
if (this.cacheDebuggingEnabled) { if (this.cacheDebuggingEnabled) {
await p await p
@ -60,39 +67,30 @@ export class GradleUserHomeCache extends AbstractCache {
} }
} }
await Promise.all(processes) const results = await Promise.all(processes)
this.saveMetadataForCacheResults(results)
} }
private async restoreArtifactBundle( private async restoreArtifactBundle(
bundle: string, bundle: string,
cacheKey: string,
bundlePattern: string, bundlePattern: string,
bundleMetaFile: string,
listener: CacheEntryListener listener: CacheEntryListener
): Promise<void> { ): Promise<CacheResult> {
const cacheKey = fs.readFileSync(bundleMetaFile, 'utf-8').trim()
listener.markRequested(cacheKey) listener.markRequested(cacheKey)
const restoredEntry = await this.restoreCache([bundlePattern], cacheKey) const restoredEntry = await this.restoreCache([bundlePattern], cacheKey)
if (restoredEntry) { if (restoredEntry) {
core.info(`Restored ${bundle} with key ${cacheKey} to ${bundlePattern}`) core.info(`Restored ${bundle} with key ${cacheKey} to ${bundlePattern}`)
listener.markRestored(restoredEntry.key, restoredEntry.size) listener.markRestored(restoredEntry.key, restoredEntry.size)
return new CacheResult(bundle, cacheKey)
} else { } else {
core.info(`Did not restore ${bundle} with key ${cacheKey} to ${bundlePattern}`) core.info(`Did not restore ${bundle} with key ${cacheKey} to ${bundlePattern}`)
tryDelete(bundleMetaFile) return new CacheResult(bundle, undefined)
} }
} }
private getBundleMetaFile(name: string): string {
return path.resolve(this.gradleUserHome, META_FILE_DIR, `${name}.cache`)
}
private async getBundleMetaFiles(): Promise<string[]> {
const metaFiles = path.resolve(this.gradleUserHome, META_FILE_DIR, '*.cache')
const globber = await glob.create(metaFiles)
const bundleFiles = await globber.glob()
return bundleFiles
}
async beforeSave(listener: CacheListener): Promise<void> { async beforeSave(listener: CacheListener): Promise<void> {
await this.reportGradleUserHomeSize('before saving common artifacts') await this.reportGradleUserHomeSize('before saving common artifacts')
this.removeExcludedPaths() this.removeExcludedPaths()
@ -113,11 +111,13 @@ export class GradleUserHomeCache extends AbstractCache {
} }
private async saveArtifactBundles(listener: CacheListener): Promise<void> { private async saveArtifactBundles(listener: CacheListener): Promise<void> {
const processes: Promise<void>[] = [] const bundleMetadata = this.loadBundleMetadata()
const processes: Promise<CacheResult>[] = []
for (const [bundle, pattern] of this.getArtifactBundles()) { for (const [bundle, pattern] of this.getArtifactBundles()) {
const entryListener = listener.entry(bundle) const entryListener = listener.entry(bundle)
const previouslyRestoredKey = bundleMetadata.get(bundle)
const p = this.saveArtifactBundle(bundle, pattern, entryListener) const p = this.saveArtifactBundle(bundle, pattern, previouslyRestoredKey, entryListener)
// Run sequentially when debugging enabled // Run sequentially when debugging enabled
if (this.cacheDebuggingEnabled) { if (this.cacheDebuggingEnabled) {
await p await p
@ -125,16 +125,17 @@ export class GradleUserHomeCache extends AbstractCache {
processes.push(p) processes.push(p)
} }
await Promise.all(processes) const results = await Promise.all(processes)
this.saveMetadataForCacheResults(results)
} }
private async saveArtifactBundle( private async saveArtifactBundle(
bundle: string, bundle: string,
artifactPath: string, artifactPath: string,
previouslyRestoredKey: string | undefined,
listener: CacheEntryListener listener: CacheEntryListener
): Promise<void> { ): Promise<CacheResult> {
const bundleMetaFile = this.getBundleMetaFile(bundle)
const globber = await glob.create(artifactPath, { const globber = await glob.create(artifactPath, {
implicitDescendants: false, implicitDescendants: false,
followSymbolicLinks: false followSymbolicLinks: false
@ -144,15 +145,9 @@ export class GradleUserHomeCache extends AbstractCache {
// Handle no matching files // Handle no matching files
if (bundleFiles.length === 0) { if (bundleFiles.length === 0) {
this.debug(`No files found to cache for ${bundle}`) this.debug(`No files found to cache for ${bundle}`)
if (fs.existsSync(bundleMetaFile)) { return new CacheResult(bundle, undefined)
tryDelete(bundleMetaFile)
}
return
} }
const previouslyRestoredKey = fs.existsSync(bundleMetaFile)
? fs.readFileSync(bundleMetaFile, 'utf-8').trim()
: ''
const cacheKey = this.createCacheKey(bundle, bundleFiles) const cacheKey = this.createCacheKey(bundle, bundleFiles)
if (previouslyRestoredKey === cacheKey) { if (previouslyRestoredKey === cacheKey) {
@ -161,7 +156,6 @@ export class GradleUserHomeCache extends AbstractCache {
core.info(`Caching ${bundle} with cache key: ${cacheKey}`) core.info(`Caching ${bundle} with cache key: ${cacheKey}`)
const savedEntry = await this.saveCache([artifactPath], cacheKey) const savedEntry = await this.saveCache([artifactPath], cacheKey)
if (savedEntry !== undefined) { if (savedEntry !== undefined) {
this.writeBundleMetaFile(bundleMetaFile, cacheKey)
listener.markSaved(savedEntry.key, savedEntry.size) listener.markSaved(savedEntry.key, savedEntry.size)
} }
} }
@ -169,6 +163,8 @@ export class GradleUserHomeCache extends AbstractCache {
for (const file of bundleFiles) { for (const file of bundleFiles) {
tryDelete(file) tryDelete(file)
} }
return new CacheResult(bundle, cacheKey)
} }
protected createCacheKey(bundle: string, files: string[]): string { protected createCacheKey(bundle: string, files: string[]): string {
@ -181,15 +177,33 @@ export class GradleUserHomeCache extends AbstractCache {
return `${cacheKeyPrefix}${bundle}-${key}` return `${cacheKeyPrefix}${bundle}-${key}`
} }
private writeBundleMetaFile(metaFile: string, cacheKey: string): void { private loadBundleMetadata(): Map<string, string> {
this.debug(`Writing bundle metafile: ${metaFile}`) const bundleMetaFile = path.resolve(this.gradleUserHome, META_FILE_DIR, 'bundles.json')
if (!fs.existsSync(bundleMetaFile)) {
const dirName = path.dirname(metaFile) return new Map<string, string>()
if (!fs.existsSync(dirName)) {
fs.mkdirSync(dirName)
} }
const filedata = fs.readFileSync(bundleMetaFile, 'utf-8')
core.debug(`Loaded bundle metadata: ${filedata}`)
return new Map(JSON.parse(filedata))
}
fs.writeFileSync(metaFile, cacheKey) private saveMetadataForCacheResults(results: CacheResult[]): void {
const metadata = new Map<string, string>()
for (const result of results) {
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 bundleMetaFile = path.resolve(bundleMetaDir, 'bundles.json')
if (!fs.existsSync(bundleMetaDir)) {
fs.mkdirSync(bundleMetaDir, {recursive: true})
}
fs.writeFileSync(bundleMetaFile, filedata, 'utf-8')
} }
protected determineGradleUserHome(rootDir: string): string { protected determineGradleUserHome(rootDir: string): string {