generated from canonical/template-operator
-
Notifications
You must be signed in to change notification settings - Fork 7
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
Browse files
Browse the repository at this point in the history
- Loading branch information
1 parent
5308a78
commit c2d653f
Showing
5 changed files
with
197 additions
and
1 deletion.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,61 @@ | ||
# Copyright 2024 Canonical Ltd. | ||
# See LICENSE file for licensing details. | ||
|
||
"""Architecture utilities module""" | ||
|
||
import logging | ||
import os | ||
import pathlib | ||
import platform | ||
|
||
import yaml | ||
from ops.charm import CharmBase | ||
from ops.model import BlockedStatus | ||
|
||
logger = logging.getLogger(__name__) | ||
|
||
|
||
class WrongArchitectureWarningCharm(CharmBase): | ||
"""A fake charm class that only signals a wrong architecture deploy.""" | ||
|
||
def __init__(self, *args): | ||
super().__init__(*args) | ||
|
||
hw_arch = platform.machine() | ||
self.unit.status = BlockedStatus( | ||
f"Charm incompatible with {hw_arch} architecture. " | ||
f"If this app is being refreshed, rollback" | ||
) | ||
raise RuntimeError( | ||
f"Incompatible architecture: this charm revision does not support {hw_arch}. " | ||
f"If this app is being refreshed, rollback with instructions from Charmhub docs. " | ||
f"If this app is being deployed for the first time, remove it and deploy it again " | ||
f"using a compatible revision." | ||
) | ||
|
||
|
||
def is_wrong_architecture() -> bool: | ||
"""Checks if charm was deployed on wrong architecture.""" | ||
charm_path = os.environ.get("CHARM_DIR", "") | ||
manifest_path = pathlib.Path(charm_path, "manifest.yaml") | ||
|
||
if not manifest_path.exists(): | ||
logger.error("Cannot check architecture: manifest file not found in %s", manifest_path) | ||
return False | ||
|
||
manifest = yaml.safe_load(manifest_path.read_text()) | ||
|
||
manifest_archs = [] | ||
for base in manifest["bases"]: | ||
base_archs = base.get("architectures", []) | ||
manifest_archs.extend(base_archs) | ||
|
||
hardware_arch = platform.machine() | ||
if ("amd64" in manifest_archs and hardware_arch == "x86_64") or ( | ||
"arm64" in manifest_archs and hardware_arch == "aarch64" | ||
): | ||
logger.debug("Charm architecture matches") | ||
return False | ||
|
||
logger.error("Charm architecture does not match") | ||
return True |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,65 @@ | ||
#!/usr/bin/env python3 | ||
# Copyright 2024 Canonical Ltd. | ||
# See LICENSE file for licensing details. | ||
|
||
from pathlib import Path | ||
|
||
import pytest | ||
import yaml | ||
from pytest_operator.plugin import OpsTest | ||
|
||
from . import markers | ||
from .helpers import get_charm | ||
|
||
METADATA = yaml.safe_load(Path("./metadata.yaml").read_text()) | ||
MYSQL_ROUTER_APP_NAME = METADATA["name"] | ||
|
||
|
||
@pytest.mark.group(1) | ||
@markers.amd64_only | ||
async def test_arm_charm_on_amd_host(ops_test: OpsTest) -> None: | ||
"""Tries deploying an arm64 charm on amd64 host.""" | ||
charm = await get_charm(".", "arm64", 1) | ||
|
||
resources = { | ||
"mysql-router-image": METADATA["resources"]["mysql-router-image"]["upstream-source"] | ||
} | ||
|
||
await ops_test.model.deploy( | ||
charm, | ||
application_name=MYSQL_ROUTER_APP_NAME, | ||
num_units=1, | ||
resources=resources, | ||
base="[email protected]", | ||
) | ||
|
||
await ops_test.model.wait_for_idle( | ||
apps=[MYSQL_ROUTER_APP_NAME], | ||
status="error", | ||
raise_on_error=False, | ||
) | ||
|
||
|
||
@pytest.mark.group(1) | ||
@markers.arm64_only | ||
async def test_amd_charm_on_arm_host(ops_test: OpsTest) -> None: | ||
"""Tries deploying an amd64 charm on arm64 host.""" | ||
charm = await get_charm(".", "amd64", 0) | ||
|
||
resources = { | ||
"mysql-router-image": METADATA["resources"]["mysql-router-image"]["upstream-source"] | ||
} | ||
|
||
await ops_test.model.deploy( | ||
charm, | ||
application_name=MYSQL_ROUTER_APP_NAME, | ||
num_units=1, | ||
resources=resources, | ||
base="[email protected]", | ||
) | ||
|
||
await ops_test.model.wait_for_idle( | ||
apps=[MYSQL_ROUTER_APP_NAME], | ||
status="error", | ||
raise_on_error=False, | ||
) |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,48 @@ | ||
#!/usr/bin/env python3 | ||
# Copyright 2024 Canonical Ltd. | ||
# See LICENSE file for licensing details. | ||
|
||
from architecture import is_wrong_architecture | ||
|
||
TEST_MANIFEST = """ | ||
bases: | ||
- architectures: | ||
- {arch} | ||
channel: '22.04' | ||
name: ubuntu | ||
""" | ||
|
||
|
||
def test_wrong_architecture_file_not_found(monkeypatch): | ||
"""Tests if the function returns False when the charm file doesn't exist.""" | ||
monkeypatch.setattr("os.environ", {"CHARM_DIR": "/tmp"}) | ||
monkeypatch.setattr("pathlib.Path.exists", lambda *args, **kwargs: False) | ||
assert not is_wrong_architecture() | ||
|
||
|
||
def test_wrong_architecture_amd64(monkeypatch): | ||
"""Tests if the function correctly identifies arch when charm is AMD.""" | ||
manifest = TEST_MANIFEST.format(arch="amd64") | ||
monkeypatch.setattr("os.environ", {"CHARM_DIR": "/tmp"}) | ||
monkeypatch.setattr("pathlib.Path.exists", lambda *args, **kwargs: True) | ||
monkeypatch.setattr("pathlib.Path.read_text", lambda *args, **kwargs: manifest) | ||
|
||
monkeypatch.setattr("platform.machine", lambda *args, **kwargs: "x86_64") | ||
assert not is_wrong_architecture() | ||
|
||
monkeypatch.setattr("platform.machine", lambda *args, **kwargs: "aarch64") | ||
assert is_wrong_architecture() | ||
|
||
|
||
def test_wrong_architecture_arm64(monkeypatch): | ||
"""Tests if the function correctly identifies arch when charm is ARM.""" | ||
manifest = TEST_MANIFEST.format(arch="arm64") | ||
monkeypatch.setattr("os.environ", {"CHARM_DIR": "/tmp"}) | ||
monkeypatch.setattr("pathlib.Path.exists", lambda *args, **kwargs: True) | ||
monkeypatch.setattr("pathlib.Path.read_text", lambda *args, **kwargs: manifest) | ||
|
||
monkeypatch.setattr("platform.machine", lambda *args, **kwargs: "x86_64") | ||
assert is_wrong_architecture() | ||
|
||
monkeypatch.setattr("platform.machine", lambda *args, **kwargs: "aarch64") | ||
assert not is_wrong_architecture() |