These actions allow caching dependencies and build outputs to eliminate duplicate work and improve workflow execution time. The main advantage of this fork over the base is sharing data across multiple jobs. You do not need to use artifacts and can skip various steps, saving as much runtime as possible.
The following actions are available:
martijnhols/actions-cache@v3
(mostly matches the base repository)martijnhols/actions-cache/restore@v3
martijnhols/actions-cache/save@v3
martijnhols/actions-cache/check@v3
While this is a fork, there are currently no plans to merge this into GitHub's actions/cache since GitHub does not appear to be reviewing PRs and so making this mergeable would be a waste of time. This repository will be available on its own.
This is the base action largely matching GitHub's actions/cache. Under the hood this calls the restore
action where you place the action, and the save
action just before the job finishes.
This can be used for caching a step such as installing dependencies which are not re-used in other jobs. If you want to reuse your data in other jobs, use one of the other actions.
path
- Required - A list of files, directories, and wildcard patterns to cache and restore. See@actions/glob
for supported patterns.key
- Required - An explicit key for restoring and saving the cacherestore-keys
- An ordered list of prefix-matched keys to use for restoring stale cache if no cache hit occurred for key.upload-chunk-size
- The chunk size used to split up large files during upload, in bytes
cache-hit
- A boolean value to indicate an exact match was found for the keyprimary-key
- The primary key for restoring or saving exactly matching cache.
See Skipping steps based on cache-hit for info on using this output
This action will read data from the cache and place it in at the provided path.
path
- Required - A list of files, directories, and wildcard patterns to cache and restore. See@actions/glob
for supported patterns.key
- Required - An explicit key for restoring and saving the cacherestore-keys
- An ordered list of prefix-matched keys to use for restoring stale cache if no cache hit occurred for key.required
- When set totrue
, the action will fail if an exact match could not be found.
cache-hit
- A boolean value to indicate an exact match was found for the keyprimary-key
- The primary key for restoring or saving exactly matching cache.
This action will save data at the provided path to the cache.
path
- Required - A list of files, directories, and wildcard patterns to cache and restore. See@actions/glob
for supported patterns.key
- Required - An explicit key for restoring and saving the cacheupload-chunk-size
- The chunk size used to split up large files during upload, in bytes
Tip: when combined with the restore
or check
action, add the id: cache
property to the restore
/check
action and use key: ${{ steps.cache.outputs.primary-key }}
in the save
action. This ensures your cache key is not recomputed, which may otherwise lead to issues.
This action will check if an exact match is available in the cache without downloading it.
path
- Required - A list of files, directories, and wildcard patterns to cache and restore. See@actions/glob
for supported patterns.key
- Required - An explicit key for restoring and saving the cache
cache-hit
- A boolean value to indicate an exact match was found for the keyprimary-key
- The primary key for restoring or saving exactly matching cache.
These recipes serve as examples. For simplicity sake some irrelevant steps (such as setup-node) are omitted.
🔗 Just caching (like the base actions/cache)
This caches node_modules
folder. Using the Skipping steps based on cache-hit solution, this only installs dependencies if the cache did not return an exact match.
If no exact match could be found, it uses a restore-key to restore an older cache since the tool we use (yarn) can reuse existing files to save time.
name: Build app
on: push
jobs:
build:
runs-on: ubuntu-latest
steps:
- uses: actions/checkout@v2
- name: Cache node_modules
id: cache
uses: martijnhols/actions-cache@v3
with:
# Cache the node_modules folder and its contents
path: node_modules
# Genarate a unique key based on the runner OS, an id, and a hash that changes whenever the `yarn.lock` file or any file in the `patches` folder changes.
key: ${{ runner.os }}-node_modules-${{ hashFiles('yarn.lock', 'patches') }}
# If no exact match is found, look for the most recent cache entry with this key:
restore-keys: ${{ runner.os }}-node_modules
- name: Install dependencies
# Only install dependencies only when no exact match was found in the cache
if: steps.cache.outputs.cache-hit != 'true'
run: yarn install
- name: Build app
run: yarn build
🔗 Just caching with manual control
This behaves the same as the Just caching recipe, but uses the restore
and save
actions manually. This has no significant benefits over using the standard action, though I prefer it for its minor readability and maintainability improvements.
name: Build app
on: push
jobs:
build:
runs-on: ubuntu-latest
steps:
- uses: actions/checkout@v2
- name: Restore "node_modules" from cache
id: cache
uses: martijnhols/actions-cache/restore@v3
with:
path: node_modules
key: ${{ runner.os }}-node_modules-${{ hashFiles('yarn.lock', 'patches') }}
restore-keys: ${{ runner.os }}-node_modules
- name: Install dependencies
if: steps.cache.outputs.cache-hit != 'true'
run: yarn install
- name: Build app
run: yarn build
- name: Save "node_modules" to cache
# No need to save identical data when an exact match was found
if: steps.cache.outputs.cache-hit != 'true'
uses: martijnhols/actions-cache/save@v3
with:
path: node_modules
# Re-use the primary-key from the restore action to ensure it is not recomputed. This could otherwise cause issues if our "build" step modifies files within one of the `hashFiles` directories.
key: ${{ steps.cache.outputs.primary-key }}
🔗 Cache build output and skipping build
This extends the Share cache across jobs recipe.
When you want to publish your build, you probably want to do this in a separate step. Using the Share cache across jobs recipe you can also reuse your build (we add this in step 1).
As a bonus, you can skip building the app entirely if an exact match was found (we add this in step 2). This is especially useful in monorepos, where only a few apps need to be build each run.
NOTE: Take extra care when choosing a cache key. Builds often involve many different configuration files, if you forget to add a file it may not trigger a rebuild when it is changed.
Step 1/2: First, let's add a publish job
name: Build app
on: push
jobs:
install:
runs-on: ubuntu-latest
steps:
- uses: actions/checkout@v2
- name: Restore "node_modules" from cache
id: cache
uses: martijnhols/actions-cache/restore@v3
with:
path: node_modules
key: ${{ runner.os }}-node_modules-${{ hashFiles('yarn.lock', 'patches') }}
restore-keys: ${{ runner.os }}-node_modules
- name: Install dependencies
if: steps.cache.outputs.cache-hit != 'true'
run: yarn install
- name: Save "node_modules" to cache
if: steps.cache.outputs.cache-hit != 'true'
uses: martijnhols/actions-cache/save@v3
with:
path: node_modules
key: ${{ steps.cache.outputs.primary-key }}
build:
needs: [install]
runs-on: ubuntu-latest
steps:
- uses: actions/checkout@v2
- name: Restore "node_modules" from cache
uses: martijnhols/actions-cache/restore@v3
with:
path: node_modules
key: ${{ runner.os }}-node_modules-${{ hashFiles('yarn.lock', 'patches') }}
required: true
- name: Build app
run: yarn build
# Notice that we do not use a "restore" in this job: the build in our imaginary project can't reuse its own build files so restoring that before building would be a waste of time.
- name: Save "build" to cache
uses: martijnhols/actions-cache/save@v3
with:
path: build
key: ${{ runner.os }}-node_modules-${{ hashFiles('yarn.lock', 'patches', 'src', '.babelrc') }}
publish:
needs: [install]
runs-on: ubuntu-latest
steps:
- uses: actions/checkout@v2
- name: Restore "build" from cache
uses: martijnhols/actions-cache/restore@v3
with:
path: build
key: ${{ runner.os }}-node_modules-${{ hashFiles('yarn.lock', 'patches', 'src', '.babelrc') }}
required: true
- name: Publish app
run: yarn publish
Step 2/2: Now we use check
to skip steps if the app was already built
(This only changes made in this yml are in the build
job)
name: Build app
on: push
jobs:
install:
runs-on: ubuntu-latest
steps:
- uses: actions/checkout@v2
- name: Restore "node_modules" from cache
id: cache
uses: martijnhols/actions-cache/restore@v3
with:
path: node_modules
key: ${{ runner.os }}-node_modules-${{ hashFiles('yarn.lock', 'patches') }}
restore-keys: ${{ runner.os }}-node_modules
- name: Install dependencies
if: steps.cache.outputs.cache-hit != 'true'
run: yarn install
- name: Save "node_modules" to cache
if: steps.cache.outputs.cache-hit != 'true'
uses: martijnhols/actions-cache/save@v3
with:
path: node_modules
key: ${{ steps.cache.outputs.primary-key }}
build:
needs: [install]
runs-on: ubuntu-latest
steps:
- uses: actions/checkout@v2
# Using martijnhols/actions-cache/check we check if a cache entry exists without downloading it
- name: Check if "build" is already cached
uses: martijnhols/actions-cache/check@v3
id: cache
with:
path: build
key: ${{ runner.os }}-node_modules-${{ hashFiles('yarn.lock', 'patches', 'src', '.babelrc') }}
- name: Restore "node_modules" from cache
# Only execute if the build isn't already in cache
if: steps.cache.outputs.cache-hit != 'true'
uses: martijnhols/actions-cache/restore@v3
with:
path: node_modules
key: ${{ runner.os }}-node_modules-${{ hashFiles('yarn.lock', 'patches') }}
required: true
- name: Build app
# Only execute if the build isn't already in cache
if: steps.cache.outputs.cache-hit != 'true'
run: yarn build
# Notice that we do not use a "restore" in this job: the build in our imaginary project can't reuse its own build files so restoring that before building would be a waste of time.
- name: Save "build" to cache
# Only execute if the build isn't already in cache
if: steps.cache.outputs.cache-hit != 'true'
uses: martijnhols/actions-cache/save@v3
with:
path: build
key: ${{ runner.os }}-node_modules-${{ hashFiles('yarn.lock', 'patches', 'src', '.babelrc') }}
publish:
needs: [install]
runs-on: ubuntu-latest
steps:
- uses: actions/checkout@v2
- name: Restore "build" from cache
uses: martijnhols/actions-cache/restore@v3
with:
path: build
key: ${{ runner.os }}-node_modules-${{ hashFiles('yarn.lock', 'patches', 'src', '.babelrc') }}
required: true
- name: Publish app
run: yarn publish
🔗 Save cache regardless of job success/failure
This extends the Just caching recipe.
When you have multiple build steps in a single job, you may want to save your data regardless of the job failing. In this case splitting up the cache actions like in the Just caching with manual control recipe and moving the save action above your flaky build step achieves this.
If you want your cache to be saved regardless of a failure during the install step, you can change the if: steps.cache.outputs.cache-hit != 'true'
line into if: always() && steps.cache.outputs.cache-hit != 'true'
.
name: Build app
on: push
jobs:
install:
runs-on: ubuntu-latest
steps:
- uses: actions/checkout@v2
- name: Restore "node_modules" from cache
id: cache
uses: martijnhols/actions-cache/restore@v3
with:
path: node_modules
key: ${{ runner.os }}-node_modules-${{ hashFiles('yarn.lock', 'patches') }}
restore-keys: ${{ runner.os }}-node_modules
- name: Install dependencies
if: steps.cache.outputs.cache-hit != 'true'
run: yarn install
- name: Save "node_modules" to cache
if: steps.cache.outputs.cache-hit != 'true'
uses: martijnhols/actions-cache/save@v3
with:
path: node_modules
key: ${{ steps.cache.outputs.primary-key }}
- name: Run flaky tests
run: yarn test
A cache key can include any of the contexts, functions, literals, and operators supported by GitHub Actions.
For example, using the hashFiles
function allows you to create a new cache when dependencies change.
- uses: martijnhols/actions-cache@v3
with:
path: |
path/to/dependencies
some/other/dependencies
key: ${{ runner.os }}-${{ hashFiles('**/lockfiles') }}
Additionally, you can use arbitrary command output in a cache key, such as a date or software version:
# http://man7.org/linux/man-pages/man1/date.1.html
- name: Get Date
id: get-date
run: |
echo "::set-output name=date::$(/bin/date -u "+%Y%m%d")"
shell: bash
- uses: martijnhols/actions-cache@v3
with:
path: path/to/dependencies
key: ${{ runner.os }}-${{ steps.get-date.outputs.date }}-${{ hashFiles('**/lockfiles') }}
See Using contexts to create cache keys
If you are only using the save action, sometimes you need to compute the cache-key before generating the artifact you wish to cache. This ensures the generated artifact does not affect the cache-key. This can be achieved with basic actions syntax:
- name: Generate react-native cache key
id: cache-key
run: echo "::set-output name=value::my-cache-key-${{ hashFiles('react-native') }}"
A repository can have up to 10GB of caches. Once the 10GB limit is reached, older caches will be evicted based on when the cache was last accessed. Caches that are not accessed within the last week will also be evicted.
Using the cache-hit
output, subsequent steps (such as install or build) can be skipped when a cache hit occurs on the key.
Example:
steps:
- uses: actions/checkout@v2
- uses: martijnhols/actions-cache@v3
id: cache
with:
path: path/to/dependencies
key: ${{ runner.os }}-${{ hashFiles('**/lockfiles') }}
- name: Install Dependencies
if: steps.cache.outputs.cache-hit != 'true'
run: /install.sh
Note: The
id
defined in the cache step must match theid
in theif
statement (i.e.steps.[ID].outputs.cache-hit
)
The scripts and documentation in this project are released under the MIT License