diff --git a/.github/workflows/build_charm.md b/.github/workflows/build_charm.md index 33e0bcae..5f6db7cd 100644 --- a/.github/workflows/build_charm.md +++ b/.github/workflows/build_charm.md @@ -16,3 +16,8 @@ with: cache: true ``` remember to add your charm's branch(es) to charmcraftcache by running `ccc add` or by [opening an issue](https://github.com/canonical/charmcraftcache-hub/issues/new?assignees=&labels=add-charm&projects=&template=add_charm_branch.yaml&title=Add+charm+branch). + +### Required charmcraft.yaml syntax +Only [ST124 - Multi-base platforms in craft tools](https://docs.google.com/document/d/1QVHxZumruKVZ3yJ2C74qWhvs-ye5I9S6avMBDHs2YcQ/edit) "shorthand notation" syntax is supported in charmcraft.yaml + +Follow [step #1 from charmcraftst124's documentation](https://github.com/canonical/charmcraftst124?tab=readme-ov-file#step-1-update-charmcraftyaml-to-supported-syntax) diff --git a/.github/workflows/build_charm.yaml b/.github/workflows/build_charm.yaml index 6cac8df8..35504031 100644 --- a/.github/workflows/build_charm.yaml +++ b/.github/workflows/build_charm.yaml @@ -51,11 +51,11 @@ on: outputs: artifact-prefix: description: Charm packages are uploaded to GitHub artifacts beginning with this prefix - value: ${{ jobs.collect-bases.outputs.artifact-prefix-with-inputs }} + value: ${{ jobs.collect-platforms.outputs.artifact-prefix-with-inputs }} jobs: - collect-bases: - name: Collect bases for charm | ${{ inputs.path-to-charm-directory }} + collect-platforms: + name: Collect platforms for charm | ${{ inputs.path-to-charm-directory }} runs-on: ubuntu-latest timeout-minutes: 5 steps: @@ -68,27 +68,32 @@ jobs: github-token: ${{ secrets.GITHUB_TOKEN }} - name: Install CLI run: pipx install git+https://github.com/canonical/data-platform-workflows@'${{ steps.workflow-version.outputs.sha }}'#subdirectory=python/cli + - name: Install charmcraftst124 + run: pipx install charmcraftst124 - name: Checkout uses: actions/checkout@v4 - - name: Collect charm bases to build from charmcraft.yaml + - name: Check if supported ST124 shorthand notation syntax is used in charmcraft.yaml + working-directory: ${{ inputs.path-to-charm-directory }} + run: charmcraftst124 check-charmcraft-yaml -v + - name: Collect charm platforms to build from charmcraft.yaml id: collect - run: collect-charm-bases --directory='${{ inputs.path-to-charm-directory }}' --cache='${{ inputs.cache }}' + run: collect-charm-platforms --directory='${{ inputs.path-to-charm-directory }}' --cache='${{ inputs.cache }}' outputs: - bases: ${{ steps.collect.outputs.bases }} + platforms: ${{ steps.collect.outputs.platforms }} artifact-prefix-with-inputs: ${{ inputs.artifact-prefix || steps.collect.outputs.default_prefix }} build: strategy: matrix: - base: ${{ fromJSON(needs.collect-bases.outputs.bases) }} - name: 'Build charm | base #${{ matrix.base.id }}' + platform: ${{ fromJSON(needs.collect-platforms.outputs.platforms) }} + name: 'Build charm | ${{ matrix.platform.name }}' needs: - - collect-bases - runs-on: ${{ matrix.base.runner }} + - collect-platforms + runs-on: ${{ matrix.platform.runner }} timeout-minutes: 120 steps: - name: (GitHub-hosted ARM runner) Install libpq-dev - if: ${{ matrix.base.runner == 'Ubuntu_ARM64_4C_16G_02' }} + if: ${{ matrix.platform.runner == 'Ubuntu_ARM64_4C_16G_02' }} # Needed for `charmcraftcache` to resolve dependencies (for postgresql charms with psycopg2) run: | sudo apt-get update @@ -136,6 +141,7 @@ jobs: poetry config warnings.export false pipx install charmcraftcache + pipx install charmcraftst124 - run: snap list - name: Pack charm id: pack @@ -143,12 +149,10 @@ jobs: run: | if '${{ inputs.cache }}' then - sg lxd -c "charmcraftcache pack -v --bases-index='${{ matrix.base.id }}'" + echo 'Cache not yet supported with ST124 syntax' + exit 1 else - # Workaround for https://github.com/canonical/charmcraft/issues/1389 on charmcraft 2 - touch requirements.txt - - sg lxd -c "charmcraft pack -v --bases-index='${{ matrix.base.id }}'" + sg lxd -c "charmcraftst124 pack -v --platform='${{ matrix.platform.name }}'" fi env: # Used by charmcraftcache (to avoid GitHub API rate limit) @@ -157,14 +161,14 @@ jobs: if: ${{ failure() && steps.pack.outcome == 'failure' }} uses: actions/upload-artifact@v4 with: - name: logs-charmcraft-build-${{ needs.collect-bases.outputs.artifact-prefix-with-inputs }}-base-${{ matrix.base.id }} + name: logs-charmcraft-build-${{ needs.collect-platforms.outputs.artifact-prefix-with-inputs }}-platform-${{ matrix.platform.name_in_artifact }} path: ~/.local/state/charmcraft/log/ if-no-files-found: error - run: touch .empty - name: Upload charm package uses: actions/upload-artifact@v4 with: - name: ${{ needs.collect-bases.outputs.artifact-prefix-with-inputs }}-base-${{ matrix.base.id }} + name: ${{ needs.collect-platforms.outputs.artifact-prefix-with-inputs }}-platform-${{ matrix.platform.name_in_artifact }} # .empty file required to preserve directory structure # See https://github.com/actions/upload-artifact/issues/344#issuecomment-1379232156 path: | diff --git a/.github/workflows/build_rock.md b/.github/workflows/build_rock.md index c933765b..89d2eaa4 100644 --- a/.github/workflows/build_rock.md +++ b/.github/workflows/build_rock.md @@ -8,4 +8,16 @@ jobs: build: name: Build rock uses: canonical/data-platform-workflows/.github/workflows/build_rock.yaml@v0.0.0 -``` \ No newline at end of file +``` + +### Supported `platforms` syntax in rockcraft.yaml +Only "shorthand notation" is supported + +Example rockcraft.yaml +```yaml +platforms: + amd64: + arm64: +``` + +`build-on` and `build-for` are not supported diff --git a/.github/workflows/build_rock.yaml b/.github/workflows/build_rock.yaml index a46fa4a3..2e295984 100644 --- a/.github/workflows/build_rock.yaml +++ b/.github/workflows/build_rock.yaml @@ -28,10 +28,10 @@ on: outputs: artifact-prefix: description: Rock packages are uploaded to GitHub artifacts beginning with this prefix - value: ${{ jobs.collect-bases.outputs.artifact-prefix-with-inputs }} + value: ${{ jobs.collect-platforms.outputs.artifact-prefix-with-inputs }} jobs: - collect-bases: + collect-platforms: name: Collect platforms for rock | ${{ inputs.path-to-rock-directory }} runs-on: ubuntu-latest timeout-minutes: 5 @@ -49,19 +49,19 @@ jobs: uses: actions/checkout@v4 - name: Collect rock platforms to build from rockcraft.yaml id: collect - run: collect-rock-bases --directory='${{ inputs.path-to-rock-directory }}' + run: collect-rock-platforms --directory='${{ inputs.path-to-rock-directory }}' outputs: - bases: ${{ steps.collect.outputs.bases }} + platforms: ${{ steps.collect.outputs.platforms }} artifact-prefix-with-inputs: ${{ inputs.artifact-prefix || steps.collect.outputs.default_prefix }} build: strategy: matrix: - base: ${{ fromJSON(needs.collect-bases.outputs.bases) }} - name: 'Build rock | ${{ matrix.base.id }}' + platform: ${{ fromJSON(needs.collect-platforms.outputs.platforms) }} + name: 'Build rock | ${{ matrix.platform.name }}' needs: - - collect-bases - runs-on: ${{ matrix.base.runner }} + - collect-platforms + runs-on: ${{ matrix.platform.runner }} timeout-minutes: 15 steps: - name: Get workflow version @@ -93,19 +93,19 @@ jobs: - name: Pack rock id: pack working-directory: ${{ inputs.path-to-rock-directory }} - run: sg lxd -c "rockcraft pack -v --platform='${{ matrix.base.id }}'" + run: sg lxd -c "rockcraft pack -v --platform='${{ matrix.platform.name }}'" - name: Upload rockcraft logs if: ${{ failure() && steps.pack.outcome == 'failure' }} uses: actions/upload-artifact@v4 with: - name: logs-rockcraft-build-${{ inputs.artifact-prefix }}-architecture-${{ matrix.base.id }} + name: logs-rockcraft-build-${{ inputs.artifact-prefix }}-platform-${{ matrix.platform.name }} path: ~/.local/state/rockcraft/log/ if-no-files-found: error - run: touch .empty - name: Upload rock package uses: actions/upload-artifact@v4 with: - name: ${{ needs.collect-bases.outputs.artifact-prefix-with-inputs }}-architecture-${{ matrix.base.id }} + name: ${{ needs.collect-platforms.outputs.artifact-prefix-with-inputs }}-platform-${{ matrix.platform.name }} # .empty file required to preserve directory structure # See https://github.com/actions/upload-artifact/issues/344#issuecomment-1379232156 path: | diff --git a/.github/workflows/build_snap.md b/.github/workflows/build_snap.md index 1ac88e00..181cf6a8 100644 --- a/.github/workflows/build_snap.md +++ b/.github/workflows/build_snap.md @@ -8,4 +8,35 @@ jobs: build: name: Build snap uses: canonical/data-platform-workflows/.github/workflows/build_snap.yaml@v0.0.0 -``` \ No newline at end of file +``` + +### Supported `platforms` and `architectures` syntax in snapcraft.yaml +See https://snapcraft.io/docs/architectures#how-to-create-a-snap-for-a-specific-architecture + +#### core24 +Only `platforms` is supported. `architectures` is not supported + +Only "shorthand notation" is supported + +Example snapcraft.yaml +```yaml +platforms: + amd64: + arm64: +``` + +`build-on` and `build-for` are not supported + +#### core22 +Only `architectures` is supported. `platforms` is not supported + +`architectures` must be a list of dictionaries. Each dictionary in the list must contain a `build-on` key + +Example snapcraft.yaml +```yaml +architectures: + - build-on: [amd64] + build-for: [amd64] + - build-on: [arm64] + build-for: [arm64] +``` diff --git a/.github/workflows/build_snap.yaml b/.github/workflows/build_snap.yaml index d7e46faa..f8ecee00 100644 --- a/.github/workflows/build_snap.yaml +++ b/.github/workflows/build_snap.yaml @@ -31,11 +31,11 @@ on: outputs: artifact-prefix: description: Snap packages are uploaded to GitHub artifacts beginning with this prefix - value: ${{ jobs.collect-bases.outputs.artifact-prefix-with-inputs }} + value: ${{ jobs.collect-platforms.outputs.artifact-prefix-with-inputs }} jobs: - collect-bases: - name: Collect architectures for snap | ${{ inputs.path-to-snap-project-directory }} + collect-platforms: + name: Collect platforms for snap | ${{ inputs.path-to-snap-project-directory }} runs-on: ubuntu-latest timeout-minutes: 5 steps: @@ -50,21 +50,21 @@ jobs: run: pipx install git+https://github.com/canonical/data-platform-workflows@'${{ steps.workflow-version.outputs.sha }}'#subdirectory=python/cli - name: Checkout uses: actions/checkout@v4 - - name: Collect snap architectures to build from snapcraft.yaml + - name: Collect snap platforms to build from snapcraft.yaml id: collect - run: collect-snap-bases --directory='${{ inputs.path-to-snap-project-directory }}' + run: collect-snap-platforms --directory='${{ inputs.path-to-snap-project-directory }}' outputs: - bases: ${{ steps.collect.outputs.bases }} + platforms: ${{ steps.collect.outputs.platforms }} artifact-prefix-with-inputs: ${{ inputs.artifact-prefix || steps.collect.outputs.default_prefix }} build: strategy: matrix: - base: ${{ fromJSON(needs.collect-bases.outputs.bases) }} - name: 'Build snap | ${{ matrix.base.id }}' + platform: ${{ fromJSON(needs.collect-platforms.outputs.platforms) }} + name: 'Build snap | ${{ matrix.platform.name }}' needs: - - collect-bases - runs-on: ${{ matrix.base.runner }} + - collect-platforms + runs-on: ${{ matrix.platform.runner }} timeout-minutes: 30 steps: - name: Get workflow version @@ -96,19 +96,19 @@ jobs: - name: Pack snap id: pack working-directory: ${{ inputs.path-to-snap-project-directory }} - run: sg lxd -c "snapcraft pack -v --build-for='${{ matrix.base.id }}'" + run: sg lxd -c "snapcraft pack -v --build-for='${{ matrix.platform.name }}'" - name: Upload snapcraft logs if: ${{ failure() && steps.pack.outcome == 'failure' }} uses: actions/upload-artifact@v4 with: - name: logs-snapcraft-build-${{ inputs.artifact-prefix }}-architecture-${{ matrix.base.id }} + name: logs-snapcraft-build-${{ inputs.artifact-prefix }}-platform-${{ matrix.platform.name }} path: ~/.local/state/snapcraft/log/ if-no-files-found: error - run: touch .empty - name: Upload snap package uses: actions/upload-artifact@v4 with: - name: ${{ needs.collect-bases.outputs.artifact-prefix-with-inputs }}-architecture-${{ matrix.base.id }} + name: ${{ needs.collect-platforms.outputs.artifact-prefix-with-inputs }}-platform-${{ matrix.platform.name }} # .empty file required to preserve directory structure # See https://github.com/actions/upload-artifact/issues/344#issuecomment-1379232156 path: | diff --git a/python/cli/data_platform_workflows_cli/craft_tools/collect_bases.py b/python/cli/data_platform_workflows_cli/craft_tools/collect_bases.py deleted file mode 100644 index 22c6d85a..00000000 --- a/python/cli/data_platform_workflows_cli/craft_tools/collect_bases.py +++ /dev/null @@ -1,112 +0,0 @@ -# Copyright 2022 Canonical Ltd. -# See LICENSE file for licensing details. -"""Collect bases to build - -charmcraft: "bases" -snapcraft: "architectures" -rockcraft: "platforms" - -snaps & rocks are usually built on multiple architectures but only one Ubuntu version/base -charms (subordinate) can be built on multiple Ubuntu versions -""" - -import argparse -import json -import logging -import pathlib -import sys - -import yaml - -from .. import github_actions -from . import craft - -logging.basicConfig(level=logging.INFO, stream=sys.stdout) -RUNNERS = { - craft.Architecture.X64: "ubuntu-latest", - craft.Architecture.ARM64: "Ubuntu_ARM64_4C_16G_02", -} - - -def get_bases(*, craft_: craft.Craft, yaml_data): - """Get architecture for each base - - For charms, multiple bases can have the same architecture - (e.g. Ubuntu 20.04 X64 and Ubuntu 22.04 X64) - - For snaps & rocks, the Ubuntu version is the same for all architectures. - """ - if craft_ is craft.Craft.ROCK: - # https://canonical-rockcraft.readthedocs-hosted.com/en/latest/reference/rockcraft.yaml/#platforms - return [craft.Architecture(arch) for arch in yaml_data["platforms"]] - if craft_ is craft.Craft.SNAP: - bases = yaml_data.get("architectures") - if not bases: - # Default to X64 - return [craft.Architecture.X64] - elif craft_ is craft.Craft.CHARM: - bases = yaml_data["bases"] - else: - raise ValueError - arch_for_bases = [] - for platform in bases: - if craft_ is craft.Craft.SNAP: - # https://snapcraft.io/docs/explanation-architectures - build_on_architectures = platform["build-on"] - elif craft_ is craft.Craft.CHARM: - # https://discourse.charmhub.io/t/charmcraft-bases-provider-support/4713 - build_on = platform.get("build-on") - if build_on: - assert isinstance(build_on, list) and len(build_on) == 1 - platform = build_on[0] - build_on_architectures = platform.get("architectures") - if not build_on_architectures: - # Default to X64 - arch_for_bases.append(craft.Architecture.X64) - continue - else: - raise ValueError - assert ( - len(build_on_architectures) == 1 - ), f"Multiple architectures ({build_on_architectures}) in one ({craft_.value}craft.yaml) base/architecture entry not supported. Use one entry per architecture" - arch_for_bases.append(craft.Architecture(build_on_architectures[0])) - return arch_for_bases - - -def collect(craft_: craft.Craft): - """Collect bases to build from *craft.yaml""" - parser = argparse.ArgumentParser() - parser.add_argument("--directory", required=True) - if craft_ is craft.Craft.CHARM: - parser.add_argument("--cache", required=True) - args = parser.parse_args() - craft_file = pathlib.Path(args.directory, f"{craft_.value}craft.yaml") - if craft_ is craft.Craft.SNAP: - craft_file = craft_file.parent / "snap" / craft_file.name - yaml_data = yaml.safe_load(craft_file.read_text()) - bases_ = get_bases(craft_=craft_, yaml_data=yaml_data) - bases = [] - for index, architecture in enumerate(bases_): - # id used to select base in `*craft pack` - if craft_ is craft.Craft.CHARM: - id_ = index - else: - id_ = architecture.value - bases.append({"id": id_, "runner": RUNNERS[architecture]}) - github_actions.output["bases"] = json.dumps(bases) - default_prefix = f'packed-{craft_.value}-{args.directory.replace("/", "-")}' - if craft_ is craft.Craft.CHARM: - default_prefix = f'packed-{craft_.value}-cache-{args.cache}-{args.directory.replace("/", "-")}' - github_actions.output["default_prefix"] = default_prefix - - -def snap(): - collect(craft.Craft.SNAP) - - -def rock(): - collect(craft.Craft.ROCK) - - -def charm(): - collect(craft.Craft.CHARM) diff --git a/python/cli/data_platform_workflows_cli/craft_tools/collect_platforms.py b/python/cli/data_platform_workflows_cli/craft_tools/collect_platforms.py new file mode 100644 index 00000000..82567a0f --- /dev/null +++ b/python/cli/data_platform_workflows_cli/craft_tools/collect_platforms.py @@ -0,0 +1,100 @@ +# Copyright 2022 Canonical Ltd. +# See LICENSE file for licensing details. +"""Collect platforms to build + +charmcraft: Only ST124 shorthand notation `platforms` are supported +snapcraft: (ST124 not supported) core22 `architectures` and core24 shorthand `platforms` supported +rockcraft: (ST124 not supported) shorthand `platforms` supported + +snaps & rocks are usually built on multiple architectures but only one Ubuntu version/base +charms (subordinate) can be built on multiple Ubuntu versions +""" + +import argparse +import json +import logging +import pathlib +import sys + +import yaml + +from .. import github_actions +from . import craft + +logging.basicConfig(level=logging.INFO, stream=sys.stdout) +RUNNERS = { + craft.Architecture.X64: "ubuntu-latest", + craft.Architecture.ARM64: "Ubuntu_ARM64_4C_16G_02", +} + + +def collect(craft_: craft.Craft): + """Collect platforms to build from *craft.yaml""" + parser = argparse.ArgumentParser() + parser.add_argument("--directory", required=True) + if craft_ is craft.Craft.CHARM: + parser.add_argument("--cache", required=True) + args = parser.parse_args() + craft_file = pathlib.Path(args.directory, f"{craft_.value}craft.yaml") + if craft_ is craft.Craft.SNAP: + craft_file = craft_file.parent / "snap" / craft_file.name + yaml_data = yaml.safe_load(craft_file.read_text()) + platforms = [] + if craft_ is craft.Craft.CHARM: + for platform in yaml_data["platforms"]: + # Example `platform`: "ubuntu@22.04:amd64" + architecture = craft.Architecture(platform.split(":")[-1]) + platforms.append( + { + "name": platform, + "runner": RUNNERS[architecture], + "name_in_artifact": platform.replace(":", "-"), + } + ) + elif craft_ is craft.Craft.ROCK: + for platform in yaml_data["platforms"]: + # Example `platform`: "amd64" + architecture = craft.Architecture(platform) + platforms.append({"name": platform, "runner": RUNNERS[architecture]}) + elif craft_ is craft.Craft.SNAP: + if yaml_data["base"] == "core24": + platforms_ = yaml_data["platforms"] + if not isinstance(platforms_, dict): + raise TypeError("Expected type 'dict' for snapcraft.yaml 'platforms'") + for value in platforms_.values(): + if value is not None: + raise ValueError( + "Only shorthand notation supported in snapcraft.yaml 'platforms'. " + "'build-on' and 'build-for' not supported" + ) + for platform in platforms_: + # Example `platform`: "amd64" + architecture = craft.Architecture(platform) + platforms.append({"name": platform, "runner": RUNNERS[architecture]}) + elif yaml_data["base"] == "core22": + for entry in yaml_data["architectures"]: + # Example: "amd64" + platform = entry["build-on"] + architecture = craft.Architecture(platform) + platforms.append({"name": platform, "runner": RUNNERS[architecture]}) + else: + raise ValueError(f'Unsupported snapcraft.yaml base: {repr(yaml_data["base"])}') + else: + raise ValueError + github_actions.output["platforms"] = json.dumps(platforms) + default_prefix = f'packed-{craft_.value}-{args.directory.replace("/", "-")}' + if craft_ is craft.Craft.CHARM: + default_prefix = f'packed-{craft_.value}-cache-{args.cache}-{args.directory.replace("/", "-")}' + github_actions.output["default_prefix"] = default_prefix + + +def snap(): + collect(craft.Craft.SNAP) + + +def rock(): + collect(craft.Craft.ROCK) + + +def charm(): + collect(craft.Craft.CHARM) diff --git a/python/cli/data_platform_workflows_cli/parse_snap_version.py b/python/cli/data_platform_workflows_cli/parse_snap_version.py index c088b918..29c06615 100644 --- a/python/cli/data_platform_workflows_cli/parse_snap_version.py +++ b/python/cli/data_platform_workflows_cli/parse_snap_version.py @@ -17,8 +17,6 @@ def main(): install_flag = f"'--revision={args.revision}'" elif args.channel: install_flag = f"'--channel={args.channel}'" - elif args.channel_input_name == "charmcraft-snap-channel": - install_flag = "'--channel=2.x/stable'" else: install_flag = None github_actions.output["install_flag"] = install_flag diff --git a/python/cli/pyproject.toml b/python/cli/pyproject.toml index eb4b4f9f..bf7dae2b 100644 --- a/python/cli/pyproject.toml +++ b/python/cli/pyproject.toml @@ -9,9 +9,9 @@ readme = "README.md" [tool.poetry.scripts] redact-secrets = "data_platform_workflows_cli.redact_secrets:main" -collect-snap-bases = "data_platform_workflows_cli.craft_tools.collect_bases:snap" -collect-rock-bases = "data_platform_workflows_cli.craft_tools.collect_bases:rock" -collect-charm-bases = "data_platform_workflows_cli.craft_tools.collect_bases:charm" +collect-snap-platforms = "data_platform_workflows_cli.craft_tools.collect_platforms:snap" +collect-rock-platforms = "data_platform_workflows_cli.craft_tools.collect_platforms:rock" +collect-charm-platforms = "data_platform_workflows_cli.craft_tools.collect_platforms:charm" release-snap = "data_platform_workflows_cli.craft_tools.release:snap" release-rock = "data_platform_workflows_cli.craft_tools.release:rock" release-charm = "data_platform_workflows_cli.craft_tools.release:charm" diff --git a/python/pytest_plugins/pytest_operator_cache/pytest_operator_cache/_plugin.py b/python/pytest_plugins/pytest_operator_cache/pytest_operator_cache/_plugin.py index f7e1b26d..3bbf12b7 100644 --- a/python/pytest_plugins/pytest_operator_cache/pytest_operator_cache/_plugin.py +++ b/python/pytest_plugins/pytest_operator_cache/pytest_operator_cache/_plugin.py @@ -3,8 +3,6 @@ import subprocess import typing -import yaml - def pytest_configure(config): if os.environ.get("CI") == "true": @@ -19,9 +17,7 @@ def pytest_configure(config): ) -async def build_charm( - self, charm_path: typing.Union[str, os.PathLike], bases_index: int = None -) -> pathlib.Path: +async def build_charm(self, charm_path: typing.Union[str, os.PathLike]) -> pathlib.Path: charm_path = pathlib.Path(charm_path) architecture = subprocess.run( ["dpkg", "--print-architecture"], @@ -30,24 +26,8 @@ async def build_charm( encoding="utf-8", ).stdout.strip() assert architecture in ("amd64", "arm64") - if bases_index is not None: - charmcraft_yaml = yaml.safe_load((charm_path / "charmcraft.yaml").read_text()) - assert charmcraft_yaml["type"] == "charm" - base = charmcraft_yaml["bases"][bases_index] - # Handle multiple base formats - # See https://discourse.charmhub.io/t/charmcraft-bases-provider-support/4713 - build_on = base.get("build-on", [base])[0] - version = build_on["channel"] - architectures = build_on.get("architectures", ["amd64"]) - assert ( - len(architectures) == 1 - ), f"Multiple architectures ({architectures}) in one (charmcraft.yaml) base not supported. Use one base per architecture" - assert ( - architectures[0] == architecture - ), f"Architecture for {bases_index=} ({architectures[0]}) does not match host architecture ({architecture})" - packed_charms = list(charm_path.glob(f"*{version}-{architecture}.charm")) - else: - packed_charms = list(charm_path.glob(f"*-{architecture}.charm")) + # TODO unpin 22.04 (temporary solution while multi-base integration testing not supported by data-platform-workflows) + packed_charms = list(charm_path.glob(f"*-22.04-{architecture}.charm")) if len(packed_charms) == 1: # python-libjuju's model.deploy(), juju deploy, and juju bundle files expect local charms # to begin with `./` or `/` to distinguish them from Charmhub charms. @@ -58,11 +38,11 @@ async def build_charm( # `pathlib.Path`.) return packed_charms[0].resolve(strict=True) elif len(packed_charms) > 1: - message = f"More than one matching .charm file found at {charm_path=} for {architecture=}: {packed_charms}." - if bases_index is None: - message += " Specify `bases_index`" - raise ValueError(message) + raise ValueError( + f"More than one matching .charm file found at {charm_path=} for {architecture=} and " + f"Ubuntu 22.04: {packed_charms}." + ) else: raise ValueError( - f"Unable to find .charm file for {architecture=} and {bases_index=} at {charm_path=}" + f"Unable to find .charm file for {architecture=} and Ubuntu 22.04 at {charm_path=}" )