This is a quick GitHub Action that leverages the API to forcefully unregister all offline self-hosted runners.
Warning As you'd expect, this Action force-removes all offline self-hosted runners in the scope provided! It does not actually do anything to the runners (VMs, bare metal boxes, etc). This is useful if the runners are ephemeral containers and no longer exist.
Name | Description | Options | Defaults |
---|---|---|---|
GITHUB_PAT |
Personal access token (PAT) of the appropriate scope | n/a, store as a secret | n/a, store as a secret |
SCOPE_TYPE |
Scope to remove offline self-hosted runners from | "repository" "organization" "enterprise" |
"repository" |
SCOPE_NAME |
Name of the repo (owner/repo), organization, or enterprise slug | any string | GITHUB_REPOSITORY |
DRY_RUN |
Whether this is a "dry run" that'll only print runners, instead of delete | literally anything, it'll change any non-None value to True |
False |
FUZZY_NAME |
String to match within the runner name | any string | `` (an empty string, which matches anything) |
ℹ️ The personal access token (PAT) assumes it can use GITHUB_TOKEN
, but may need additional permissions depending on what you're doing with it - outlined below.
- "repository" =
repo
scope - "organization" =
repo
scope and the account must be an organization admin - "enterprise" =
manage_runners:enterprise
and the account must be an enterprise admin
ℹ️ The name matching works on matching a substring in a string, no regular expressions or other fanciness. Simple is better than complex.
Just remove all the offline self-hosted runners on this repository
- name: Delete offline self-hosted runners
uses: some-natalie/runner-reaper@v2
env:
GITHUB_PAT: ${{ secrets.RUNNER_REAPER }}
Clean up my enterprise pool of all offline self-hosted runners
- name: Delete offline self-hosted runners
uses: some-natalie/runner-reaper@v2
env:
GITHUB_PAT: ${{ secrets.RUNNER_REAPER }}
SCOPE_TYPE: "enterprise"
SCOPE_NAME: "my-enterprise-slug"
Remove all the offline runners in this organization with test
anywhere in the name.
- name: Delete offline self-hosted runners
uses: some-natalie/runner-reaper@v2
env:
GITHUB_PAT: ${{ secrets.RUNNER_REAPER }}
SCOPE_TYPE: "organization"
SCOPE_NAME: "my-awesome-org"
FUZZY_NAME: "test"
If you're using GitHub Enterprise Server or GitHub AE, this works out of the box just fine as long as you are using self-hosted runners that are capable of running Docker Actions. This Action picks up the API URL for your Server or AE automatically every time it's run, so no extra configuration is needed to just use it.
The dependencies are very deliberately chosen to be minimal - it uses python:3-slim
as the base image and only requests
for a pip package, which both likely exist on internal "approved-by-corporate-IT" package registries. If you can't use GitHub Actions to manage itself, you can still run the main.py
script manually or on a cron job with the inputs as environment variables.
If/when self-hosted runners don't unregister themselves nicely, it gets ugly. As someone who works with ephemeral self-hosted runners, when I start testing new images out, I end up with a bunch of "offline" runners that no longer exist to unregister. There's no GUI way I've found to remove all offline runners en masse, and given my terrible allergy to manual click-through-boxes work, my repositories with self-hosted runners look like 👇 pretty fast.