mirror of
https://github.com/julia-actions/cache.git
synced 2026-02-12 09:26:53 +08:00
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
This commit is contained in:
23
.github/workflows/CI.yml
vendored
23
.github/workflows/CI.yml
vendored
@@ -25,12 +25,11 @@ jobs:
|
|||||||
outputs:
|
outputs:
|
||||||
cache-name: ${{ steps.name.outputs.cache-name }}
|
cache-name: ${{ steps.name.outputs.cache-name }}
|
||||||
steps:
|
steps:
|
||||||
- name: Generate random file
|
- name: Generate random cache-name
|
||||||
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_name=$(head -n 100 </dev/urandom | shasum -a 256 | cut -d ' ' -f 1)
|
||||||
|
echo "cache-name=$cache_name" | tee -a "$GITHUB_OUTPUT"
|
||||||
|
|
||||||
test-save:
|
test-save:
|
||||||
needs: generate-key
|
needs: generate-key
|
||||||
@@ -54,7 +53,8 @@ jobs:
|
|||||||
id: cache
|
id: cache
|
||||||
uses: ./
|
uses: ./
|
||||||
with:
|
with:
|
||||||
cache-name: ${{ needs.generate-key.outputs.cache-name }}
|
cache-name: ${{ needs.generate-key.outputs.cache-name }}-matrix
|
||||||
|
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: |
|
||||||
@@ -74,7 +74,8 @@ jobs:
|
|||||||
id: cache
|
id: cache
|
||||||
uses: ./
|
uses: ./
|
||||||
with:
|
with:
|
||||||
cache-name: ${{ needs.generate-key.outputs.cache-name }}
|
cache-name: ${{ needs.generate-key.outputs.cache-name }}-nomatrix
|
||||||
|
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: |
|
||||||
@@ -106,7 +107,9 @@ jobs:
|
|||||||
id: cache
|
id: cache
|
||||||
uses: ./
|
uses: ./
|
||||||
with:
|
with:
|
||||||
cache-name: ${{ needs.generate-key.outputs.cache-name }}
|
cache-name: ${{ needs.generate-key.outputs.cache-name }}-matrix
|
||||||
|
# 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: |
|
||||||
@@ -138,7 +141,9 @@ jobs:
|
|||||||
id: cache
|
id: cache
|
||||||
uses: ./
|
uses: ./
|
||||||
with:
|
with:
|
||||||
cache-name: ${{ needs.generate-key.outputs.cache-name }}
|
cache-name: ${{ needs.generate-key.outputs.cache-name }}-nomatrix
|
||||||
|
# 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: |
|
||||||
|
|||||||
@@ -130,21 +130,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 }}
|
||||||
|
|
||||||
|
|||||||
@@ -1,41 +1,49 @@
|
|||||||
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
|
||||||
@@ -45,14 +53,15 @@ function handle_caches()
|
|||||||
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
|
||||||
|
|
||||||
|
|||||||
Reference in New Issue
Block a user