Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

ci: periodically run the unit tests of all GitHub-hosted published charms #1365

Merged
merged 24 commits into from
Sep 17, 2024
Merged
Show file tree
Hide file tree
Changes from 17 commits
Commits
Show all changes
24 commits
Select commit Hold shift + click to select a range
dc9288f
Allow manual execution.
tonyandrewmeyer Aug 20, 2024
068e978
WiP
tonyandrewmeyer Aug 26, 2024
dcf47b7
Initial workflow for testing viability.
tonyandrewmeyer Sep 5, 2024
8296d9c
Use api/v2 to avoid all the login hassle. Alter the existing file.
tonyandrewmeyer Sep 5, 2024
c095696
Remove the non-Canonical charms for now.
tonyandrewmeyer Sep 5, 2024
91df5c7
Typo
tonyandrewmeyer Sep 5, 2024
ef8ccb8
Remove non-ops charms. Don't run the ops tests. Handle special requir…
tonyandrewmeyer Sep 5, 2024
67c2051
Add tests for the charmcraft profiles as well.
tonyandrewmeyer Sep 5, 2024
4406f31
Install charmcraft.
tonyandrewmeyer Sep 6, 2024
c239ae0
Reorder to avoid --force.
tonyandrewmeyer Sep 6, 2024
0d1744a
Expand the skip list.
tonyandrewmeyer Sep 6, 2024
0c01745
Adjust the script name.
tonyandrewmeyer Sep 6, 2024
0009082
Pass an author explicitly.
tonyandrewmeyer Sep 6, 2024
9012e7c
Only use the standard lib and existing ops dependencies.
tonyandrewmeyer Sep 10, 2024
c931b7c
Try the generated version to see if it works.
tonyandrewmeyer Sep 10, 2024
68b9883
Add timeouts for the downloads.
tonyandrewmeyer Sep 10, 2024
6241bf9
Style and linting fixes.
tonyandrewmeyer Sep 10, 2024
a4b8ab9
Update .github/update-published-charms-tests-workflow.py
tonyandrewmeyer Sep 10, 2024
99d2f90
Move the copyright comment above the PEP 723 comment.
tonyandrewmeyer Sep 10, 2024
45aaa8c
Edit the raw text, rather than parsing the YAML and then adjusting it…
tonyandrewmeyer Sep 10, 2024
fcc2f49
Revert to the manually crafted file.
tonyandrewmeyer Sep 10, 2024
9c985d7
Lint.
tonyandrewmeyer Sep 10, 2024
815f78f
Align with main@HEAD.
tonyandrewmeyer Sep 12, 2024
6007933
Add comment explaining how to update the file.
tonyandrewmeyer Sep 17, 2024
File filter

Filter by extension

Filter by extension


Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
144 changes: 144 additions & 0 deletions .github/update-published-charms-tests-workflow.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,144 @@
#! /usr/bin/env python
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

It wasn't obvious to me that this script needed to be run manually 🙈

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

What would you recommend here? I could add a section to CONTRIBUTING.md that explains that this exists and how to update it, or I could just add that in the docstring for the script itself. Or I guess it could be run automatically on a schedule, which is probably cleanest, but I wasn't sure if we were sold enough on this idea to do that straight off.

Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Yeah, I think manual updates are a good start. We can always run this automatically later if desired.


# /// script
tonyandrewmeyer marked this conversation as resolved.
Show resolved Hide resolved
# dependencies = [
# "PyYAML",
# ]
# ///

# Copyright 2024 Canonical Ltd.
#
# Licensed under the Apache License, Version 2.0 (the "License");
# you may not use this file except in compliance with the License.
# You may obtain a copy of the License at
#
# http://www.apache.org/licenses/LICENSE-2.0
#
# Unless required by applicable law or agreed to in writing, software
# distributed under the License is distributed on an "AS IS" BASIS,
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
# See the License for the specific language governing permissions and
# limitations under the License.

"""Update a GitHub workload that runs `tox -e unit` on all published charms.

Charms that are not hosted on GitHub are skipped, as well as any charms where
the source URL could not be found.
"""

import json
import logging
import pathlib
import urllib.error
import urllib.parse
import urllib.request

import yaml

logger = logging.getLogger(__name__)


URL_BASE = 'https://api.charmhub.io/v2/charms/info'
WORKFLOW = pathlib.Path(__file__).parent / 'workflows' / 'published-charms-tests.yaml'

SKIP = {
# Handled by db-charm-tests.yaml
'postgresql-operator',
'postgresql-k8s-operator',
'mysql-operator',
'mysql-k8s-operator',
# Handled by hello-charm-tests.yaml
'hello-kubecon', # Not in the canonical org anyway (jnsgruk).
'hello-juju-charm', # Not in the canonical org anyway (juju).
# Handled by observability-charms-tests.yaml
'alertmanager-k8s-operator',
'prometheus-k8s-operator',
'grafana-k8s-operator',
# This has a redirect, which is too complicated to handle for now.
'bundle-jupyter',
# The charms are in a subfolder, which this can't handle yet.
'jimm',
'notebook-operators',
# Not ops.
'charm-prometheus-libvirt-exporter',
'juju-dashboard',
'charm-openstack-service-checks',
}


def packages():
"""Get the list of published charms from Charmhub."""
logger.info('Fetching the list of published charms')
url = 'https://charmhub.io/packages.json'
with urllib.request.urlopen(url, timeout=120) as response: # noqa: S310 (unsafe URL)
data = response.read().decode()
packages = json.loads(data)['packages']
return packages


def get_source_url(charm: str):
"""Get the source URL for a charm."""
logger.info("Looking for a 'source' URL for %s", charm)
try:
with urllib.request.urlopen( # noqa: S310 (unsafe URL)
f'{URL_BASE}/{charm}?fields=result.links', timeout=30
) as response:
data = json.loads(response.read().decode())
return data['result']['links']['source'][0]
except (urllib.error.HTTPError, KeyError):
pass
logger.info("Looking for a 'bugs-url' URL for %s", charm)
try:
with urllib.request.urlopen( # noqa: S310 (unsafe URL)
f'{URL_BASE}/{charm}?fields=result.bugs-url', timeout=30
) as response:
data = json.loads(response.read().decode())
return data['result']['bugs-url']
except (urllib.error.HTTPError, KeyError):
pass
logger.warning('Could not find a source URL for %s', charm)
return None


def url_to_charm_name(url: str):
"""Get the charm name from a URL."""
if not url:
return None
parsed = urllib.parse.urlparse(url)
if parsed.netloc != 'github.com':
logger.info('URL %s is not a GitHub URL', url)
return None
if not parsed.path.startswith('/canonical'):
# TODO: Maybe we can include some of these anyway?
# 'juju-solutions' and 'charmed-kubernetes' seem viable, for example.
logger.info('URL %s is not a Canonical charm', url)
return None
try:
return urllib.parse.urlparse(url).path.split('/')[2]
except IndexError:
logger.warning('Could not get charm name from URL %s', url)
return None


def main():
"""Update the workflow file."""
logging.basicConfig(level=logging.INFO)
charms = (url_to_charm_name(get_source_url(package['name'])) for package in packages())
with WORKFLOW.open('r') as f:
workflow = yaml.safe_load(f)
workflow['jobs']['charm-tests']['strategy']['matrix']['include'] = [
{'charm-repo': f'canonical/{charm}'} for charm in charms if charm and charm not in SKIP
]
with WORKFLOW.open('w') as f:
yaml.dump(workflow, f)
# yaml.safe_load transforms "on" to "true". See https://github.com/yaml/pyyaml/issues/376
# The 'run' step that patches the requirements files also ends up looking
# rather unfriendly, but it's still functionally the same, so works.
with WORKFLOW.open('r') as f:
content = f.read().replace('true:', 'on:')
tonyandrewmeyer marked this conversation as resolved.
Show resolved Hide resolved
with WORKFLOW.open('w') as f:
f.write(content)


if __name__ == '__main__':
main()
104 changes: 104 additions & 0 deletions .github/workflows/published-charms-tests.yaml
Original file line number Diff line number Diff line change
@@ -0,0 +1,104 @@
name: Broad Charm Compatibility Tests
on:
schedule:
- cron: 0 1 25 * *
workflow_dispatch: null
jobs:
charm-tests:
runs-on: ubuntu-latest
steps:
- name: Checkout the ${{ matrix.charm-repo }} repository
uses: actions/checkout@v4
with:
repository: ${{ matrix.charm-repo }}
- name: Checkout the operator repository
uses: actions/checkout@v4
with:
path: myops
- name: Install patch dependencies
run: pip install poetry~=1.6
- name: Update 'ops' dependency in test charm to latest
run: "rm -rf myops/test\nif [ -e \"test-requirements.txt\" ]; then\n sed -i\
tonyandrewmeyer marked this conversation as resolved.
Show resolved Hide resolved
\ -e \"/^ops[ ><=]/d\" -e \"/canonical\\/operator/d\" -e \"/#egg=ops/d\" test-requirements.txt\n\
\ echo -e \"\\ngit+$GITHUB_SERVER_URL/$GITHUB_REPOSITORY@$GITHUB_SHA#egg=ops\"\
\ >> test-requirements.txt\nfi\nif [ -e \"requirements-charmcraft.txt\" ];\
\ then\n sed -i -e \"/^ops[ ><=]/d\" -e \"/canonical\\/operator/d\" -e \"\
/#egg=ops/d\" requirements-charmcraft.txt\n echo -e \"\\ngit+$GITHUB_SERVER_URL/$GITHUB_REPOSITORY@$GITHUB_SHA#egg=ops\"\
\ >> requirements-charmcraft.txt\nfi\nif [ -e \"requirements.txt\" ]; then\n\
\ sed -i -e \"/^ops[ ><=]/d\" -e \"/canonical\\/operator/d\" -e \"/#egg=ops/d\"\
\ requirements.txt\n echo -e \"\\ngit+$GITHUB_SERVER_URL/$GITHUB_REPOSITORY@$GITHUB_SHA#egg=ops\"\
\ >> requirements.txt\nelse\n sed -i -e \"s/^ops[ ><=].*/ops = {path = \\\
\"myops\\\"}/\" pyproject.toml\n poetry lock --no-update\nfi\n"
- name: Install dependencies
run: pip install tox~=4.2
- name: Run the charm's unit tests
run: tox -vve unit
strategy:
fail-fast: false
matrix:
include:
- charm-repo: canonical/data-platform-libs
- charm-repo: canonical/kafka-operator
- charm-repo: canonical/mongodb-operator
- charm-repo: canonical/traefik-k8s-operator
- charm-repo: canonical/mysql-router-k8s-operator
- charm-repo: canonical/pgbouncer-k8s-operator
- charm-repo: canonical/discourse-k8s-operator
- charm-repo: canonical/s3-integrator
- charm-repo: canonical/zookeeper-operator
- charm-repo: canonical/loki-k8s-operator
- charm-repo: canonical/openfga-operator
- charm-repo: canonical/indico-operator
- charm-repo: canonical/trino-k8s-operator
- charm-repo: canonical/route53-lego-k8s-operator
- charm-repo: canonical/wordpress-k8s-operator
- charm-repo: canonical/temporal-k8s-operator
- charm-repo: canonical/jenkins-agent-operator
- charm-repo: canonical/seldon-core-operator
- charm-repo: canonical/nginx-ingress-integrator-operator
- charm-repo: canonical/oathkeeper-operator
- charm-repo: canonical/jenkins-agent-k8s-operator
- charm-repo: canonical/grafana-agent-k8s-operator
- charm-repo: canonical/namecheap-lego-k8s-operator
- charm-repo: canonical/superset-k8s-operator
- charm-repo: canonical/identity-platform-login-ui-operator
- charm-repo: canonical/temporal-admin-k8s-operator
- charm-repo: canonical/dex-auth-operator
- charm-repo: canonical/temporal-ui-k8s-operator
- charm-repo: canonical/temporal-worker-k8s-operator
- charm-repo: canonical/content-cache-k8s-operator
- charm-repo: canonical/ranger-k8s-operator
- charm-repo: canonical/oauth2-proxy-k8s-operator
- charm-repo: canonical/livepatch-k8s-operator
- charm-repo: canonical/self-signed-certificates-operator
- charm-repo: canonical/saml-integrator-operator
- charm-repo: canonical/smtp-integrator-operator
- charm-repo: canonical/manual-tls-certificates-operator
- charm-repo: canonical/hardware-observer-operator
charmcraft-profile-tests:
runs-on: ubuntu-latest
steps:
- name: Install charmcraft
run: sudo snap install charmcraft --classic
- name: Charmcraft init
run: charmcraft init --profile=${{ matrix.profile }} --author=charm-tech
- name: Checkout the operator repository
uses: actions/checkout@v4
with:
path: myops
- name: Update 'ops' dependency in test charm to latest
run: "rm -rf myops/test\nif [ -e \"requirements.txt\" ]; then\n sed -i -e \"\
/^ops[ ><=]/d\" -e \"/canonical\\/operator/d\" -e \"/#egg=ops/d\" requirements.txt\n\
\ echo -e \"\\ngit+$GITHUB_SERVER_URL/$GITHUB_REPOSITORY@$GITHUB_SHA#egg=ops\"\
\ >> requirements.txt\nfi\n"
- name: Install dependencies
run: pip install tox~=4.2
- name: Run the charm's unit tests
run: tox -vve unit
strategy:
fail-fast: false
matrix:
include:
- profile: machine
- profile: kubernetes
- profile: simple
1 change: 1 addition & 0 deletions .github/workflows/tiobe.yaml
Original file line number Diff line number Diff line change
Expand Up @@ -3,6 +3,7 @@ name: TIOBE Quality Checks
on:
schedule:
- cron: '0 7 1 * *'
workflow_dispatch:
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

unrelated change

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Yeah, I'm not sure what the story is with this, because main@HEAD has:

on:
  workflow_dispatch:
  schedule:
    - cron:  '0 7 1 * *'

So I don't understand why there's no complaint about a conflict, or the workflow_dispatch line being both above and below (and I could just remove the below one, which I must have had in an earlier version in my fork). I'll try to figure out what's happening here.


jobs:
TICS:
Expand Down
Loading