2021-10-30 07:15:20 -06:00
|
|
|
import * as core from '@actions/core'
|
|
|
|
import * as cache from '@actions/cache'
|
|
|
|
import * as github from '@actions/github'
|
2021-12-07 12:29:37 -07:00
|
|
|
import {CacheListener} from './cache-reporting'
|
2021-11-05 06:54:31 -06:00
|
|
|
import {isCacheDebuggingEnabled, getCacheKeyPrefix, hashStrings, handleCacheFailure} from './cache-utils'
|
2021-10-30 07:15:20 -06:00
|
|
|
|
2021-11-28 10:19:56 -07:00
|
|
|
const CACHE_PROTOCOL_VERSION = 'v5-'
|
2021-10-30 07:15:20 -06:00
|
|
|
const JOB_CONTEXT_PARAMETER = 'workflow-job-context'
|
|
|
|
|
2021-11-28 10:19:56 -07:00
|
|
|
/**
|
|
|
|
* Represents a key used to restore a cache entry.
|
|
|
|
* The Github Actions cache will first try for an exact match on the key.
|
|
|
|
* If that fails, it will try for a prefix match on any of the restoreKeys.
|
|
|
|
*/
|
|
|
|
class CacheKey {
|
|
|
|
key: string
|
|
|
|
restoreKeys: string[]
|
|
|
|
|
|
|
|
constructor(key: string, restoreKeys: string[]) {
|
|
|
|
this.key = key
|
|
|
|
this.restoreKeys = restoreKeys
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Generates a cache key specific to the current job execution.
|
|
|
|
* The key is constructed from the following inputs:
|
|
|
|
* - A user-defined prefix (optional)
|
|
|
|
* - The cache protocol version
|
|
|
|
* - The name of the cache
|
|
|
|
* - The runner operating system
|
|
|
|
* - The name of the Job being executed
|
|
|
|
* - The matrix values for the Job being executed (job context)
|
|
|
|
* - The SHA of the commit being executed
|
|
|
|
*
|
|
|
|
* Caches are restored by trying to match the these key prefixes in order:
|
|
|
|
* - The full key with SHA
|
|
|
|
* - A previous key for this Job + matrix
|
|
|
|
* - Any previous key for this Job (any matrix)
|
|
|
|
* - Any previous key for this cache on the current OS
|
|
|
|
*/
|
2021-10-30 07:15:20 -06:00
|
|
|
function generateCacheKey(cacheName: string): CacheKey {
|
2021-11-28 10:19:56 -07:00
|
|
|
const cacheKeyBase = `${getCacheKeyPrefix()}${CACHE_PROTOCOL_VERSION}${cacheName}`
|
2021-10-30 07:15:20 -06:00
|
|
|
|
|
|
|
// At the most general level, share caches for all executions on the same OS
|
|
|
|
const runnerOs = process.env['RUNNER_OS'] || ''
|
2021-11-28 10:19:56 -07:00
|
|
|
const cacheKeyForOs = `${cacheKeyBase}|${runnerOs}`
|
2021-10-30 07:15:20 -06:00
|
|
|
|
|
|
|
// Prefer caches that run this job
|
|
|
|
const cacheKeyForJob = `${cacheKeyForOs}|${github.context.job}`
|
|
|
|
|
|
|
|
// Prefer (even more) jobs that run this job with the same context (matrix)
|
|
|
|
const cacheKeyForJobContext = `${cacheKeyForJob}[${determineJobContext()}]`
|
|
|
|
|
|
|
|
// Exact match on Git SHA
|
|
|
|
const cacheKey = `${cacheKeyForJobContext}-${github.context.sha}`
|
|
|
|
|
|
|
|
return new CacheKey(cacheKey, [cacheKeyForJobContext, cacheKeyForJob, cacheKeyForOs])
|
|
|
|
}
|
|
|
|
|
|
|
|
function determineJobContext(): string {
|
|
|
|
// By default, we hash the full `matrix` data for the run, to uniquely identify this job invocation
|
2021-11-28 10:19:56 -07:00
|
|
|
// The only way we can obtain the `matrix` data is via the `workflow-job-context` parameter in action.yml.
|
2021-10-30 07:15:20 -06:00
|
|
|
const workflowJobContext = core.getInput(JOB_CONTEXT_PARAMETER)
|
|
|
|
return hashStrings([workflowJobContext])
|
|
|
|
}
|
|
|
|
|
|
|
|
export abstract class AbstractCache {
|
|
|
|
private cacheName: string
|
|
|
|
private cacheDescription: string
|
|
|
|
private cacheKeyStateKey: string
|
|
|
|
private cacheResultStateKey: string
|
|
|
|
|
|
|
|
protected readonly cacheDebuggingEnabled: boolean
|
|
|
|
|
|
|
|
constructor(cacheName: string, cacheDescription: string) {
|
|
|
|
this.cacheName = cacheName
|
|
|
|
this.cacheDescription = cacheDescription
|
|
|
|
this.cacheKeyStateKey = `CACHE_KEY_${cacheName}`
|
|
|
|
this.cacheResultStateKey = `CACHE_RESULT_${cacheName}`
|
|
|
|
this.cacheDebuggingEnabled = isCacheDebuggingEnabled()
|
|
|
|
}
|
|
|
|
|
2021-11-28 10:19:56 -07:00
|
|
|
/**
|
|
|
|
* Restores the cache entry, finding the closest match to the currently running job.
|
|
|
|
* If the target output already exists, caching will be skipped.
|
|
|
|
*/
|
2021-10-30 07:21:27 -06:00
|
|
|
async restore(listener: CacheListener): Promise<void> {
|
2021-10-30 07:15:20 -06:00
|
|
|
if (this.cacheOutputExists()) {
|
|
|
|
core.info(`${this.cacheDescription} already exists. Not restoring from cache.`)
|
|
|
|
return
|
|
|
|
}
|
2021-11-28 10:19:56 -07:00
|
|
|
const entryListener = listener.entry(this.cacheDescription)
|
2021-10-30 07:15:20 -06:00
|
|
|
|
|
|
|
const cacheKey = this.prepareCacheKey()
|
|
|
|
|
|
|
|
this.debug(
|
|
|
|
`Requesting ${this.cacheDescription} with
|
|
|
|
key:${cacheKey.key}
|
|
|
|
restoreKeys:[${cacheKey.restoreKeys}]`
|
|
|
|
)
|
|
|
|
|
|
|
|
const cacheResult = await this.restoreCache(this.getCachePath(), cacheKey.key, cacheKey.restoreKeys)
|
2021-11-28 10:19:56 -07:00
|
|
|
entryListener.markRequested(cacheKey.key, cacheKey.restoreKeys)
|
2021-10-30 07:15:20 -06:00
|
|
|
|
|
|
|
if (!cacheResult) {
|
2021-11-27 16:07:07 -07:00
|
|
|
core.info(`${this.cacheDescription} cache not found. Will initialize empty.`)
|
2021-10-30 07:15:20 -06:00
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2021-10-31 20:35:28 -06:00
|
|
|
core.saveState(this.cacheResultStateKey, cacheResult.key)
|
2021-11-28 10:19:56 -07:00
|
|
|
entryListener.markRestored(cacheResult.key, cacheResult.size)
|
|
|
|
|
2021-10-31 20:35:28 -06:00
|
|
|
core.info(`Restored ${this.cacheDescription} from cache key: ${cacheResult.key}`)
|
2021-10-30 07:15:20 -06:00
|
|
|
|
|
|
|
try {
|
2021-10-30 07:21:27 -06:00
|
|
|
await this.afterRestore(listener)
|
2021-10-30 07:15:20 -06:00
|
|
|
} catch (error) {
|
|
|
|
core.warning(`Restore ${this.cacheDescription} failed in 'afterRestore': ${error}`)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
prepareCacheKey(): CacheKey {
|
|
|
|
const cacheKey = generateCacheKey(this.cacheName)
|
|
|
|
core.saveState(this.cacheKeyStateKey, cacheKey.key)
|
|
|
|
return cacheKey
|
|
|
|
}
|
|
|
|
|
|
|
|
protected async restoreCache(
|
|
|
|
cachePath: string[],
|
|
|
|
cacheKey: string,
|
|
|
|
cacheRestoreKeys: string[] = []
|
2021-10-30 12:17:41 -06:00
|
|
|
): Promise<cache.CacheEntry | undefined> {
|
2021-10-30 07:15:20 -06:00
|
|
|
try {
|
|
|
|
return await cache.restoreCache(cachePath, cacheKey, cacheRestoreKeys)
|
|
|
|
} catch (error) {
|
2021-11-05 06:54:31 -06:00
|
|
|
handleCacheFailure(error, `Failed to restore ${cacheKey}`)
|
2021-10-30 07:15:20 -06:00
|
|
|
return undefined
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-10-30 07:21:27 -06:00
|
|
|
protected async afterRestore(_listener: CacheListener): Promise<void> {}
|
2021-10-30 07:15:20 -06:00
|
|
|
|
2021-11-28 10:19:56 -07:00
|
|
|
/**
|
|
|
|
* Saves the cache entry based on the current cache key, unless:
|
|
|
|
* - If the cache output existed before restore, then it is not saved.
|
|
|
|
* - If the cache was restored with the exact key, we cannot overwrite it.
|
|
|
|
*
|
|
|
|
* If the cache entry was restored with a partial match on a restore key, then
|
|
|
|
* it is saved with the exact key.
|
|
|
|
*/
|
2021-10-30 07:21:27 -06:00
|
|
|
async save(listener: CacheListener): Promise<void> {
|
2021-10-30 07:15:20 -06:00
|
|
|
if (!this.cacheOutputExists()) {
|
2021-11-05 08:35:45 -06:00
|
|
|
core.info(`No ${this.cacheDescription} to cache.`)
|
2021-10-30 07:15:20 -06:00
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2021-11-28 10:19:56 -07:00
|
|
|
// Retrieve the state set in the previous 'restore' step.
|
|
|
|
const cacheKeyFromRestore = core.getState(this.cacheKeyStateKey)
|
|
|
|
const cacheResultFromRestore = core.getState(this.cacheResultStateKey)
|
2021-10-30 07:15:20 -06:00
|
|
|
|
2021-11-28 10:19:56 -07:00
|
|
|
if (!cacheKeyFromRestore) {
|
2021-11-05 08:35:45 -06:00
|
|
|
core.info(`${this.cacheDescription} existed prior to cache restore. Not saving.`)
|
2021-10-30 07:15:20 -06:00
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2021-11-28 10:19:56 -07:00
|
|
|
if (cacheResultFromRestore && cacheKeyFromRestore === cacheResultFromRestore) {
|
|
|
|
core.info(`Cache hit occurred on the cache key ${cacheKeyFromRestore}, not saving cache.`)
|
2021-10-30 07:15:20 -06:00
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
try {
|
2021-10-30 07:21:27 -06:00
|
|
|
await this.beforeSave(listener)
|
2021-10-30 07:15:20 -06:00
|
|
|
} catch (error) {
|
|
|
|
core.warning(`Save ${this.cacheDescription} failed in 'beforeSave': ${error}`)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2021-11-28 10:19:56 -07:00
|
|
|
core.info(`Caching ${this.cacheDescription} with cache key: ${cacheKeyFromRestore}`)
|
2021-10-30 07:15:20 -06:00
|
|
|
const cachePath = this.getCachePath()
|
2021-11-28 10:19:56 -07:00
|
|
|
const savedEntry = await this.saveCache(cachePath, cacheKeyFromRestore)
|
2021-10-30 07:15:20 -06:00
|
|
|
|
2021-10-30 12:17:41 -06:00
|
|
|
if (savedEntry) {
|
|
|
|
listener.entry(this.cacheDescription).markSaved(savedEntry.key, savedEntry.size)
|
|
|
|
}
|
2021-10-30 07:15:20 -06:00
|
|
|
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2021-10-30 07:21:27 -06:00
|
|
|
protected async beforeSave(_listener: CacheListener): Promise<void> {}
|
2021-10-30 07:15:20 -06:00
|
|
|
|
2021-10-30 12:17:41 -06:00
|
|
|
protected async saveCache(cachePath: string[], cacheKey: string): Promise<cache.CacheEntry | undefined> {
|
2021-10-30 07:15:20 -06:00
|
|
|
try {
|
2021-10-30 12:17:41 -06:00
|
|
|
return await cache.saveCache(cachePath, cacheKey)
|
2021-10-30 07:15:20 -06:00
|
|
|
} catch (error) {
|
2021-11-05 06:54:31 -06:00
|
|
|
handleCacheFailure(error, `Failed to save cache entry ${cacheKey}`)
|
2021-10-30 07:15:20 -06:00
|
|
|
}
|
2021-10-30 12:17:41 -06:00
|
|
|
return undefined
|
2021-10-30 07:15:20 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
protected debug(message: string): void {
|
|
|
|
if (this.cacheDebuggingEnabled) {
|
|
|
|
core.info(message)
|
|
|
|
} else {
|
|
|
|
core.debug(message)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
protected abstract cacheOutputExists(): boolean
|
|
|
|
protected abstract getCachePath(): string[]
|
|
|
|
}
|