Skip to content

Commit

Permalink
Merge tag 'cime6.0.222' into feature/update_to_cime6.0.222
Browse files Browse the repository at this point in the history
cime6.0.222
  • Loading branch information
mvertens committed Mar 9, 2024
2 parents cd2dc67 + eb13e2c commit b2bc6f5
Show file tree
Hide file tree
Showing 115 changed files with 1,485 additions and 3,560 deletions.
187 changes: 187 additions & 0 deletions .github/scripts/ghcr-prune.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,187 @@
import argparse
import logging
import requests
import re
import json
from datetime import datetime
from datetime import timedelta

description = """
This script can be used to prune container images hosted on ghcr.io.\n
Our testing workflow will build and push container images to ghcr.io
that are only used for testing. This script is used to cleanup these
temporary images.
You can filter containers by any combination of name, age, and untagged.
"""

parser = argparse.ArgumentParser(description=description, formatter_class=argparse.RawTextHelpFormatter)

parser.add_argument("--token", required=True, help='GitHub token with "repo" scope')
parser.add_argument("--org", required=True, help="Organization name")
parser.add_argument("--name", required=True, help="Package name")
parser.add_argument(
"--age", type=int, help="Filter versions by age, removing anything older than"
)
parser.add_argument(
"--filter", help="Filter which versions are consider for pruning", default=".*"
)
parser.add_argument("--untagged", action="store_true", help="Prune untagged versions")
parser.add_argument(
"--dry-run", action="store_true", help="Does not actually delete anything"
)

logging_group = parser.add_argument_group("logging")
logging_group.add_argument(
"--log-level", choices=("DEBUG", "INFO", "WARNING", "ERROR"), default="INFO"
)

kwargs = vars(parser.parse_args())

logging.basicConfig(level=kwargs["log_level"])

logger = logging.getLogger("ghcr-prune")


class GitHubPaginate:
"""Iterator for GitHub API.
Provides small wrapper for GitHub API to utilize paging in API calls.
https://docs.github.com/en/rest/using-the-rest-api/using-pagination-in-the-rest-api?apiVersion=2022-11-28
"""
def __init__(self, token, org, name, age, filter, untagged, **_):
self.token = token
self.session = None
self.url = (
f"https://api.github.com/orgs/{org}/packages/container/{name}/versions"
)
self.expired = datetime.now() - timedelta(days=age)
self.filter = re.compile(filter)
self.page = None
self.untagged = untagged

def create_session(self):
self.session = requests.Session()
self.session.headers.update(
{
"Accept": "application/vnd.github+json",
"Authorization": f"Bearer {self.token}",
"X-GitHub-Api-Version": "2022-11-28",
}
)

def grab_page(self):
if self.session is None:
raise Exception("Must create session first")

if self.url is None:
raise Exception("No more pages")

response = self.session.get(self.url)

response.raise_for_status()

remaining = int(response.headers["X-RateLimit-Remaining"])

logger.debug(f"Remaining api limit {remaining}")

if remaining <= 0:
reset = response.headers["X-RateLimit-Reset"]

raise Exception(f"Hit ratelimit will reset at {reset}")

try:
self.url = self.get_next_url(response.headers["Link"])
except Exception as e:
logger.debug(f"No Link header found {e}")

self.url = None

return self.filter_results(response.json())

def get_next_url(self, link):
match = re.match("<([^>]*)>.*", link)

if match is None:
raise Exception("Could not determine next link")

return match.group(1)

def filter_results(self, data):
results = []

logger.info(f"Processing {len(data)} containers")

for x in data:
url = x["url"]
updated_at = datetime.strptime(x["updated_at"], "%Y-%m-%dT%H:%M:%SZ")

logger.debug(f"Processing\n{json.dumps(x, indent=2)}")

try:
tag = x["metadata"]["container"]["tags"][0]
except IndexError:
logger.info(f'Found untagged version {x["id"]}')

if self.untagged:
results.append(url)

continue

if not self.filter.match(tag):
logger.info(f"Skipping {tag}, did not match filter")

continue

if updated_at < self.expired:
logger.info(
f"Pruning {tag}, updated at {updated_at}, expiration {self.expired}"
)

results.append(url)
else:
logger.info(f"Skipping {tag}, more recent than {self.expired}")

return results

def __iter__(self):
self.create_session()

return self

def __next__(self):
if self.page is None or len(self.page) == 0:
try:
self.page = self.grab_page()
except Exception as e:
logger.debug(f"StopIteration condition {e!r}")

raise StopIteration from None

try:
item = self.page.pop(0)
except IndexError:
raise StopIteration from None

return item

def remove_container(self, url):
if self.session is None:
raise Exception("Must create session first")

response = self.session.delete(url)

response.raise_for_status()

logger.debug(f"{response.headers}")


pager = GitHubPaginate(**kwargs)

for url in pager:
if kwargs["dry_run"]:
logger.info(f"Pruning {url}")
else:
pager.remove_container(url)
60 changes: 0 additions & 60 deletions .github/workflows/container.yml

This file was deleted.

24 changes: 24 additions & 0 deletions .github/workflows/ghcr-prune.yml
Original file line number Diff line number Diff line change
@@ -0,0 +1,24 @@
name: Prune ghcr.io container images
on:
schedule:
# run once a day
- cron: '0 2 * * *'

# Temporary to test
pull_request:

permissions: {}

jobs:
prune:
permissions:
packages: write
runs-on: ubuntu-latest
steps:
- uses: actions/checkout@v4
- uses: actions/setup-python@v5
- run: |
pip install requests
# remove containers older than 14 days and only generated by testing workflow
python .github/scripts/ghcr-prune.py --token ${{ secrets.GITHUB_TOKEN }} --org esmci --name cime --age 14 --filter sha- --untagged
Loading

0 comments on commit b2bc6f5

Please sign in to comment.