Compare commits

..

5 Commits
v1.5.1 ... v1

Author SHA1 Message Date
Ian Butterworth
dc1a3cdeac make depot if not restored (#91) 2024-01-16 10:48:25 -05:00
Curtis Vogt
4491ed7a86 Pass cache-name between save/restore jobs (#103)
* Pass cache-name between save/restore

* Use save job name in cache-name

* Re-order test jobs

* Fix typo in registry warning
2024-01-16 09:40:31 -06:00
Curtis Vogt
b84ca24db8 Avoid corrupting existing cloned Julia registries (#102)
* Reproduce add-julia-registry issue

* Skip registries restore when already present

* Expand ~

* Refactor paths step to use bash array
2024-01-15 19:28:33 -06:00
Curtis Vogt
b3b34e3264 Test cache action against Julia 1.0 and nightly (#101)
* CI test action on Julia 1.0

* Avoid quoting Julia shell string

* Test against Julia nightly
2024-01-15 15:20:46 -06:00
Curtis Vogt
0c5d92d73a Delete cache entries only on the workflow branch (#97)
* Delete cache entries on the workflow branch

* Grant permissions for cache cleanup

* Add delete-old-caches required for testing purposes

* Revise help message

* Faster generate-key

* Use distinct cache-names for matrix/no-matrix jobs

* Remove redundant permissions

* Better fork detection logic
2024-01-15 10:15:36 -06:00
3 changed files with 263 additions and 127 deletions

View File

@@ -20,117 +20,177 @@ permissions:
contents: read contents: read
jobs: jobs:
generate-key: generate-prefix:
runs-on: ubuntu-latest runs-on: ubuntu-latest
outputs: outputs:
cache-name: ${{ steps.name.outputs.cache-name }} cache-prefix: ${{ steps.name.outputs.cache-prefix }}
steps: steps:
- name: Generate random file - name: Generate random cache-prefix
shell: 'julia --color=yes {0}'
run: 'write("random.txt", string(rand(10)))'
- name: Set cache-name as output
id: name id: name
run: echo "cache-name=${{ hashFiles('random.txt') }}" >> $GITHUB_OUTPUT run: |
cache_prefix=$(head -n 100 </dev/urandom | shasum -a 256 | cut -d ' ' -f 1)
echo "cache-prefix=$cache_prefix" >>"$GITHUB_OUTPUT"
test-save: test-save:
needs: generate-key needs: generate-prefix
runs-on: ${{ matrix.os }}
outputs:
cache-name: ${{ steps.cache-name.outputs.cache-name }}
strategy: strategy:
matrix: matrix:
dep: nested:
- name: pandoc_jll - name: matrix
version: "3"
invalid-chars: "," # Use invalid characters in job matrix to ensure we escape them invalid-chars: "," # Use invalid characters in job matrix to ensure we escape them
version:
- "1.0"
- "1"
- "nightly"
os: os:
- ubuntu-latest - ubuntu-latest
- windows-latest - windows-latest
- macOS-latest - macOS-latest
exclude:
# Test Julia "1.0" on Linux only
- version: "1.0"
os: windows-latest
- version: "1.0"
os: macOS-latest
fail-fast: false fail-fast: false
runs-on: ${{ matrix.os }}
env: env:
JULIA_DEPOT_PATH: /tmp/julia-depot JULIA_DEPOT_PATH: /tmp/julia-depot
steps: steps:
- uses: actions/checkout@b4ffde65f46336ab88eb53be808477a3936bae11 - uses: actions/checkout@b4ffde65f46336ab88eb53be808477a3936bae11
- name: Set cache-name
id: cache-name
shell: bash
run: |
echo "cache-name=${{ needs.generate-prefix.outputs.cache-prefix }}-${{ github.job }}" >>"$GITHUB_OUTPUT"
- uses: julia-actions/setup-julia@v1
with:
version: ${{ matrix.version }}
- name: Save cache - name: Save cache
id: cache id: cache
uses: ./ uses: ./
with: with:
cache-name: ${{ needs.generate-key.outputs.cache-name }} cache-name: ${{ steps.cache-name.outputs.cache-name }}
delete-old-caches: required
- name: Check no artifacts dir - name: Check no artifacts dir
shell: 'julia --color=yes {0}' shell: julia --color=yes {0}
run: | run: |
dir = joinpath(first(DEPOT_PATH), "artifacts") dir = joinpath(first(DEPOT_PATH), "artifacts")
@assert !isdir(dir) @assert !isdir(dir)
- name: Install a small binary - name: Install a small binary
shell: 'julia --color=yes {0}' shell: julia --color=yes {0}
run: 'using Pkg; Pkg.add(PackageSpec(name="${{ matrix.dep.name }}", version="${{ matrix.dep.version }}"))' run: |
using Pkg
if VERSION >= v"1.3"
Pkg.add(PackageSpec(name="pandoc_jll", version="3"))
else
Pkg.add(PackageSpec(name="Scratch", version="1"))
using Scratch
get_scratch!("test")
end
test-restore:
needs: test-save
runs-on: ${{ matrix.os }}
strategy:
matrix:
nested:
- name: matrix
invalid-chars: "," # Use invalid characters in job matrix to ensure we escape them
version:
- "1.0"
- "1"
- "nightly"
os:
- ubuntu-latest
- windows-latest
- macOS-latest
exclude:
# Test Julia "1.0" on Linux only
- version: "1.0"
os: windows-latest
- version: "1.0"
os: macOS-latest
fail-fast: false
env:
JULIA_DEPOT_PATH: /tmp/julia-depot
steps:
- uses: actions/checkout@b4ffde65f46336ab88eb53be808477a3936bae11
- uses: julia-actions/setup-julia@v1
with:
version: ${{ matrix.version }}
- name: Restore cache
id: cache
uses: ./
with:
cache-name: ${{ needs.test-save.outputs.cache-name }}
# Cannot require a successful cache delete on forked PRs as the permissions for actions is limited to read
delete-old-caches: ${{ github.event.pull_request.head.repo.fork && 'false' || 'required' }}
- name: Test cache-hit output
shell: julia --color=yes {0}
run: |
@show ENV["cache-hit"]
@assert ENV["cache-hit"] == "true"
env:
cache-hit: ${{ steps.cache.outputs.cache-hit }}
- name: Check existance or emptiness of affected dirs
shell: julia --color=yes {0}
run: |
# These dirs should exist as they've been cached
artifacts_dir = joinpath(first(DEPOT_PATH), "artifacts")
if VERSION >= v"1.3"
@assert !isempty(readdir(artifacts_dir))
else
@assert !isdir(artifacts_dir)
end
packages_dir = joinpath(first(DEPOT_PATH), "packages")
@assert !isempty(readdir(packages_dir))
compiled_dir = joinpath(first(DEPOT_PATH), "compiled")
@assert !isempty(readdir(compiled_dir))
scratchspaces_dir = joinpath(first(DEPOT_PATH), "scratchspaces")
@assert !isempty(readdir(scratchspaces_dir))
logs_dir = joinpath(first(DEPOT_PATH), "logs")
@assert !isempty(readdir(logs_dir))
# Do tests with no matrix also given the matrix is auto-included in cache key # Do tests with no matrix also given the matrix is auto-included in cache key
test-save-nomatrix: test-save-nomatrix:
needs: generate-key needs: generate-prefix
runs-on: ubuntu-latest runs-on: ubuntu-latest
outputs:
cache-name: ${{ steps.cache-name.outputs.cache-name }}
steps: steps:
- uses: actions/checkout@b4ffde65f46336ab88eb53be808477a3936bae11 - uses: actions/checkout@b4ffde65f46336ab88eb53be808477a3936bae11
- name: Set cache-name
id: cache-name
run: |
echo "cache-name=${{ needs.generate-prefix.outputs.cache-prefix }}-${{ github.job }}" >>"$GITHUB_OUTPUT"
- name: Save cache - name: Save cache
id: cache id: cache
uses: ./ uses: ./
with: with:
cache-name: ${{ needs.generate-key.outputs.cache-name }} cache-name: ${{ steps.cache-name.outputs.cache-name }}
delete-old-caches: required
- name: Check no artifacts dir - name: Check no artifacts dir
shell: 'julia --color=yes {0}' shell: julia --color=yes {0}
run: | run: |
dir = joinpath(first(DEPOT_PATH), "artifacts") dir = joinpath(first(DEPOT_PATH), "artifacts")
@assert !isdir(dir) @assert !isdir(dir)
- name: Install a small binary - name: Install a small binary
shell: 'julia --color=yes {0}' shell: julia --color=yes {0}
run: 'using Pkg; Pkg.add("pandoc_jll")'
test-restore:
needs: [generate-key, test-save]
strategy:
matrix:
dep:
- name: pandoc_jll
version: "3"
invalid-chars: "," # Use invalid characters in job matrix to ensure we escape them
os:
- ubuntu-latest
- windows-latest
- macOS-latest
fail-fast: false
runs-on: ${{ matrix.os }}
env:
JULIA_DEPOT_PATH: /tmp/julia-depot
steps:
- uses: actions/checkout@b4ffde65f46336ab88eb53be808477a3936bae11
- name: Restore cache
id: cache
uses: ./
with:
cache-name: ${{ needs.generate-key.outputs.cache-name }}
- name: Test cache-hit output
shell: 'julia --color=yes {0}'
run: | run: |
@show ENV["cache-hit"] using Pkg
@assert ENV["cache-hit"] == "true" if VERSION >= v"1.3"
env: Pkg.add(PackageSpec(name="pandoc_jll", version="3"))
cache-hit: ${{ steps.cache.outputs.cache-hit }} else
- name: Check existance or emptiness of affected dirs Pkg.add(PackageSpec(name="Scratch", version="1"))
shell: 'julia --color=yes {0}' using Scratch
run: | get_scratch!("test")
# These dirs should exist as they've been cached end
artifacts_dir = joinpath(first(DEPOT_PATH), "artifacts")
@assert !isempty(readdir(artifacts_dir))
packages_dir = joinpath(first(DEPOT_PATH), "packages")
@assert !isempty(readdir(packages_dir))
compiled_dir = joinpath(first(DEPOT_PATH), "compiled")
@assert !isempty(readdir(compiled_dir))
scratchspaces_dir = joinpath(first(DEPOT_PATH), "scratchspaces")
@assert !isempty(readdir(scratchspaces_dir))
logs_dir = joinpath(first(DEPOT_PATH), "logs")
@assert !isempty(readdir(logs_dir))
test-restore-nomatrix: test-restore-nomatrix:
needs: [generate-key, test-save-nomatrix] needs: test-save-nomatrix
runs-on: ubuntu-latest runs-on: ubuntu-latest
steps: steps:
- uses: actions/checkout@b4ffde65f46336ab88eb53be808477a3936bae11 - uses: actions/checkout@b4ffde65f46336ab88eb53be808477a3936bae11
@@ -138,20 +198,26 @@ jobs:
id: cache id: cache
uses: ./ uses: ./
with: with:
cache-name: ${{ needs.generate-key.outputs.cache-name }} cache-name: ${{ needs.test-save-nomatrix.outputs.cache-name }}
# Cannot require a successful cache delete on forked PRs as the permissions for actions is limited to read
delete-old-caches: ${{ github.event.pull_request.head.repo.fork && 'false' || 'required' }}
- name: Test cache-hit output - name: Test cache-hit output
shell: 'julia --color=yes {0}' shell: julia --color=yes {0}
run: | run: |
@show ENV["cache-hit"] @show ENV["cache-hit"]
@assert ENV["cache-hit"] == "true" @assert ENV["cache-hit"] == "true"
env: env:
cache-hit: ${{ steps.cache.outputs.cache-hit }} cache-hit: ${{ steps.cache.outputs.cache-hit }}
- name: Check existance or emptiness of affected dirs - name: Check existance or emptiness of affected dirs
shell: 'julia --color=yes {0}' shell: julia --color=yes {0}
run: | run: |
# These dirs should exist as they've been cached # These dirs should exist as they've been cached
artifacts_dir = joinpath(first(DEPOT_PATH), "artifacts") artifacts_dir = joinpath(first(DEPOT_PATH), "artifacts")
@assert !isempty(readdir(artifacts_dir)) if VERSION >= v"1.3"
@assert !isempty(readdir(artifacts_dir))
else
@assert !isdir(artifacts_dir)
end
packages_dir = joinpath(first(DEPOT_PATH), "packages") packages_dir = joinpath(first(DEPOT_PATH), "packages")
@assert !isempty(readdir(packages_dir)) @assert !isempty(readdir(packages_dir))
compiled_dir = joinpath(first(DEPOT_PATH), "compiled") compiled_dir = joinpath(first(DEPOT_PATH), "compiled")
@@ -161,3 +227,54 @@ jobs:
logs_dir = joinpath(first(DEPOT_PATH), "logs") logs_dir = joinpath(first(DEPOT_PATH), "logs")
@assert !isempty(readdir(logs_dir)) @assert !isempty(readdir(logs_dir))
test-save-cloned-registry:
needs: generate-prefix
runs-on: ubuntu-latest
outputs:
cache-name: ${{ steps.cache-name.outputs.cache-name }}
steps:
- uses: actions/checkout@b4ffde65f46336ab88eb53be808477a3936bae11
- name: Set cache-name
id: cache-name
run: |
echo "cache-name=${{ needs.generate-prefix.outputs.cache-prefix }}-${{ github.job }}" >>"$GITHUB_OUTPUT"
- name: Save cache
uses: ./
with:
cache-name: ${{ steps.cache-name.outputs.cache-name }}
# Cannot require a successful cache delete on forked PRs as the permissions for actions is limited to read
delete-old-caches: ${{ github.event.pull_request.head.repo.fork && 'false' || 'required' }}
- name: Add General registry clone
shell: julia --color=yes {0}
run: |
using Pkg
Pkg.Registry.add("General")
env:
JULIA_PKG_SERVER: ""
# Set the registry worktree to an older state to simulate the cache storing an old version of the registry.
- name: Use outdated General worktree
run: git -C ~/.julia/registries/General reset --hard HEAD~20
test-restore-cloned-registry:
needs: test-save-cloned-registry
runs-on: ubuntu-latest
steps:
- uses: actions/checkout@b4ffde65f46336ab88eb53be808477a3936bae11
- name: Add General registry clone
shell: julia --color=yes {0}
run: |
using Pkg
Pkg.Registry.add("General")
env:
JULIA_PKG_SERVER: ""
- name: Restore cache
uses: ./
with:
cache-name: ${{ needs.test-save-cloned-registry.outputs.cache-name }}
# Cannot require a successful cache delete on forked PRs as the permissions for actions is limited to read
delete-old-caches: ${{ github.event.pull_request.head.repo.fork && 'false' || 'required' }}
- name: Test registry is not corrupt
shell: julia --color=yes {0}
run: |
using Pkg
Pkg.Registry.update()

View File

@@ -69,19 +69,32 @@ runs:
else else
depot="~/.julia" depot="~/.julia"
fi fi
echo "depot=$depot" >> $GITHUB_OUTPUT echo "depot=$depot" | tee -a "$GITHUB_OUTPUT"
[ "${{ inputs.cache-artifacts }}" = "true" ] && A_PATH="${depot}/artifacts"
echo "artifacts-path=$A_PATH" >> $GITHUB_OUTPUT cache_paths=()
[ "${{ inputs.cache-packages }}" = "true" ] && P_PATH="${depot}/packages" artifacts_path="${depot}/artifacts"
echo "packages-path=$P_PATH" >> $GITHUB_OUTPUT [ "${{ inputs.cache-artifacts }}" = "true" ] && cache_paths+=("$artifacts_path")
[ "${{ inputs.cache-registries }}" = "true" ] && R_PATH="${depot}/registries" packages_path="${depot}/packages"
echo "registries-path=$R_PATH" >> $GITHUB_OUTPUT [ "${{ inputs.cache-packages }}" = "true" ] && cache_paths+=("$packages_path")
[ "${{ inputs.cache-compiled }}" = "true" ] && PCC_PATH="${depot}/compiled" registries_path="${depot}/registries"
echo "compiled-path=$PCC_PATH" >> $GITHUB_OUTPUT if [ "${{ inputs.cache-registries }}" = "true" ]; then
[ "${{ inputs.cache-scratchspaces }}" = "true" ] && S_PATH="${depot}/scratchspaces" if [ ! -d "${registries_path/#\~/$HOME}" ]; then
echo "scratchspaces-path=$S_PATH" >> $GITHUB_OUTPUT cache_paths+=("$registries_path")
[ "${{ inputs.cache-logs }}" = "true" ] && L_PATH="${depot}/logs" else
echo "logs-path=$L_PATH" >> $GITHUB_OUTPUT echo "::warning::Julia depot registries already exist. Skipping restoring of cached registries to avoid potential merge conflicts when updating. Please ensure that \`julia-actions/cache\` precedes any workflow steps which add registries."
fi
fi
compiled_path="${depot}/compiled"
[ "${{ inputs.cache-compiled }}" = "true" ] && cache_paths+=("$compiled_path")
scratchspaces_path="${depot}/scratchspaces"
[ "${{ inputs.cache-scratchspaces }}" = "true" ] && cache_paths+=("$scratchspaces_path")
logs_path="${depot}/logs"
[ "${{ inputs.cache-logs }}" = "true" ] && cache_paths+=("$logs_path")
{
echo "cache-paths<<EOF"
printf "%s\n" "${cache_paths[@]}"
echo "EOF"
} | tee -a "$GITHUB_OUTPUT"
shell: bash shell: bash
env: env:
PATH_DELIMITER: ${{ runner.OS == 'Windows' && ';' || ':' }} PATH_DELIMITER: ${{ runner.OS == 'Windows' && ';' || ':' }}
@@ -109,20 +122,17 @@ runs:
id: cache id: cache
with: with:
path: | path: |
${{ steps.paths.outputs.artifacts-path }} ${{ steps.paths.outputs.cache-paths }}
${{ steps.paths.outputs.packages-path }}
${{ steps.paths.outputs.registries-path }}
${{ steps.paths.outputs.scratchspaces-path }}
${{ steps.paths.outputs.compiled-path }}
${{ steps.paths.outputs.logs-path }}
key: ${{ steps.keys.outputs.key }} key: ${{ steps.keys.outputs.key }}
restore-keys: ${{ steps.keys.outputs.restore-key }} restore-keys: ${{ steps.keys.outputs.restore-key }}
enableCrossOsArchive: false enableCrossOsArchive: false
- name: list restored depot directory sizes # if it wasn't restored make the depot anyway as a signal that this action ran
if: ${{ steps.cache.outputs.cache-hit == 'true' }} # for other julia actions to check, like https://github.com/julia-actions/julia-buildpkg/pull/41
run: du -shc ${{ steps.paths.outputs.depot }}/* || true - name: make depot if not restored, then list depot directory sizes
run: |
mkdir -p ${{ steps.paths.outputs.depot }}
du -shc ${{ steps.paths.outputs.depot }}/* || true
shell: bash shell: bash
# github and actions/cache doesn't provide a way to update a cache at a given key, so we delete any # github and actions/cache doesn't provide a way to update a cache at a given key, so we delete any
@@ -130,21 +140,21 @@ runs:
# Not windows # Not windows
- uses: pyTooling/Actions/with-post-step@adef08d3bdef092282614f3b683897cefae82ee3 - uses: pyTooling/Actions/with-post-step@adef08d3bdef092282614f3b683897cefae82ee3
if: ${{ inputs.delete-old-caches == 'true' && runner.OS != 'Windows' }} if: ${{ inputs.delete-old-caches != 'false' && runner.OS != 'Windows' }}
with: with:
# seems like there has to be a `main` step in this action. Could list caches for info if we wanted # seems like there has to be a `main` step in this action. Could list caches for info if we wanted
# main: julia ${{ github.action_path }}/handle_caches.jl "${{ github.repository }}" "list" # main: julia ${{ github.action_path }}/handle_caches.jl "${{ github.repository }}" "list"
main: echo "" main: echo ""
post: julia $GITHUB_ACTION_PATH/handle_caches.jl "${{ github.repository }}" "rm" "${{ steps.keys.outputs.restore-key }}" post: julia $GITHUB_ACTION_PATH/handle_caches.jl rm "${{ github.repository }}" "${{ steps.keys.outputs.restore-key }}" "${{ github.ref }}" "${{ inputs.delete-old-caches != 'required' }}"
env: env:
GH_TOKEN: ${{ inputs.token }} GH_TOKEN: ${{ inputs.token }}
# Windows (because this action uses command prompt on windows) # Windows (because this action uses command prompt on windows)
- uses: pyTooling/Actions/with-post-step@adef08d3bdef092282614f3b683897cefae82ee3 - uses: pyTooling/Actions/with-post-step@adef08d3bdef092282614f3b683897cefae82ee3
if: ${{ inputs.delete-old-caches == 'true' && runner.OS == 'Windows' }} if: ${{ inputs.delete-old-caches != 'false' && runner.OS == 'Windows' }}
with: with:
main: echo "" main: echo ""
post: cd %GITHUB_ACTION_PATH% && julia handle_caches.jl "${{ github.repository }}" "rm" "${{ steps.keys.outputs.restore-key }}" post: cd %GITHUB_ACTION_PATH% && julia handle_caches.jl rm "${{ github.repository }}" "${{ steps.keys.outputs.restore-key }}" "${{ github.ref }}" "${{ inputs.delete-old-caches != 'required' }}"
env: env:
GH_TOKEN: ${{ inputs.token }} GH_TOKEN: ${{ inputs.token }}

View File

@@ -1,58 +1,67 @@
using Pkg, Dates using Pkg, Dates
function handle_caches() function handle_caches()
repo = ARGS[1] subcommand = ARGS[1]
func = ARGS[2]
restore_key = get(ARGS, 3, "")
if func == "list" if subcommand == "list"
repo = ARGS[2]
println("Listing existing caches") println("Listing existing caches")
run(`gh cache list --limit 100 --repo $repo`) run(`gh cache list --limit 100 --repo $repo`)
elseif func == "rm" elseif subcommand == "rm"
caches = String[] repo, restore_key, ref = ARGS[2:4]
failed = String[] allow_failure = ARGS[5] == "true"
for _ in 1:5 # limit to avoid accidental rate limiting
hits = split(strip(read(`gh cache list --limit 100 --repo $repo`, String)), keepempty=false) endpoint = "/repos/$repo/actions/caches"
search_again = length(hits) == 100 page = 1
filter!(startswith(restore_key), hits) per_page = 100
isempty(hits) && break escaped_restore_key = replace(restore_key, "\"" => "\\\"")
# We can delete everything that matches the restore key because the new cache is saved later. query = ".actions_caches[] | select(.key | startswith(\"$escaped_restore_key\")) | .id"
for c in hits
deletions = String[]
failures = String[]
while 1 <= page <= 5 # limit to avoid accidental rate limiting
cmd = `gh api -X GET $endpoint -F ref=$ref -F per_page=$per_page -F page=$page --jq $query`
ids = split(read(cmd, String); keepempty=false)
page = length(ids) == per_page ? page + 1 : -1
# We can delete all cache entries on this branch that matches the restore key
# because the new cache is saved later.
for id in ids
try try
run(`gh cache delete $(split(c)[1]) --repo $repo`) run(`gh cache delete $id --repo $repo`)
push!(caches, c) push!(deletions, id)
catch e catch e
@error e @error e
push!(failed, c) push!(failures, id)
end end
end end
search_again || break
end end
if isempty(failed) && isempty(caches) if isempty(failures) && isempty(deletions)
println("No existing caches found for restore key `$restore_key`") println("No existing caches found on ref `$ref` matching restore key `$restore_key`")
else else
if !isempty(failed) if !isempty(failures)
println("Failed to delete $(length(failed)) existing caches for restore key `$restore_key`") println("Failed to delete $(length(failures)) existing caches on ref `$ref` matching restore key `$restore_key`")
println.(failed) println.(failures)
@info """ @info """
To delete caches you need to grant the following to the default `GITHUB_TOKEN` by adding To delete caches you need to grant the following to the default `GITHUB_TOKEN` by adding
this to your yml: this to your workflow:
``` ```
permissions: permissions:
actions: write actions: write
contents: read contents: read
``` ```
(Note this won't work for fork PRs but should once merged) (Note this won't work for fork PRs but should once merged)
Or provide a token with `repo` scope via the `token` input option. Or provide a token with `repo` scope via the `token` input option.
See https://cli.github.com/manual/gh_cache_delete See https://cli.github.com/manual/gh_cache_delete
""" """
allow_failure || exit(1)
end end
if !isempty(caches) if !isempty(deletions)
println("$(length(caches)) existing caches deleted that match restore key `$restore_key`:") println("Deleted $(length(deletions)) caches on ref `$ref` matching restore key `$restore_key`")
println.(caches) println.(deletions)
end end
end end
else else
throw(ArgumentError("Unexpected second argument: $func")) throw(ArgumentError("Unexpected subcommand: $subcommand"))
end end
end end