diff --git a/cou/apps/base.py b/cou/apps/base.py index f3f8f5fb..4b745ea2 100644 --- a/cou/apps/base.py +++ b/cou/apps/base.py @@ -38,7 +38,7 @@ UpgradeStep, ) from cou.utils.app_utils import upgrade_packages -from cou.utils.juju_utils import COUModel +from cou.utils.juju_utils import COUModel, Machine from cou.utils.openstack import ( DISTRO_TO_OPENSTACK_MAPPING, OpenStackCodenameLookup, @@ -59,8 +59,8 @@ class ApplicationUnit: name: str os_version: OpenStackRelease + machine: Machine workload_version: str = "" - machine: str = "" @dataclass @@ -77,6 +77,8 @@ class OpenStackApplication: :type model: COUModel :param charm: Name of the charm. :type charm: str + :param machines: dictionary with Machine + :type machines: dict[str, Machine] :param units: Units representation of an application. :type units: list[ApplicationUnit] :raises ApplicationError: When there are no compatible OpenStack release for the @@ -95,6 +97,7 @@ class OpenStackApplication: config: dict model: COUModel charm: str + machines: dict[str, Machine] units: list[ApplicationUnit] = field(default_factory=lambda: []) packages_to_hold: Optional[list] = field(default=None, init=False) wait_timeout: int = field(default=STANDARD_IDLE_TIMEOUT, init=False) @@ -177,7 +180,7 @@ def _populate_units(self) -> None: name=name, workload_version=unit.workload_version, os_version=compatible_os_version, - machine=unit.machine, + machine=self.machines[unit.machine], ) ) @@ -390,16 +393,8 @@ def channel_codename(self) -> OpenStackRelease: :return: OpenStackRelease object :rtype: OpenStackRelease """ - try: - # get the OpenStack release from the channel track of the application. - os_track_release_channel = OpenStackRelease(self._get_track_from_channel(self.channel)) - except ValueError: - logger.debug( - "The current channel of '%s' does not exist or is unexpectedly formatted", - self.name, - ) - os_track_release_channel = self.current_os_release - return os_track_release_channel + # get the OpenStack release from the channel track of the application. + return OpenStackRelease(self._get_track_from_channel(self.channel)) @property def can_upgrade_current_channel(self) -> bool: diff --git a/cou/apps/factory.py b/cou/apps/factory.py index 2136f634..167dd42a 100644 --- a/cou/apps/factory.py +++ b/cou/apps/factory.py @@ -22,7 +22,7 @@ from juju.client._definitions import ApplicationStatus from cou.apps.base import OpenStackApplication -from cou.utils.juju_utils import COUModel +from cou.utils.juju_utils import COUModel, Machine from cou.utils.openstack import is_charm_supported logger = logging.getLogger(__name__) @@ -41,6 +41,7 @@ def create( config: dict, model: COUModel, charm: str, + machines: dict[str, Machine], ) -> Optional[OpenStackApplication]: """Create the OpenStackApplication or registered subclasses. @@ -48,6 +49,8 @@ def create( decorator can be instantiated and used with their customized methods. :param name: Name of the application :type name: str + :param machines: Machines in the model + :type machines: dict[str, Machine] :param status: Status of the application :type status: ApplicationStatus :param config: Configuration of the application @@ -62,7 +65,14 @@ def create( # pylint: disable=too-many-arguments if is_charm_supported(charm): app_class = cls.charms.get(charm, OpenStackApplication) - return app_class(name=name, status=status, config=config, model=model, charm=charm) + return app_class( + name=name, + status=status, + config=config, + model=model, + charm=charm, + machines=machines, + ) logger.debug( "'%s' is not a supported OpenStack related application and will be ignored.", name, diff --git a/cou/steps/analyze.py b/cou/steps/analyze.py index f8517884..ccb0e99e 100644 --- a/cou/steps/analyze.py +++ b/cou/steps/analyze.py @@ -121,6 +121,7 @@ async def _populate(cls, model: juju_utils.COUModel) -> list[OpenStackApplicatio :rtype: List[OpenStackApplication] """ juju_status = await model.get_status() + juju_machines = await model.get_machines() apps = { AppFactory.create( name=app, @@ -128,6 +129,10 @@ async def _populate(cls, model: juju_utils.COUModel) -> list[OpenStackApplicatio config=await model.get_application_config(app), model=model, charm=await model.get_charm_name(app), + machines={ + unit_status.machine: juju_machines[unit_status.machine] + for unit_status in app_status.units.values() + }, ) for app, app_status in juju_status.applications.items() if app_status diff --git a/cou/utils/juju_utils.py b/cou/utils/juju_utils.py index d4c2bc23..2c59bc7b 100644 --- a/cou/utils/juju_utils.py +++ b/cou/utils/juju_utils.py @@ -16,6 +16,7 @@ import asyncio import logging import os +from dataclasses import dataclass from datetime import datetime from typing import Any, Callable, Optional @@ -129,6 +130,15 @@ async def wrapper(*args: Any, **kwargs: Any) -> Any: # pylint: disable=W9011 return _wrapper +@dataclass(frozen=True) +class Machine: + """Representation of a juju machine.""" + + machine_id: str + apps: tuple[str] + az: Optional[str] = None # simple deployments may not have azs + + class COUModel: """COU model object. @@ -499,3 +509,24 @@ async def _wait_for_active_idle() -> None: apps = await self._get_supported_apps() await _wait_for_active_idle() + + async def get_machines(self) -> dict[str, Machine]: + """Get all the machines in the model. + + :return: Dictionary of the machines found in the model. E.g: {'0': Machine0} + :rtype: dict[str, Machine] + """ + model = await self._get_model() + + return { + machine.id: Machine( + machine_id=machine.id, + apps=tuple( + unit.application + for unit in self._model.units.values() + if unit.machine.id == machine.id + ), + az=machine.hardware_characteristics.get("availability-zone"), + ) + for machine in model.machines.values() + } diff --git a/tests/unit/apps/test_auxiliary.py b/tests/unit/apps/test_auxiliary.py index f7844c1f..eb8cfce1 100644 --- a/tests/unit/apps/test_auxiliary.py +++ b/tests/unit/apps/test_auxiliary.py @@ -29,14 +29,14 @@ from tests.unit.apps.utils import add_steps -def test_auxiliary_app(status, config, model): +def test_auxiliary_app(status, config, model, apps_machines): # version 3.8 on rabbitmq can be from ussuri to yoga. In that case it will be set as yoga. expected_units = [ ApplicationUnit( name="rabbitmq-server/0", os_version=OpenStackRelease("yoga"), workload_version="3.8", - machine="0/lxd/19", + machine=apps_machines["rmq"]["0/lxd/19"], ) ] @@ -46,6 +46,7 @@ def test_auxiliary_app(status, config, model): config["auxiliary_ussuri"], model, "rabbitmq-server", + apps_machines["rmq"], ) assert app.channel == "3.8/stable" assert app.is_valid_track(app.channel) is True @@ -57,13 +58,13 @@ def test_auxiliary_app(status, config, model): assert app.current_os_release == "yoga" -def test_auxiliary_app_cs(status, config, model): +def test_auxiliary_app_cs(status, config, model, apps_machines): expected_units = [ ApplicationUnit( name="rabbitmq-server/0", os_version=OpenStackRelease("yoga"), workload_version="3.8", - machine="0/lxd/19", + machine=apps_machines["rmq"]["0/lxd/19"], ) ] rmq_status = status["rabbitmq_server"] @@ -75,6 +76,7 @@ def test_auxiliary_app_cs(status, config, model): config["auxiliary_ussuri"], model, "rabbitmq-server", + apps_machines["rmq"], ) assert app.channel == "stable" assert app.is_valid_track(app.channel) is True @@ -85,7 +87,9 @@ def test_auxiliary_app_cs(status, config, model): assert app.current_os_release == "yoga" -def test_auxiliary_upgrade_plan_ussuri_to_victoria_change_channel(status, config, model): +def test_auxiliary_upgrade_plan_ussuri_to_victoria_change_channel( + status, config, model, apps_machines +): target = OpenStackRelease("victoria") app = RabbitMQServer( "rabbitmq-server", @@ -93,6 +97,7 @@ def test_auxiliary_upgrade_plan_ussuri_to_victoria_change_channel(status, config config["auxiliary_ussuri"], model, "rabbitmq-server", + apps_machines["rmq"], ) upgrade_plan = app.generate_upgrade_plan(target) @@ -152,7 +157,7 @@ def test_auxiliary_upgrade_plan_ussuri_to_victoria_change_channel(status, config assert upgrade_plan == expected_plan -def test_auxiliary_upgrade_plan_ussuri_to_victoria(status, config, model): +def test_auxiliary_upgrade_plan_ussuri_to_victoria(status, config, model, apps_machines): target = OpenStackRelease("victoria") rmq_status = status["rabbitmq_server"] # rabbitmq already on channel 3.9 on ussuri @@ -163,6 +168,7 @@ def test_auxiliary_upgrade_plan_ussuri_to_victoria(status, config, model): config["auxiliary_ussuri"], model, "rabbitmq-server", + apps_machines["rmq"], ) upgrade_plan = app.generate_upgrade_plan(target) @@ -216,7 +222,9 @@ def test_auxiliary_upgrade_plan_ussuri_to_victoria(status, config, model): assert upgrade_plan == expected_plan -def test_auxiliary_upgrade_plan_ussuri_to_victoria_ch_migration(status, config, model): +def test_auxiliary_upgrade_plan_ussuri_to_victoria_ch_migration( + status, config, model, apps_machines +): target = OpenStackRelease("victoria") rmq_status = status["rabbitmq_server"] rmq_status.charm = "cs:amd64/focal/rabbitmq-server-638" @@ -227,6 +235,7 @@ def test_auxiliary_upgrade_plan_ussuri_to_victoria_ch_migration(status, config, config["auxiliary_ussuri"], model, "rabbitmq-server", + apps_machines["rmq"], ) upgrade_plan = app.generate_upgrade_plan(target) expected_plan = ApplicationUpgradePlan( @@ -283,7 +292,7 @@ def test_auxiliary_upgrade_plan_ussuri_to_victoria_ch_migration(status, config, assert upgrade_plan == expected_plan -def test_auxiliary_upgrade_plan_unknown_track(status, config, model): +def test_auxiliary_upgrade_plan_unknown_track(status, config, model, apps_machines): rmq_status = status["rabbitmq_server"] # 2.0 is an unknown track rmq_status.charm_channel = "2.0/stable" @@ -294,10 +303,13 @@ def test_auxiliary_upgrade_plan_unknown_track(status, config, model): config["auxiliary_ussuri"], model, "rabbitmq-server", + apps_machines["rmq"], ) -def test_auxiliary_app_unknown_version_raise_ApplicationError(status, config, model): +def test_auxiliary_app_unknown_version_raise_ApplicationError( + status, config, model, apps_machines +): with pytest.raises(ApplicationError): RabbitMQServer( "rabbitmq-server", @@ -305,10 +317,11 @@ def test_auxiliary_app_unknown_version_raise_ApplicationError(status, config, mo config["auxiliary_ussuri"], model, "rabbitmq-server", + apps_machines["rmq"], ) -def test_auxiliary_raise_error_unknown_series(status, config, model): +def test_auxiliary_raise_error_unknown_series(status, config, model, apps_machines): app_status = status["rabbitmq_server"] app_status.series = "foo" with pytest.raises(ApplicationError): @@ -318,10 +331,11 @@ def test_auxiliary_raise_error_unknown_series(status, config, model): config["auxiliary_ussuri"], model, "rabbitmq-server", + apps_machines["rmq"], ) -def test_auxiliary_raise_error_os_not_on_lookup(status, config, model, mocker): +def test_auxiliary_raise_error_os_not_on_lookup(status, config, model, mocker, apps_machines): # change OpenStack release to a version that is not on openstack_to_track_mapping.csv mocker.patch( "cou.apps.core.OpenStackApplication.current_os_release", @@ -335,12 +349,13 @@ def test_auxiliary_raise_error_os_not_on_lookup(status, config, model, mocker): config["auxiliary_ussuri"], model, "rabbitmq-server", + apps_machines["rmq"], ) with pytest.raises(ApplicationError): app.possible_current_channels -def test_auxiliary_raise_halt_upgrade(status, config, model): +def test_auxiliary_raise_halt_upgrade(status, config, model, apps_machines): target = OpenStackRelease("victoria") # source is already configured to wallaby, so the plan halt with target victoria app = RabbitMQServer( @@ -349,12 +364,13 @@ def test_auxiliary_raise_halt_upgrade(status, config, model): config["auxiliary_wallaby"], model, "rabbitmq-server", + apps_machines["rmq"], ) with pytest.raises(HaltUpgradePlanGeneration): app.generate_upgrade_plan(target) -def test_auxiliary_no_suitable_channel(status, config, model): +def test_auxiliary_no_suitable_channel(status, config, model, apps_machines): # OPENSTACK_TO_TRACK_MAPPING can't find a track for rabbitmq, focal, zed. target = OpenStackRelease("zed") app_status = status["rabbitmq_server"] @@ -365,19 +381,21 @@ def test_auxiliary_no_suitable_channel(status, config, model): config["auxiliary_wallaby"], model, "rabbitmq-server", + apps_machines["rmq"], ) with pytest.raises(ApplicationError): app.target_channel(target) -def test_ceph_mon_app(status, config, model): - """Test the correctness of instantiating CephMon.""" +def test_ceph_mon_app(status, config, model, apps_machines): + """Test the correctness of instantiating CephMonApplication.""" app = CephMon( "ceph-mon", - status["ceph-mon_xena"], + status["ceph_mon_pacific"], config["auxiliary_xena"], model, "ceph-mon", + apps_machines["ceph-mon"], ) assert app.channel == "pacific/stable" assert app.os_origin == "cloud:focal-xena" @@ -386,7 +404,7 @@ def test_ceph_mon_app(status, config, model): name="ceph-mon/0", os_version=OpenStackRelease("xena"), workload_version="16.2.0", - machine="7", + machine=apps_machines["ceph-mon"]["6"], ) ] assert app.apt_source_codename == "xena" @@ -394,19 +412,16 @@ def test_ceph_mon_app(status, config, model): assert app.is_subordinate is False -def test_ceph_mon_upgrade_plan_xena_to_yoga( - status, - config, - model, -): +def test_ceph_mon_upgrade_plan_xena_to_yoga(status, config, model, apps_machines): """Test when ceph version changes between os releases.""" target = OpenStackRelease("yoga") app = CephMon( "ceph-mon", - status["ceph-mon_xena"], + status["ceph_mon_pacific"], config["auxiliary_xena"], model, "ceph-mon", + apps_machines["ceph-mon"], ) upgrade_plan = app.generate_upgrade_plan(target) @@ -473,15 +488,17 @@ def test_ceph_mon_upgrade_plan_ussuri_to_victoria( status, config, model, + apps_machines, ): """Test when ceph version remains the same between os releases.""" target = OpenStackRelease("victoria") app = CephMon( "ceph-mon", - status["ceph-mon_ussuri"], + status["ceph_mon_octopus"], config["auxiliary_ussuri"], model, "ceph-mon", + apps_machines["ceph-mon"], ) upgrade_plan = app.generate_upgrade_plan(target) @@ -538,13 +555,14 @@ def test_ceph_mon_upgrade_plan_ussuri_to_victoria( assert upgrade_plan == expected_plan -def test_ovn_principal(status, config, model): +def test_ovn_principal(status, config, model, apps_machines): app = OvnPrincipal( "ovn-central", - status["ovn_central_ussuri_22"], + status["ovn_central_22"], config["auxiliary_ussuri"], model, "ovn-central", + apps_machines["ovn-central"], ) assert app.channel == "22.03/stable" assert app.os_origin == "distro" @@ -554,7 +572,7 @@ def test_ovn_principal(status, config, model): assert app.is_subordinate is False -def test_ovn_workload_ver_lower_than_22_principal(status, config, model): +def test_ovn_workload_ver_lower_than_22_principal(status, config, model, apps_machines): target = OpenStackRelease("victoria") exp_error_msg_ovn_upgrade = ( @@ -566,10 +584,11 @@ def test_ovn_workload_ver_lower_than_22_principal(status, config, model): app_ovn_central = OvnPrincipal( "ovn-central", - status["ovn_central_ussuri_20"], + status["ovn_central_20"], config["auxiliary_ussuri"], model, "ovn-central", + apps_machines["ovn-central"], ) with pytest.raises(ApplicationError, match=exp_error_msg_ovn_upgrade): @@ -577,8 +596,8 @@ def test_ovn_workload_ver_lower_than_22_principal(status, config, model): @pytest.mark.parametrize("channel", ["55.7", "19.03"]) -def test_ovn_no_compatible_os_release(status, config, model, channel): - ovn_central_status = status["ovn_central_ussuri_22"] +def test_ovn_no_compatible_os_release(status, config, model, channel, apps_machines): + ovn_central_status = status["ovn_central_22"] ovn_central_status.charm_channel = channel with pytest.raises(ApplicationError): OvnPrincipal( @@ -587,17 +606,19 @@ def test_ovn_no_compatible_os_release(status, config, model, channel): config["auxiliary_ussuri"], model, "ovn-central", + apps_machines["ovn-central"], ) -def test_ovn_principal_upgrade_plan(status, config, model): +def test_ovn_principal_upgrade_plan(status, config, model, apps_machines): target = OpenStackRelease("victoria") app = OvnPrincipal( "ovn-central", - status["ovn_central_ussuri_22"], + status["ovn_central_22"], config["auxiliary_ussuri"], model, "ovn-central", + apps_machines["ovn-central"], ) upgrade_plan = app.generate_upgrade_plan(target) @@ -651,15 +672,16 @@ def test_ovn_principal_upgrade_plan(status, config, model): assert upgrade_plan == expected_plan -def test_mysql_innodb_cluster_upgrade(status, config, model): +def test_mysql_innodb_cluster_upgrade(status, config, model, apps_machines): target = OpenStackRelease("victoria") # source is already configured to wallaby, so the plan halt with target victoria app = MysqlInnodbCluster( "mysql-innodb-cluster", - status["mysql-innodb-cluster"], + status["mysql_innodb_cluster"], config["auxiliary_ussuri"], model, "mysql-innodb-cluster", + apps_machines["mysql-innodb-cluster"], ) upgrade_plan = app.generate_upgrade_plan(target) expected_plan = ApplicationUpgradePlan( diff --git a/tests/unit/apps/test_auxiliary_subordinate.py b/tests/unit/apps/test_auxiliary_subordinate.py index 8c10d717..ae6f9b4f 100644 --- a/tests/unit/apps/test_auxiliary_subordinate.py +++ b/tests/unit/apps/test_auxiliary_subordinate.py @@ -57,11 +57,7 @@ def test_auxiliary_subordinate_upgrade_plan_to_victoria(apps, model): def test_ovn_subordinate(status, model): app = OvnSubordinate( - "ovn-chassis", - status["ovn_chassis_ussuri_22"], - {}, - model, - "ovn-chassis", + "ovn-chassis", status["ovn_chassis_focal_22"], {}, model, "ovn-chassis", {} ) assert app.channel == "22.03/stable" assert app.os_origin == "" @@ -83,10 +79,11 @@ def test_ovn_workload_ver_lower_than_22_subordinate(status, model): app_ovn_chassis = OvnSubordinate( "ovn-chassis", - status["ovn_chassis_ussuri_20"], + status["ovn_chassis_focal_20"], {}, model, "ovn-chassis", + {}, ) with pytest.raises(ApplicationError, match=exp_error_msg_ovn_upgrade): @@ -97,10 +94,11 @@ def test_ovn_subordinate_upgrade_plan(status, model): target = OpenStackRelease("victoria") app = OvnSubordinate( "ovn-chassis", - status["ovn_chassis_ussuri_22"], + status["ovn_chassis_focal_22"], {}, model, "ovn-chassis", + {}, ) upgrade_plan = app.generate_upgrade_plan(target) @@ -125,7 +123,7 @@ def test_ovn_subordinate_upgrade_plan_cant_upgrade_charm(status, model): # ovn chassis 22.03 is considered yoga. If it's not necessary to upgrade # the charm code, there is no steps to upgrade. target = OpenStackRelease("victoria") - app_status = status["ovn_chassis_ussuri_22"] + app_status = status["ovn_chassis_focal_22"] app_status.can_upgrade_to = "" app = OvnSubordinate( "ovn-chassis", @@ -133,6 +131,7 @@ def test_ovn_subordinate_upgrade_plan_cant_upgrade_charm(status, model): {}, model, "ovn-chassis", + {}, ) expected_plan = ApplicationUpgradePlan( @@ -149,10 +148,11 @@ def test_ceph_dashboard_upgrade_plan_ussuri_to_victoria(status, config, model): target = OpenStackRelease("victoria") app = AuxiliarySubordinateApplication( "ceph-dashboard", - status["ceph_dashboard_ussuri"], + status["ceph_dashboard_octopus"], config["auxiliary_ussuri"], model, "ceph-dashboard", + {}, ) upgrade_plan = app.generate_upgrade_plan(target) @@ -178,10 +178,11 @@ def test_ceph_dashboard_upgrade_plan_xena_to_yoga(status, config, model): target = OpenStackRelease("yoga") app = AuxiliarySubordinateApplication( "ceph-dashboard", - status["ceph_dashboard_xena"], + status["ceph_dashboard_pacific"], config["auxiliary_xena"], model, "ceph-dashboard", + {}, ) upgrade_plan = app.generate_upgrade_plan(target) diff --git a/tests/unit/apps/test_channel_based.py b/tests/unit/apps/test_channel_based.py index 0b0d9a6e..cd192f0a 100644 --- a/tests/unit/apps/test_channel_based.py +++ b/tests/unit/apps/test_channel_based.py @@ -22,71 +22,78 @@ from tests.unit.apps.utils import add_steps -def test_application_versionless(status, config, model): +def test_application_versionless(status, config, model, apps_machines): app = ChannelBasedApplication( "glance-simplestreams-sync", - status["glance_simplestreams_sync_ussuri"], + status["glance_simplestreams_sync_focal_ussuri"], config["openstack_ussuri"], model, "glance-simplestreams-sync", + apps_machines["glance-simplestreams-sync"], ) assert app.current_os_release == "ussuri" assert app.is_versionless is True -def test_application_gnocchi_ussuri(status, config, model): +def test_application_gnocchi_ussuri(status, config, model, apps_machines): app = ChannelBasedApplication( "gnocchi", - status["gnocchi_ussuri"], + status["gnocchi_focal_ussuri"], config["openstack_ussuri"], model, "gnocchi", + apps_machines["gnocchi"], ) assert app.current_os_release == "ussuri" assert app.is_versionless is False -def test_application_gnocchi_xena(status, config, model): +def test_application_gnocchi_xena(status, config, model, apps_machines): # workload version is the same for xena and yoga, but current_os_release # is based on the channel. app = ChannelBasedApplication( "gnocchi", - status["gnocchi_xena"], + status["gnocchi_focal_xena"], config["openstack_xena"], model, "gnocchi", + apps_machines["gnocchi"], ) assert app.current_os_release == "xena" assert app.is_versionless is False -def test_application_designate_bind_ussuri(status, config, model): +def test_application_designate_bind_ussuri(status, config, model, apps_machines): # workload version is the same from ussuri to yoga, but current_os_release # is based on the channel. app_config = config["openstack_ussuri"] app_config["action-managed-upgrade"] = {"value": False} app = ChannelBasedApplication( "designate-bind", - status["designate_bind_ussuri"], + status["designate_bind_focal_ussuri"], app_config, model, "designate-bind", + apps_machines["designate-bind"], ) assert app.current_os_release == "ussuri" assert app.is_versionless is False -def test_application_versionless_upgrade_plan_ussuri_to_victoria(status, config, model): +def test_application_versionless_upgrade_plan_ussuri_to_victoria( + status, config, model, apps_machines +): target = OpenStackRelease("victoria") app_config = config["openstack_ussuri"] # Does not have action-managed-upgrade app_config.pop("action-managed-upgrade") app = ChannelBasedApplication( "glance-simplestreams-sync", - status["glance_simplestreams_sync_ussuri"], + status["glance_simplestreams_sync_focal_ussuri"], app_config, model, "glance-simplestreams-sync", + apps_machines["glance-simplestreams-sync"], ) upgrade_plan = app.generate_upgrade_plan(target) @@ -137,17 +144,18 @@ def test_application_versionless_upgrade_plan_ussuri_to_victoria(status, config, assert upgrade_plan == expected_plan -def test_application_gnocchi_upgrade_plan_ussuri_to_victoria(status, config, model): +def test_application_gnocchi_upgrade_plan_ussuri_to_victoria(status, config, model, apps_machines): # Gnocchi from ussuri to victoria upgrade the workload version from 4.3.4 to 4.4.0. target = OpenStackRelease("victoria") app_config = config["openstack_ussuri"] app_config["action-managed-upgrade"] = {"value": False} app = ChannelBasedApplication( "gnocchi", - status["gnocchi_ussuri"], + status["gnocchi_focal_ussuri"], app_config, model, "gnocchi", + apps_machines["gnocchi"], ) upgrade_plan = app.generate_upgrade_plan(target) @@ -208,16 +216,19 @@ def test_application_gnocchi_upgrade_plan_ussuri_to_victoria(status, config, mod assert upgrade_plan == expected_plan -def test_application_designate_bind_upgrade_plan_ussuri_to_victoria(status, config, model): +def test_application_designate_bind_upgrade_plan_ussuri_to_victoria( + status, config, model, apps_machines +): target = OpenStackRelease("victoria") app_config = config["openstack_ussuri"] app_config["action-managed-upgrade"] = {"value": False} app = ChannelBasedApplication( "designate-bind", - status["designate_bind_ussuri"], + status["designate_bind_focal_ussuri"], app_config, model, "designate-bind", + apps_machines["designate-bind"], ) upgrade_plan = app.generate_upgrade_plan(target) diff --git a/tests/unit/apps/test_core.py b/tests/unit/apps/test_core.py index e67723b4..88a01648 100644 --- a/tests/unit/apps/test_core.py +++ b/tests/unit/apps/test_core.py @@ -33,15 +33,36 @@ from tests.unit.apps.utils import add_steps -def test_application_eq(status, config, model): +def test_application_eq(status, config, model, apps_machines): """Name of the app is used as comparison between Applications objects.""" - status_keystone_1 = status["keystone_ussuri"] + status_keystone_1 = status["keystone_focal_ussuri"] config_keystone_1 = config["openstack_ussuri"] - status_keystone_2 = status["keystone_wallaby"] + status_keystone_2 = status["keystone_focal_wallaby"] config_keystone_2 = config["openstack_wallaby"] - keystone_1 = Keystone("keystone", status_keystone_1, config_keystone_1, model, "keystone") - keystone_2 = Keystone("keystone", status_keystone_2, config_keystone_2, model, "keystone") - keystone_3 = Keystone("keystone_foo", status_keystone_1, config_keystone_1, model, "keystone") + keystone_1 = Keystone( + "keystone", + status_keystone_1, + config_keystone_1, + model, + "keystone", + apps_machines["keystone"], + ) + keystone_2 = Keystone( + "keystone", + status_keystone_2, + config_keystone_2, + model, + "keystone", + apps_machines["keystone"], + ) + keystone_3 = Keystone( + "keystone_foo", + status_keystone_1, + config_keystone_1, + model, + "keystone", + apps_machines["keystone"], + ) # keystone_1 is equal to keystone_2 because they have the same name # even if they have different status and config. @@ -92,9 +113,9 @@ def assert_application( assert app.is_valid_track(app.channel) == exp_is_valid_track -def test_application_ussuri(status, config, units, model): +def test_application_ussuri(status, config, units, model, apps_machines): target = OpenStackRelease("victoria") - app_status = status["keystone_ussuri"] + app_status = status["keystone_focal_ussuri"] app_config = config["openstack_ussuri"] exp_is_from_charm_store = False exp_os_origin = "distro" @@ -110,7 +131,9 @@ def test_application_ussuri(status, config, units, model): exp_is_subordinate = False exp_is_valid_track = True - app = Keystone("my_keystone", app_status, app_config, model, "keystone") + app = Keystone( + "my_keystone", app_status, app_config, model, "keystone", apps_machines["keystone"] + ) assert app.wait_for_model is True assert_application( app, @@ -136,25 +159,31 @@ def test_application_ussuri(status, config, units, model): ) -def test_application_different_wl(status, config, model): +def test_application_different_wl(status, config, model, apps_machines): """Different OpenStack Version on units if workload version is different.""" exp_error_msg = ( "Units of application my_keystone are running mismatched OpenStack versions: " r"'ussuri': \['keystone\/0', 'keystone\/1'\], 'victoria': \['keystone\/2'\]. " "This is not currently handled." ) - app_status = status["keystone_ussuri_victoria"] + app_status = status["keystone_focal_ussuri"] + app_status.units["keystone/2"].workload_version = "18.1.0" app_config = config["openstack_ussuri"] - app = Keystone("my_keystone", app_status, app_config, model, "keystone") + app = Keystone( + "my_keystone", app_status, app_config, model, "keystone", apps_machines["keystone"] + ) with pytest.raises(MismatchedOpenStackVersions, match=exp_error_msg): app.current_os_release -def test_application_cs(status, config, units, model): +def test_application_cs(status, config, units, model, apps_machines): """Test when application is from charm store.""" target = OpenStackRelease("victoria") - app_status = status["keystone_ussuri_cs"] + + app_status = status["keystone_focal_ussuri"] + app_status.charm = "cs:amd64/focal/keystone-638" + app_config = config["openstack_ussuri"] exp_os_origin = "distro" exp_units = units["units_ussuri"] @@ -170,7 +199,9 @@ def test_application_cs(status, config, units, model): exp_is_subordinate = False exp_is_valid_track = True - app = Keystone("my_keystone", app_status, app_config, model, "keystone") + app = Keystone( + "my_keystone", app_status, app_config, model, "keystone", apps_machines["keystone"] + ) assert_application( app, "my_keystone", @@ -195,12 +226,12 @@ def test_application_cs(status, config, units, model): ) -def test_application_wallaby(status, config, units, model): +def test_application_wallaby(status, config, units, model, apps_machines): target = OpenStackRelease("xena") exp_units = units["units_wallaby"] exp_is_from_charm_store = False app_config = config["openstack_wallaby"] - app_status = status["keystone_wallaby"] + app_status = status["keystone_focal_wallaby"] exp_os_origin = "cloud:focal-wallaby" exp_channel = app_status.charm_channel exp_series = app_status.series @@ -213,7 +244,9 @@ def test_application_wallaby(status, config, units, model): exp_is_subordinate = False exp_is_valid_track = True - app = Keystone("my_keystone", app_status, app_config, model, "keystone") + app = Keystone( + "my_keystone", app_status, app_config, model, "keystone", apps_machines["keystone"] + ) assert_application( app, "my_keystone", @@ -238,32 +271,34 @@ def test_application_wallaby(status, config, units, model): ) -def test_application_no_origin_config(status, model): +def test_application_no_origin_config(status, model, apps_machines): app = Keystone( "my_keystone", - status["keystone_ussuri"], + status["keystone_focal_ussuri"], {}, model, "keystone", + apps_machines["keystone"], ) assert app.os_origin == "" assert app.apt_source_codename is None -def test_application_empty_origin_config(status, model): +def test_application_empty_origin_config(status, model, apps_machines): app = Keystone( "my_keystone", - status["keystone_ussuri"], + status["keystone_focal_ussuri"], {"source": {"value": ""}}, model, "keystone", + apps_machines["keystone"], ) assert app.apt_source_codename is None -def test_application_unexpected_channel(status, config, model): +def test_application_unexpected_channel(status, config, model, apps_machines): target = OpenStackRelease("xena") - app_status = status["keystone_wallaby"] + app_status = status["keystone_focal_wallaby"] # channel is set to a previous OpenStack release app_status.charm_channel = "ussuri/stable" app = Keystone( @@ -272,6 +307,7 @@ def test_application_unexpected_channel(status, config, model): config["openstack_wallaby"], model, "keystone", + apps_machines["keystone"], ) with pytest.raises(ApplicationError): app.generate_upgrade_plan(target) @@ -281,53 +317,75 @@ def test_application_unexpected_channel(status, config, model): "source_value", ["ppa:myteam/ppa", "cloud:xenial-proposed/ocata", "http://my.archive.com/ubuntu main"], ) -def test_application_unknown_source(status, model, source_value): +def test_application_unknown_source(status, model, source_value, apps_machines): app = Keystone( "my_keystone", - status["keystone_ussuri"], + status["keystone_focal_ussuri"], {"source": {"value": source_value}}, model, "keystone", + apps_machines["keystone"], ) with pytest.raises(ApplicationError): app.apt_source_codename @pytest.mark.asyncio -async def test_application_check_upgrade(status, config, model): +async def test_application_check_upgrade(status, config, model, apps_machines): target = OpenStackRelease("victoria") - app_status = status["keystone_ussuri"] + app_status = status["keystone_focal_ussuri"] app_config = config["openstack_ussuri"] # workload version changed from ussuri to victoria mock_status = AsyncMock() - mock_status.return_value.applications = {"my_keystone": status["keystone_victoria"]} + mock_status.return_value.applications = {"my_keystone": status["keystone_focal_victoria"]} model.get_status = mock_status - app = Keystone("my_keystone", app_status, app_config, model, "keystone") + app = Keystone( + "my_keystone", + app_status, + app_config, + model, + "keystone", + machines=apps_machines["keystone"], + ) await app._check_upgrade(target) @pytest.mark.asyncio -async def test_application_check_upgrade_fail(status, config, model): +async def test_application_check_upgrade_fail(status, config, model, apps_machines): exp_error_msg = "Cannot upgrade units 'keystone/0, keystone/1, keystone/2' to victoria." target = OpenStackRelease("victoria") - app_status = status["keystone_ussuri"] + app_status = status["keystone_focal_ussuri"] app_config = config["openstack_ussuri"] # workload version didn't change from ussuri to victoria mock_status = AsyncMock() mock_status.return_value.applications = {"my_keystone": app_status} model.get_status = mock_status - app = Keystone("my_keystone", app_status, app_config, model, "keystone") + app = Keystone( + "my_keystone", + app_status, + app_config, + model, + "keystone", + machines=apps_machines["keystone"], + ) with pytest.raises(ApplicationError, match=exp_error_msg): await app._check_upgrade(target) -def test_upgrade_plan_ussuri_to_victoria(status, config, model): +def test_upgrade_plan_ussuri_to_victoria(status, config, model, apps_machines): target = OpenStackRelease("victoria") - app_status = status["keystone_ussuri"] + app_status = status["keystone_focal_ussuri"] app_config = config["openstack_ussuri"] - app = Keystone("my_keystone", app_status, app_config, model, "keystone") + app = Keystone( + "my_keystone", + app_status, + app_config, + model, + "keystone", + machines=apps_machines["keystone"], + ) upgrade_plan = app.generate_upgrade_plan(target) expected_plan = ApplicationUpgradePlan( description=f"Upgrade plan for '{app.name}' to {target}" @@ -387,11 +445,16 @@ def test_upgrade_plan_ussuri_to_victoria(status, config, model): assert upgrade_plan == expected_plan -def test_upgrade_plan_ussuri_to_victoria_ch_migration(status, config, model): +def test_upgrade_plan_ussuri_to_victoria_ch_migration(status, config, model, apps_machines): target = OpenStackRelease("victoria") - app_status = status["keystone_ussuri_cs"] + + app_status = status["keystone_focal_ussuri"] + app_status.charm = "cs:amd64/focal/keystone-638" + app_config = config["openstack_ussuri"] - app = Keystone("my_keystone", app_status, app_config, model, "keystone") + app = Keystone( + "my_keystone", app_status, app_config, model, "keystone", apps_machines["keystone"] + ) upgrade_plan = app.generate_upgrade_plan(target) expected_plan = ApplicationUpgradePlan( description=f"Upgrade plan for '{app.name}' to {target}" @@ -451,13 +514,15 @@ def test_upgrade_plan_ussuri_to_victoria_ch_migration(status, config, model): assert upgrade_plan == expected_plan -def test_upgrade_plan_channel_on_next_os_release(status, config, model): +def test_upgrade_plan_channel_on_next_os_release(status, config, model, apps_machines): target = OpenStackRelease("victoria") - app_status = status["keystone_ussuri"] + app_status = status["keystone_focal_ussuri"] app_config = config["openstack_ussuri"] # channel it's already on next OpenStack release app_status.charm_channel = "victoria/stable" - app = Keystone("my_keystone", app_status, app_config, model, "keystone") + app = Keystone( + "my_keystone", app_status, app_config, model, "keystone", apps_machines["keystone"] + ) upgrade_plan = app.generate_upgrade_plan(target) expected_plan = ApplicationUpgradePlan( @@ -509,13 +574,17 @@ def test_upgrade_plan_channel_on_next_os_release(status, config, model): assert upgrade_plan == expected_plan -def test_upgrade_plan_origin_already_on_next_openstack_release(status, config, model): +def test_upgrade_plan_origin_already_on_next_openstack_release( + status, config, model, apps_machines +): target = OpenStackRelease("victoria") - app_status = status["keystone_ussuri"] + app_status = status["keystone_focal_ussuri"] app_config = config["openstack_ussuri"] # openstack-origin already configured for next OpenStack release app_config["openstack-origin"]["value"] = "cloud:focal-victoria" - app = Keystone("my_keystone", app_status, app_config, model, "keystone") + app = Keystone( + "my_keystone", app_status, app_config, model, "keystone", apps_machines["keystone"] + ) upgrade_plan = app.generate_upgrade_plan(target) expected_plan = ApplicationUpgradePlan( description=f"Upgrade plan for '{app.name}' to {target}" @@ -565,31 +634,31 @@ def test_upgrade_plan_origin_already_on_next_openstack_release(status, config, m assert upgrade_plan == expected_plan -def test_upgrade_plan_application_already_upgraded(status, config, model): +def test_upgrade_plan_application_already_upgraded(status, config, model, apps_machines): exp_error_msg = ( "Application 'my_keystone' already configured for release equal or greater " "than victoria. Ignoring." ) target = OpenStackRelease("victoria") - app_status = status["keystone_wallaby"] + app_status = status["keystone_focal_wallaby"] app_config = config["openstack_wallaby"] - app = Keystone("my_keystone", app_status, app_config, model, "keystone") + app = Keystone( + "my_keystone", app_status, app_config, model, "keystone", apps_machines["keystone"] + ) # victoria is lesser than wallaby, so application should not generate a plan. with pytest.raises(HaltUpgradePlanGeneration, match=exp_error_msg): app.generate_upgrade_plan(target) -def test_upgrade_plan_application_already_disable_action_managed(status, config, model): +def test_upgrade_plan_application_already_disable_action_managed( + status, config, model, apps_machines +): target = OpenStackRelease("victoria") - app_status = status["keystone_ussuri"] + app_status = status["keystone_focal_ussuri"] app_config = config["openstack_ussuri"] app_config["action-managed-upgrade"]["value"] = False app = Keystone( - "my_keystone", - app_status, - app_config, - model, - "keystone", + "my_keystone", app_status, app_config, model, "keystone", apps_machines["keystone"] ) upgrade_plan = app.generate_upgrade_plan(target) expected_plan = ApplicationUpgradePlan( diff --git a/tests/unit/apps/test_factory.py b/tests/unit/apps/test_factory.py index 194ceb1c..1f97f614 100644 --- a/tests/unit/apps/test_factory.py +++ b/tests/unit/apps/test_factory.py @@ -25,6 +25,7 @@ def test_app_factory_not_supported_openstack_charm(mock_is_charm_supported): config=MagicMock(), model=MagicMock(), charm=charm, + machines=MagicMock(), ) assert my_app is None mock_is_charm_supported.assert_called_once_with(charm) @@ -40,7 +41,7 @@ def __init__(self, *_, **__): pass assert charm in factory.AppFactory.charms - foo = factory.AppFactory.create("my-foo", MagicMock(), {}, MagicMock(), charm) + foo = factory.AppFactory.create("my-foo", MagicMock(), {}, MagicMock(), charm, MagicMock()) mock_is_charm_supported.assert_called_once_with(charm) assert foo is not None assert isinstance(foo, Foo) diff --git a/tests/unit/apps/test_subordinate.py b/tests/unit/apps/test_subordinate.py index 8a44c9f5..5278c8c7 100644 --- a/tests/unit/apps/test_subordinate.py +++ b/tests/unit/apps/test_subordinate.py @@ -26,8 +26,8 @@ def test_post_init(status, model): - app_status = status["keystone-ldap"] - app = SubordinateApplication("my_keystone_ldap", app_status, {}, model, "keystone-ldap") + app_status = status["keystone_ldap_focal_ussuri"] + app = SubordinateApplication("my_keystone_ldap", app_status, {}, model, "keystone-ldap", {}) assert app.channel == "ussuri/stable" assert app.charm_origin == "ch" assert app.os_origin == "" @@ -35,15 +35,15 @@ def test_post_init(status, model): def test_current_os_release(status, model): - app_status = status["keystone-ldap"] - app = SubordinateApplication("my_keystone_ldap", app_status, {}, model, "keystone-ldap") + app_status = status["keystone_ldap_focal_ussuri"] + app = SubordinateApplication("my_keystone_ldap", app_status, {}, model, "keystone-ldap", {}) assert app.current_os_release == OpenStackRelease("ussuri") def test_generate_upgrade_plan(status, model): target = OpenStackRelease("victoria") - app_status = status["keystone-ldap"] - app = SubordinateApplication("my_keystone_ldap", app_status, {}, model, "keystone-ldap") + app_status = status["keystone_ldap_focal_ussuri"] + app = SubordinateApplication("my_keystone_ldap", app_status, {}, model, "keystone-ldap", {}) upgrade_plan = app.generate_upgrade_plan(target) expected_plan = ApplicationUpgradePlan( @@ -78,9 +78,9 @@ def test_generate_upgrade_plan(status, model): ], ) def test_channel_valid(status, model, channel): - app_status = status["keystone-ldap"] + app_status = status["keystone_ldap_focal_ussuri"] app_status.charm_channel = channel - app = SubordinateApplication("my_keystone_ldap", app_status, {}, model, "keystone-ldap") + app = SubordinateApplication("my_keystone_ldap", app_status, {}, model, "keystone-ldap", {}) assert app.channel == channel @@ -95,10 +95,10 @@ def test_channel_valid(status, model, channel): ], ) def test_channel_setter_invalid(status, model, channel): - app_status = status["keystone-ldap"] + app_status = status["keystone_ldap_focal_ussuri"] app_status.charm_channel = channel with pytest.raises(ApplicationError): - SubordinateApplication("my_keystone_ldap", app_status, {}, model, "keystone-ldap") + SubordinateApplication("my_keystone_ldap", app_status, {}, model, "keystone-ldap", {}) @pytest.mark.parametrize( @@ -111,10 +111,10 @@ def test_channel_setter_invalid(status, model, channel): ) def test_generate_plan_ch_migration(status, model, channel): target = OpenStackRelease("wallaby") - app_status = status["keystone-ldap-cs"] + app_status = status["keystone_ldap_focal_ussuri"] app_status.charm = "cs:amd64/focal/keystone-ldap-437" app_status.charm_channel = f"ussuri/{channel}" - app = SubordinateApplication("my_keystone_ldap", app_status, {}, model, "keystone-ldap") + app = SubordinateApplication("my_keystone_ldap", app_status, {}, model, "keystone-ldap", {}) upgrade_plan = app.generate_upgrade_plan(target) @@ -148,9 +148,9 @@ def test_generate_plan_ch_migration(status, model, channel): ], ) def test_generate_plan_from_to(status, model, from_os, to_os): - app_status = status["keystone-ldap"] + app_status = status["keystone_ldap_focal_ussuri"] app_status.charm_channel = f"{from_os}/stable" - app = SubordinateApplication("my_keystone_ldap", app_status, {}, model, "keystone-ldap") + app = SubordinateApplication("my_keystone_ldap", app_status, {}, model, "keystone-ldap", {}) upgrade_plan = app.generate_upgrade_plan(OpenStackRelease(to_os)) @@ -183,9 +183,9 @@ def test_generate_plan_from_to(status, model, from_os, to_os): ], ) def test_generate_plan_in_same_version(status, model, from_to): - app_status = status["keystone-ldap"] + app_status = status["keystone_ldap_focal_ussuri"] app_status.charm_channel = f"{from_to}/stable" - app = SubordinateApplication("my_keystone_ldap", app_status, {}, model, "keystone-ldap") + app = SubordinateApplication("my_keystone_ldap", app_status, {}, model, "keystone-ldap", {}) upgrade_plan = app.generate_upgrade_plan(OpenStackRelease(from_to)) expected_plan = ApplicationUpgradePlan( diff --git a/tests/unit/conftest.py b/tests/unit/conftest.py index e0a3f96a..23c645bc 100644 --- a/tests/unit/conftest.py +++ b/tests/unit/conftest.py @@ -13,6 +13,7 @@ # limitations under the License. from collections import OrderedDict +from itertools import zip_longest from pathlib import Path from unittest.mock import AsyncMock, MagicMock, PropertyMock, patch @@ -26,379 +27,437 @@ from cou.apps.core import Keystone from cou.apps.subordinate import SubordinateApplication from cou.commands import CLIargs +from cou.utils.juju_utils import Machine from cou.utils.openstack import OpenStackRelease +STANDARD_AZS = ["zone-1", "zone-2", "zone-3"] -def generate_unit(workload_version, machine): +KEYSTONE_UNITS = ["keystone/0", "keystone/1", "keystone/2"] +KEYSTONE_MACHINES = ["0/lxd/12", "1/lxd/12", "2/lxd/13"] +KEYSTONE_WORKLOADS = { + "ussuri": "17.0.1", + "victoria": "18.1.0", + "wallaby": "19.1.0", +} + +CINDER_UNITS = ["cinder/0", "cinder/1", "cinder/2"] +CINDER_MACHINES = ["0/lxd/5", "1/lxd/5", "2/lxd/5"] +CINDER_WORKLOADS = {"ussuri": "16.4.2"} + +NOVA_UNITS = ["nova-compute/0", "nova-compute/1", "nova-compute/2"] +NOVA_MACHINES = ["0", "1", "2"] +NOVA_WORKLOADS = {"ussuri": "21.0.0"} + +RMQ_UNITS = ["rabbitmq-server/0"] +RMQ_MACHINES = ["0/lxd/19"] +RMQ_WORKLOADS = {"3.8": "3.8"} + +CEPH_MON_UNITS = ["ceph-mon/0"] +CEPH_MON_MACHINES = ["6"] + +CEPH_OSD_UNITS = ["ceph-osd/0"] +CEPH_OSD_MACHINES = ["7"] + +CEPH_WORKLOADS = {"octopus": "15.2.0", "pacific": "16.2.0"} + +OVN_UNITS = ["ovn-central/0"] +OVN_MACHINES = ["0/lxd/7"] +OVN_WORKLOADS = {"22.03": "22.03.2", "20.03": "20.03.2"} + +MYSQL_UNITS = ["mysql/0"] +MYSQL_MACHINES = ["0/lxd/7"] +MYSQL_WORKLOADS = {"8.0": "8.0"} + +GLANCE_SIMPLE_UNITS = ["glance-simplestreams-sync/0"] +GLANCE_SIMPLE_MACHINES = ["4/lxd/5"] + +DESIGNATE_UNITS = ["designate-bind/0", "designate-bind/1"] +DESIGNATE_MACHINES = ["1/lxd/6", "2/lxd/6"] +DESIGNATE_WORKLOADS = {"ussuri": "9.16.1"} + +GNOCCHI_UNITS = ["gnocchi/0", "gnocchi/1", "gnocchi/2"] +GNOCCHI_MACHINES = ["3/lxd/6", "4/lxd/6", "5/lxd/5"] +GNOCCHI_WORKLOADS = {"ussuri": "4.3.4", "xena": "4.4.1"} + +MY_APP_UNITS = ["my-app/0"] +MY_APP_MACHINES = ["0/lxd/11"] + + +def _generate_unit(workload_version, machine): unit = MagicMock(spec_set=UnitStatus()) unit.workload_version = workload_version unit.machine = machine return unit +def _generate_units(units_machines_workloads): + unit = MagicMock(spec_set=UnitStatus()) + + ordered_units = OrderedDict() + for unit_machine_workload in units_machines_workloads: + unit, machine, workload = unit_machine_workload + ordered_units[unit] = _generate_unit(workload, machine) + + return ordered_units + + +@pytest.fixture +def apps_machines(): + return { + **_generate_apps_machines("keystone", KEYSTONE_MACHINES, STANDARD_AZS), + **_generate_apps_machines("cinder", CINDER_MACHINES, STANDARD_AZS), + **_generate_apps_machines("nova-compute", NOVA_MACHINES, STANDARD_AZS), + **_generate_apps_machines("rmq", RMQ_MACHINES, STANDARD_AZS), + **_generate_apps_machines("ceph-mon", CEPH_MON_MACHINES, STANDARD_AZS), + **_generate_apps_machines("ovn-central", OVN_MACHINES, STANDARD_AZS), + **_generate_apps_machines("mysql-innodb-cluster", MYSQL_MACHINES, STANDARD_AZS), + **_generate_apps_machines( + "glance-simplestreams-sync", GLANCE_SIMPLE_MACHINES, STANDARD_AZS + ), + **_generate_apps_machines("gnocchi", GNOCCHI_MACHINES, STANDARD_AZS), + **_generate_apps_machines("designate-bind", DESIGNATE_MACHINES, STANDARD_AZS), + **_generate_apps_machines("ceph-osd", CEPH_OSD_MACHINES, STANDARD_AZS), + **_generate_apps_machines("my-app", MY_APP_MACHINES, STANDARD_AZS), + } + + +def _generate_apps_machines(charm, machines, azs): + machines_azs = zip(machines, azs) + return {charm: {machine_id: Machine(machine_id, (), az) for machine_id, az in machines_azs}} + + @pytest.fixture def status(): - mock_keystone_ussuri = MagicMock(spec_set=ApplicationStatus()) - mock_keystone_ussuri.series = "focal" - mock_keystone_ussuri.charm_channel = "ussuri/stable" - mock_keystone_ussuri.charm = "ch:amd64/focal/keystone-638" - mock_keystone_ussuri.subordinate_to = [] - mock_keystone_ussuri.units = OrderedDict( - [ - ("keystone/0", generate_unit("17.0.1", "0/lxd/12")), - ("keystone/1", generate_unit("17.0.1", "1/lxd/12")), - ("keystone/2", generate_unit("17.0.1", "2/lxd/13")), - ] - ) + return { + **generate_keystone_status(), + **generate_cinder_status(), + **generate_nova_status(), + **generate_rmq_status(), + **generate_ceph_mon_status(), + **generate_ceph_osd_status(), + **generate_ovn_central_status(), + **generate_mysql_innodb_cluster_status(), + **generate_glance_simplestreams_sync_status(), + **generate_gnocchi_status(), + **generate_ovn_chassis_status(), + **generate_ceph_dashboard_status(), + **generate_keystone_ldap_status(), + **generate_designate_bind_status(), + **generate_mysql_router_status(), + **generate_my_app(), + } - mock_keystone_bionic_ussuri = MagicMock(spec_set=ApplicationStatus()) - mock_keystone_bionic_ussuri.series = "bionic" - mock_keystone_bionic_ussuri.charm_channel = "ussuri/stable" - mock_keystone_bionic_ussuri.charm = "ch:amd64/bionic/keystone-638" - mock_keystone_bionic_ussuri.subordinate_to = [] - mock_keystone_bionic_ussuri.units = OrderedDict( - [ - ("keystone/0", generate_unit("17.0.1", "0/lxd/12")), - ("keystone/1", generate_unit("17.0.1", "1/lxd/12")), - ("keystone/2", generate_unit("17.0.1", "2/lxd/13")), - ] - ) - mock_cinder_ussuri = MagicMock(spec_set=ApplicationStatus()) - mock_cinder_ussuri.series = "focal" - mock_cinder_ussuri.charm_channel = "ussuri/stable" - mock_cinder_ussuri.charm = "ch:amd64/focal/cinder-633" - mock_cinder_ussuri.subordinate_to = [] - mock_cinder_ussuri.units = OrderedDict( - [ - ("cinder/0", generate_unit("16.4.2", "0/lxd/5")), - ("cinder/1", generate_unit("16.4.2", "1/lxd/5")), - ("cinder/2", generate_unit("16.4.2", "2/lxd/5")), - ] +def generate_keystone_status(): + mock_keystone_focal_ussuri = _generate_status( + "focal", + "ussuri/stable", + "ch:amd64/focal/keystone-638", + [], + KEYSTONE_UNITS, + KEYSTONE_MACHINES, + KEYSTONE_WORKLOADS["ussuri"], ) - mock_cinder_on_nova = MagicMock(spec_set=ApplicationStatus()) - mock_cinder_on_nova.series = "focal" - mock_cinder_on_nova.charm_channel = "ussuri/stable" - mock_cinder_on_nova.charm = "ch:amd64/focal/cinder-633" - mock_cinder_on_nova.subordinate_to = [] - mock_cinder_on_nova.units = OrderedDict( - [ - ("cinder/0", generate_unit("16.4.2", "0")), - ("cinder/1", generate_unit("16.4.2", "1")), - ("cinder/2", generate_unit("16.4.2", "2")), - ] + mock_keystone_focal_victoria = _generate_status( + "focal", + "wallaby/stable", + "ch:amd64/focal/keystone-638", + [], + KEYSTONE_UNITS, + KEYSTONE_MACHINES, + KEYSTONE_WORKLOADS["victoria"], ) - mock_keystone_ussuri_cs = MagicMock(spec_set=ApplicationStatus()) - mock_keystone_ussuri_cs.series = "focal" - mock_keystone_ussuri_cs.charm_channel = "stable" - mock_keystone_ussuri_cs.charm = "cs:amd64/focal/keystone-638" - mock_keystone_ussuri_cs.subordinate_to = [] - mock_keystone_ussuri_cs.units = OrderedDict( - [ - ("keystone/0", generate_unit("17.0.1", "0/lxd/12")), - ("keystone/1", generate_unit("17.0.1", "1/lxd/12")), - ("keystone/2", generate_unit("17.0.1", "2/lxd/13")), - ] + mock_keystone_focal_wallaby = _generate_status( + "focal", + "wallaby/stable", + "ch:amd64/focal/keystone-638", + [], + KEYSTONE_UNITS, + KEYSTONE_MACHINES, + KEYSTONE_WORKLOADS["wallaby"], ) - mock_keystone_victoria = MagicMock(spec_set=ApplicationStatus()) - mock_keystone_victoria.series = "focal" - mock_keystone_victoria.charm_channel = "wallaby/stable" - mock_keystone_victoria.charm = "ch:amd64/focal/keystone-638" - mock_keystone_victoria.subordinate_to = [] - mock_keystone_victoria.units = OrderedDict( - [ - ("keystone/0", generate_unit("18.1.0", "0/lxd/12")), - ("keystone/1", generate_unit("18.1.0", "1/lxd/12")), - ("keystone/2", generate_unit("18.1.0", "2/lxd/13")), - ] - ) + return { + "keystone_focal_ussuri": mock_keystone_focal_ussuri, + "keystone_focal_victoria": mock_keystone_focal_victoria, + "keystone_focal_wallaby": mock_keystone_focal_wallaby, + } - mock_keystone_ussuri_victoria = MagicMock(spec_set=ApplicationStatus()) - mock_keystone_ussuri_victoria.series = "focal" - mock_keystone_ussuri_victoria.charm_channel = "victoria/stable" - mock_keystone_ussuri_victoria.charm = "ch:amd64/focal/keystone-638" - mock_keystone_ussuri_victoria.subordinate_to = [] - mock_keystone_ussuri_victoria.units = OrderedDict( - [ - ("keystone/0", generate_unit("17.0.1", "0/lxd/12")), - ("keystone/1", generate_unit("17.0.1", "1/lxd/12")), - ("keystone/2", generate_unit("18.1.0", "2/lxd/13")), - ] - ) - mock_keystone_wallaby = MagicMock(spec_set=ApplicationStatus()) - mock_keystone_wallaby.series = "focal" - mock_keystone_wallaby.charm_channel = "wallaby/stable" - mock_keystone_wallaby.charm = "ch:amd64/focal/keystone-638" - mock_keystone_wallaby.subordinate_to = [] - mock_keystone_wallaby.units = OrderedDict( - [ - ("keystone/0", generate_unit("19.1.0", "0/lxd/12")), - ("keystone/1", generate_unit("19.1.0", "1/lxd/12")), - ("keystone/2", generate_unit("19.1.0", "2/lxd/13")), - ] +def generate_cinder_status(): + mock_cinder_focal_ussuri = _generate_status( + "focal", + "ussuri/stable", + "ch:amd64/focal/cinder-633", + [], + CINDER_UNITS, + CINDER_MACHINES, + CINDER_WORKLOADS["ussuri"], + ) + return {"cinder_focal_ussuri": mock_cinder_focal_ussuri} + + +def generate_nova_status(): + mock_nova_focal_ussuri = _generate_status( + "focal", + "ussuri/stable", + "ch:amd64/focal/nova-compute-638", + [], + NOVA_UNITS, + NOVA_MACHINES, + NOVA_WORKLOADS["ussuri"], + ) + return {"nova_focal_ussuri": mock_nova_focal_ussuri} + + +def generate_rmq_status(): + mock_rmq = _generate_status( + "focal", + "3.8/stable", + "ch:amd64/focal/rabbitmq-server-638", + [], + RMQ_UNITS, + RMQ_MACHINES, + RMQ_WORKLOADS["3.8"], + ) + mock_rmq_unknown = _generate_status( + "focal", + "80.5/stable", + "ch:amd64/focal/rabbitmq-server-638", + [], + RMQ_UNITS, + RMQ_MACHINES, + "80.5", + ) + + return {"rabbitmq_server": mock_rmq, "unknown_rabbitmq_server": mock_rmq_unknown} + + +def generate_ceph_mon_status(): + mock_ceph_mon_octopus = _generate_status( + "focal", + "octopus/stable", + "ch:amd64/focal/ceph-mon-178", + [], + CEPH_MON_UNITS, + CEPH_MON_MACHINES, + CEPH_WORKLOADS["octopus"], + ) + mock_ceph_mon_pacific = _generate_status( + "focal", + "pacific/stable", + "ch:amd64/focal/ceph-mon-178", + [], + CEPH_MON_UNITS, + CEPH_MON_MACHINES, + CEPH_WORKLOADS["pacific"], + ) + return {"ceph_mon_octopus": mock_ceph_mon_octopus, "ceph_mon_pacific": mock_ceph_mon_pacific} + + +def generate_ceph_osd_status(): + mock_ceph_osd_octopus = _generate_status( + "focal", + "octopus/stable", + "ch:amd64/focal/ceph-osd-177", + [], + CEPH_OSD_UNITS, + CEPH_OSD_MACHINES, + CEPH_WORKLOADS["octopus"], + ) + return {"ceph_osd_octopus": mock_ceph_osd_octopus} + + +def generate_ovn_central_status(): + mock_ovn_central_20 = _generate_status( + "focal", + "20.03/stable", + "ch:amd64/focal/ovn-central-178", + [], + OVN_UNITS, + OVN_MACHINES, + OVN_WORKLOADS["20.03"], + ) + mock_ovn_central_22 = _generate_status( + "focal", + "22.03/stable", + "ch:amd64/focal/ovn-central-178", + [], + OVN_UNITS, + OVN_MACHINES, + OVN_WORKLOADS["22.03"], + ) + return {"ovn_central_20": mock_ovn_central_20, "ovn_central_22": mock_ovn_central_22} + + +def generate_mysql_innodb_cluster_status(): + mock_mysql_innodb_cluster = _generate_status( + "focal", + "8.0/stable", + "ch:amd64/focal/mysql-innodb-cluster-106", + [], + MYSQL_UNITS, + MYSQL_MACHINES, + MYSQL_WORKLOADS["8.0"], + ) + return {"mysql_innodb_cluster": mock_mysql_innodb_cluster} + + +def generate_glance_simplestreams_sync_status(): + mock_glance_simplestreams_sync_focal_ussuri = _generate_status( + "focal", + "ussuri/stable", + "ch:amd64/focal/glance-simplestreams-sync-78", + [], + GLANCE_SIMPLE_UNITS, + GLANCE_SIMPLE_MACHINES, + "", # there is no workload version for glance-simplestreams-sync + ) + return {"glance_simplestreams_sync_focal_ussuri": mock_glance_simplestreams_sync_focal_ussuri} + + +def generate_designate_bind_status(): + mock_designate_bind_focal_ussuri = _generate_status( + "focal", + "ussuri/stable", + "ch:amd64/focal/designate-bind-737", + [], + DESIGNATE_UNITS, + DESIGNATE_MACHINES, + DESIGNATE_WORKLOADS["ussuri"], ) + return { + "designate_bind_focal_ussuri": mock_designate_bind_focal_ussuri, + } - # gnocchi on ussuri - mock_gnocchi_ussuri = MagicMock(spec_set=ApplicationStatus()) - mock_gnocchi_ussuri.series = "focal" - mock_gnocchi_ussuri.charm_channel = "ussuri/stable" - mock_gnocchi_ussuri.charm = "ch:amd64/focal/gnocchi-638" - mock_gnocchi_ussuri.subordinate_to = [] - mock_gnocchi_ussuri.units = OrderedDict( - [ - ("gnocchi/0", generate_unit("4.3.4", "3/lxd/6")), - ("gnocchi/1", generate_unit("4.3.4", "4/lxd/6")), - ("gnocchi/2", generate_unit("4.3.4", "5/lxd/5")), - ] - ) - # gnocchi on xena - mock_gnocchi_xena = MagicMock(spec_set=ApplicationStatus()) - mock_gnocchi_xena.series = "focal" - mock_gnocchi_xena.charm_channel = "xena/stable" - mock_gnocchi_xena.charm = "ch:amd64/focal/gnocchi-638" - mock_gnocchi_xena.subordinate_to = [] - mock_gnocchi_xena.units = OrderedDict( - [ - ("gnocchi/0", generate_unit("4.4.1", "3/lxd/6")), - ("gnocchi/1", generate_unit("4.4.1", "4/lxd/6")), - ("gnocchi/2", generate_unit("4.4.1", "5/lxd/5")), - ] +def generate_gnocchi_status(): + mock_gnocchi_focal_ussuri = _generate_status( + "focal", + "ussuri/stable", + "ch:amd64/focal/gnocchi-638", + [], + GNOCCHI_UNITS, + GNOCCHI_MACHINES, + GNOCCHI_WORKLOADS["ussuri"], + ) + mock_gnocchi_focal_xena = _generate_status( + "focal", + "xena/stable", + "ch:amd64/focal/gnocchi-638", + [], + GNOCCHI_UNITS, + GNOCCHI_MACHINES, + GNOCCHI_WORKLOADS["xena"], ) + return { + "gnocchi_focal_ussuri": mock_gnocchi_focal_ussuri, + "gnocchi_focal_xena": mock_gnocchi_focal_xena, + } - # designate-bind on ussuri - mock_designate_bind_ussuri = MagicMock(spec_set=ApplicationStatus()) - mock_designate_bind_ussuri.series = "focal" - mock_designate_bind_ussuri.charm_channel = "ussuri/stable" - mock_designate_bind_ussuri.charm = "ch:amd64/focal/designate-bind-737" - mock_designate_bind_ussuri.subordinate_to = [] - mock_designate_bind_ussuri.units = OrderedDict( - [ - ("gnocchi/0", generate_unit("9.16.1", "1/lxd/6")), - ("gnocchi/1", generate_unit("9.16.1", "2/lxd/6")), - ] + +def generate_ovn_chassis_status(): + mock_ovn_chassis_focal_22 = _generate_status( + "focal", + "22.03/stable", + "ch:amd64/focal/ovn-chassis-178", + ["nova-compute"], + [], + [], + OVN_WORKLOADS["22.03"], + ) + mock_ovn_chassis_focal_20 = _generate_status( + "focal", + "20.03/stable", + "ch:amd64/focal/ovn-chassis-178", + ["nova-compute"], + [], + [], + OVN_WORKLOADS["20.03"], ) + return { + "ovn_chassis_focal_20": mock_ovn_chassis_focal_20, + "ovn_chassis_focal_22": mock_ovn_chassis_focal_22, + } - mock_nova_ussuri = MagicMock(spec_set=ApplicationStatus()) - mock_nova_ussuri.series = "focal" - mock_nova_ussuri.charm_channel = "ussuri/stable" - mock_nova_ussuri.charm = "ch:amd64/focal/nova-compute-638" - mock_nova_ussuri.subordinate_to = [] - mock_nova_ussuri.units = OrderedDict( - [ - ("nova-compute/0", generate_unit("21.0.0", "0")), - ("nova-compute/1", generate_unit("21.0.0", "1")), - ("nova-compute/2", generate_unit("21.0.0", "2")), - ] + +def generate_keystone_ldap_status(): + mock_keystone_ldap_focal_ussuri = _generate_status( + "focal", + "ussuri/stable", + "ch:amd64/focal/keystone-ldap-437", + ["keystone"], + [], + [], + "", + ) + return {"keystone_ldap_focal_ussuri": mock_keystone_ldap_focal_ussuri} + + +def generate_ceph_dashboard_status(): + mock_ceph_dashboard_octopus = _generate_status( + "focal", + "octopus/stable", + "ch:amd64/focal/ceph-dashboard-178", + ["ceph-mon"], + [], + [], + CEPH_WORKLOADS["octopus"], + ) + mock_ceph_dashboard_pacific = _generate_status( + "focal", + "pacific/stable", + "ch:amd64/focal/ceph-dashboard-178", + ["ceph-mon"], + [], + [], + CEPH_WORKLOADS["pacific"], ) + return { + "ceph_dashboard_octopus": mock_ceph_dashboard_octopus, + "ceph_dashboard_pacific": mock_ceph_dashboard_pacific, + } - mock_nova_wallaby = MagicMock(spec_set=ApplicationStatus()) - mock_nova_wallaby.series = "focal" - mock_nova_wallaby.charm_channel = "wallaby/stable" - mock_nova_wallaby.charm = "ch:amd64/focal/nova-compute-638" - mock_nova_wallaby.subordinate_to = [] - mock_nova_wallaby.units = OrderedDict( - [ - ("nova-compute/0", generate_unit("24.1.0", "0")), - ("nova-compute/1", generate_unit("24.1.0", "1")), - ("nova-compute/2", generate_unit("24.1.0", "2")), - ] + +def generate_mysql_router_status(): + mock_mysql_router = _generate_status( + "focal", "8.0/stable", "ch:amd64/focal/mysql-router-437", ["keystone"], [], [], "" ) + return {"mysql_router": mock_mysql_router} - # glance-simplestreams-sync does not have workload_version - mock_glance_simplestreams_sync_ussuri = MagicMock(spec_set=ApplicationStatus()) - mock_glance_simplestreams_sync_ussuri.series = "focal" - mock_glance_simplestreams_sync_ussuri.charm_channel = "ussuri/stable" - mock_glance_simplestreams_sync_ussuri.charm = "ch:amd64/focal/glance-simplestreams-sync-78" - mock_glance_simplestreams_sync_ussuri.subordinate_to = [] - mock_glance_simplestreams_sync_ussuri.units = OrderedDict( - [ - ("glance-simplestreams-sync/0", generate_unit("", "4/lxd/5")), - ] + +def generate_my_app(): + mock_my_app = _generate_status( + "focal", + "12.5/stable", + "ch:amd64/focal/my-app-638", + [], + ["my-app/0"], + ["0/lxd/11"], + "12.5", ) + return {"my_app": mock_my_app} - mock_rmq = MagicMock(spec_set=ApplicationStatus()) - mock_rmq.series = "focal" - mock_rmq.charm_channel = "3.8/stable" - mock_rmq.charm = "ch:amd64/focal/rabbitmq-server-638" - mock_rmq.subordinate_to = [] - mock_rmq.units = OrderedDict([("rabbitmq-server/0", generate_unit("3.8", "0/lxd/19"))]) - - mock_rmq_unknown = MagicMock(spec_set=ApplicationStatus()) - mock_rmq_unknown.series = "focal" - mock_rmq_unknown.charm_channel = "80.5/stable" - mock_rmq_unknown.charm = "ch:amd64/focal/rabbitmq-server-638" - mock_rmq_unknown.subordinate_to = [] - mock_rmq_unknown.units = OrderedDict( - [("rabbitmq-server/0", generate_unit("80.5", "0/lxd/19"))] - ) - - mock_unknown_app = MagicMock(spec_set=ApplicationStatus()) - mock_unknown_app.series = "focal" - mock_unknown_app.charm_channel = "12.5/stable" - mock_unknown_app.charm = "ch:amd64/focal/my-app-638" - mock_unknown_app.subordinate_to = [] - mock_unknown_app.units = OrderedDict([("my-app/0", generate_unit("12.5", "0/lxd/11"))]) - - # openstack related principal application without openstack origin or source - mock_vault = MagicMock(spec_set=ApplicationStatus()) - mock_vault.series = "focal" - mock_vault.charm_channel = "1.7/stable" - mock_vault.charm = "ch:amd64/focal/vault-638" - mock_vault.subordinate_to = [] - mock_vault.units = OrderedDict([("vault/0", generate_unit("1.7", "5"))]) - - # auxiliary subordinate application - mock_mysql_router = MagicMock(spec_set=ApplicationStatus()) - mock_mysql_router.series = "focal" - mock_mysql_router.charm_channel = "8.0/stable" - mock_mysql_router.charm = "ch:amd64/focal/mysql-router-437" - mock_mysql_router.subordinate_to = ["keystone"] - mock_mysql_router.units = {} - - # OpenStack subordinate application - mock_keystone_ldap = MagicMock(spec_set=ApplicationStatus()) - mock_keystone_ldap.series = "focal" - mock_keystone_ldap.charm_channel = "ussuri/stable" - mock_keystone_ldap.charm = "ch:amd64/focal/keystone-ldap-437" - mock_keystone_ldap.subordinate_to = ["keystone"] - mock_keystone_ldap.units = {} - - # OpenStack subordinate application cs - mock_keystone_ldap_cs = MagicMock(spec_set=ApplicationStatus()) - mock_keystone_ldap_cs.series = "focal" - mock_keystone_ldap_cs.charm_channel = "stable" - mock_keystone_ldap_cs.charm = "cs:amd64/focal/keystone-ldap-437" - mock_keystone_ldap_cs.subordinate_to = ["keystone"] - mock_keystone_ldap_cs.units = {} - - # ceph-mon application on ussuri - mock_ceph_mon_ussuri = MagicMock(spec_set=ApplicationStatus()) - mock_ceph_mon_ussuri.series = "focal" - mock_ceph_mon_ussuri.charm_channel = "octopus/stable" - mock_ceph_mon_ussuri.charm = "ch:amd64/focal/ceph-mon-177" - mock_ceph_mon_ussuri.subordinate_to = [] - mock_ceph_mon_ussuri.units = OrderedDict([("ceph-mon/0", generate_unit("15.2.0", "6"))]) - - # ceph-mon application on xena - mock_ceph_mon_xena = MagicMock(spec_set=ApplicationStatus()) - mock_ceph_mon_xena.series = "focal" - mock_ceph_mon_xena.charm_channel = "pacific/stable" - mock_ceph_mon_xena.charm = "ch:amd64/focal/ceph-mon-178" - mock_ceph_mon_xena.subordinate_to = [] - mock_ceph_mon_xena.units = OrderedDict([("ceph-mon/0", generate_unit("16.2.0", "7"))]) - - # ceph-osd application on ussuri - mock_ceph_osd_ussuri = MagicMock(spec_set=ApplicationStatus()) - mock_ceph_osd_ussuri.series = "focal" - mock_ceph_osd_ussuri.charm_channel = "octopus/stable" - mock_ceph_osd_ussuri.charm = "ch:amd64/focal/ceph-osd-177" - mock_ceph_osd_ussuri.subordinate_to = [] - mock_ceph_osd_ussuri.units = OrderedDict([("ceph-osd/0", generate_unit("15.2.0", "6"))]) - - # mysql-innodb-cluster application on ussuri using 8.0 - mock_mysql_innodb_cluster_ussuri = MagicMock(spec_set=ApplicationStatus()) - mock_mysql_innodb_cluster_ussuri.series = "focal" - mock_mysql_innodb_cluster_ussuri.charm_channel = "8.0/stable" - mock_mysql_innodb_cluster_ussuri.charm = "ch:amd64/focal/mysql-innodb-cluster-106" - mock_mysql_innodb_cluster_ussuri.subordinate_to = [] - mock_mysql_innodb_cluster_ussuri.units = OrderedDict( - [("ovn-central/0", generate_unit("8.0", "0/lxd/7"))] - ) - - # ovn-central application on ussuri using 22.03 - mock_ovn_central_ussuri_22 = MagicMock(spec_set=ApplicationStatus()) - mock_ovn_central_ussuri_22.series = "focal" - mock_ovn_central_ussuri_22.charm_channel = "22.03/stable" - mock_ovn_central_ussuri_22.charm = "ch:amd64/focal/ovn-central-178" - mock_ovn_central_ussuri_22.subordinate_to = [] - mock_ovn_central_ussuri_22.units = OrderedDict( - [("ovn-central/0", generate_unit("22.03.2", "0/lxd/7"))] - ) - - # ovn-central application on ussuri using 20.03 - mock_ovn_central_ussuri_20 = MagicMock(spec_set=ApplicationStatus()) - mock_ovn_central_ussuri_20.series = "focal" - mock_ovn_central_ussuri_20.charm_channel = "20.03/stable" - mock_ovn_central_ussuri_20.charm = "ch:amd64/focal/ovn-central-178" - mock_ovn_central_ussuri_20.subordinate_to = [] - mock_ovn_central_ussuri_20.units = OrderedDict( - [("ovn-central/0", generate_unit("20.03.2", "0/lxd/7"))] - ) - - # ovn-chassis application on ussuri using 22.03 - mock_ovn_chassis_ussuri_22 = MagicMock(spec_set=ApplicationStatus()) - mock_ovn_chassis_ussuri_22.series = "focal" - mock_ovn_chassis_ussuri_22.charm_channel = "22.03/stable" - mock_ovn_chassis_ussuri_22.charm = "ch:amd64/focal/ovn-chassis-178" - mock_ovn_chassis_ussuri_22.workload_version = "22.03.2" - mock_ovn_chassis_ussuri_22.subordinate_to = ["nova-compute"] - mock_ovn_chassis_ussuri_22.units = {} - - # ovn-chassis application on ussuri using 20.03 - mock_ovn_chassis_ussuri_20 = MagicMock(spec_set=ApplicationStatus()) - mock_ovn_chassis_ussuri_20.series = "focal" - mock_ovn_chassis_ussuri_20.charm_channel = "20.03/stable" - mock_ovn_chassis_ussuri_20.charm = "ch:amd64/focal/ovn-chassis-178" - mock_ovn_chassis_ussuri_20.subordinate_to = ["nova-compute"] - mock_ovn_chassis_ussuri_20.workload_version = "20.03.2" - mock_ovn_chassis_ussuri_20.units = {} - - # ceph-dashboard on ussuri - mock_ceph_dashboard_ussuri = MagicMock(spec_set=ApplicationStatus()) - mock_ceph_dashboard_ussuri.series = "focal" - mock_ceph_dashboard_ussuri.charm_channel = "octopus/stable" - mock_ceph_dashboard_ussuri.charm = "ch:amd64/focal/ceph-dashboard-178" - mock_ceph_dashboard_ussuri.subordinate_to = ["ceph-mon"] - mock_ceph_dashboard_ussuri.units = {} - - # ceph-dashboard on xena - mock_ceph_dashboard_xena = MagicMock(spec_set=ApplicationStatus()) - mock_ceph_dashboard_xena.series = "focal" - mock_ceph_dashboard_xena.charm_channel = "pacific/stable" - mock_ceph_dashboard_xena.charm = "ch:amd64/focal/ceph-dashboard-178" - mock_ceph_dashboard_xena.subordinate_to = ["ceph-mon"] - mock_ceph_dashboard_xena.units = {} - - status = { - "keystone_ussuri": mock_keystone_ussuri, - "keystone_victoria": mock_keystone_victoria, - "keystone_wallaby": mock_keystone_wallaby, - "keystone_ussuri_victoria": mock_keystone_ussuri_victoria, - "keystone_bionic_ussuri": mock_keystone_bionic_ussuri, - "cinder_ussuri": mock_cinder_ussuri, - "glance_simplestreams_sync_ussuri": mock_glance_simplestreams_sync_ussuri, - "gnocchi_ussuri": mock_gnocchi_ussuri, - "gnocchi_xena": mock_gnocchi_xena, - "designate_bind_ussuri": mock_designate_bind_ussuri, - "rabbitmq_server": mock_rmq, - "unknown_rabbitmq_server": mock_rmq_unknown, - "keystone_ussuri_cs": mock_keystone_ussuri_cs, - "keystone_wallaby": mock_keystone_wallaby, - "unknown_app": mock_unknown_app, - "mysql_router": mock_mysql_router, - "vault": mock_vault, - "keystone-ldap": mock_keystone_ldap, - "keystone-ldap-cs": mock_keystone_ldap_cs, - "nova_ussuri": mock_nova_ussuri, - "nova_wallaby": mock_nova_wallaby, - "ceph-mon_ussuri": mock_ceph_mon_ussuri, - "ceph-mon_xena": mock_ceph_mon_xena, - "ceph_osd_ussuri": mock_ceph_osd_ussuri, - "ceph_dashboard_ussuri": mock_ceph_dashboard_ussuri, - "ceph_dashboard_xena": mock_ceph_dashboard_xena, - "cinder_ussuri_on_nova": mock_cinder_on_nova, - "mysql-innodb-cluster": mock_mysql_innodb_cluster_ussuri, - "ovn_central_ussuri_22": mock_ovn_central_ussuri_22, - "ovn_central_ussuri_20": mock_ovn_central_ussuri_20, - "ovn_chassis_ussuri_22": mock_ovn_chassis_ussuri_22, - "ovn_chassis_ussuri_20": mock_ovn_chassis_ussuri_20, - } - return status + +def _generate_status( + series, charm_channel, charm, subordinate_to, units, machines, workload_version +): + app_mock = MagicMock(spec_set=ApplicationStatus()) + app_mock.series = series + app_mock.charm_channel = charm_channel + app_mock.charm = charm + app_mock.subordinate_to = subordinate_to + # subordinates get workload version from the application + if subordinate_to: + app_mock.workload_version = workload_version + + units_machines_workloads = zip_longest( + units, machines, [workload_version], fillvalue=workload_version + ) + app_mock.units = _generate_units(units_machines_workloads) + return app_mock @pytest.fixture @@ -407,19 +466,19 @@ def full_status(status, model): mock_full_status.model.name = model.name mock_full_status.applications = OrderedDict( [ - ("keystone", status["keystone_ussuri"]), - ("cinder", status["cinder_ussuri"]), + ("keystone", status["keystone_focal_ussuri"]), + ("cinder", status["cinder_focal_ussuri"]), ("rabbitmq-server", status["rabbitmq_server"]), - ("my_app", status["unknown_app"]), - ("nova-compute", status["nova_ussuri"]), - ("ceph-osd", status["ceph_osd_ussuri"]), + ("my_app", status["my_app"]), + ("nova-compute", status["nova_focal_ussuri"]), + ("ceph-osd", status["ceph_osd_octopus"]), ] ) return mock_full_status @pytest.fixture -def units(): +def units(apps_machines): units_ussuri = [] units_wallaby = [] units_ussuri.append( @@ -427,7 +486,7 @@ def units(): name="keystone/0", os_version=OpenStackRelease("ussuri"), workload_version="17.0.1", - machine="0/lxd/12", + machine=apps_machines["keystone"]["0/lxd/12"], ) ) units_ussuri.append( @@ -435,7 +494,7 @@ def units(): name="keystone/1", os_version=OpenStackRelease("ussuri"), workload_version="17.0.1", - machine="1/lxd/12", + machine=apps_machines["keystone"]["1/lxd/12"], ) ) units_ussuri.append( @@ -443,7 +502,7 @@ def units(): name="keystone/2", os_version=OpenStackRelease("ussuri"), workload_version="17.0.1", - machine="2/lxd/13", + machine=apps_machines["keystone"]["2/lxd/13"], ) ) units_wallaby.append( @@ -451,7 +510,7 @@ def units(): name="keystone/0", os_version=OpenStackRelease("wallaby"), workload_version="19.1.0", - machine="0/lxd/12", + machine=apps_machines["keystone"]["0/lxd/12"], ) ) units_wallaby.append( @@ -459,7 +518,7 @@ def units(): name="keystone/1", os_version=OpenStackRelease("wallaby"), workload_version="19.1.0", - machine="1/lxd/12", + machine=apps_machines["keystone"]["1/lxd/12"], ) ) units_wallaby.append( @@ -467,7 +526,7 @@ def units(): name="keystone/2", os_version=OpenStackRelease("wallaby"), workload_version="19.1.0", - machine="2/lxd/13", + machine=apps_machines["keystone"]["2/lxd/13"], ) ) return {"units_ussuri": units_ussuri, "units_wallaby": units_wallaby} @@ -503,8 +562,11 @@ async def get_charm_name(value: str): @pytest.fixture -def model(config): +def model(config, apps_machines): """Define test COUModel object.""" + machines = {} + for sub_machines in apps_machines.values(): + machines = {**machines, **sub_machines} model_name = "test_model" from cou.utils import juju_utils @@ -518,55 +580,81 @@ def model(config): model.scp_from_unit = AsyncMock() model.get_application_config = mock_get_app_config = AsyncMock() mock_get_app_config.side_effect = config.get + model.get_machines = machines return model @pytest.fixture -def apps(status, config, model): - keystone_ussuri_status = status["keystone_ussuri"] - keystone_wallaby_status = status["keystone_wallaby"] - cinder_ussuri_status = status["cinder_ussuri"] +def apps(status, config, model, apps_machines): + keystone_focal_ussuri_status = status["keystone_focal_ussuri"] + keystone_focal_wallaby_status = status["keystone_focal_wallaby"] + cinder_focal_ussuri_status = status["cinder_focal_ussuri"] rmq_status = status["rabbitmq_server"] - keystone_ldap_status = status["keystone-ldap"] - keystone_bionic_ussuri_status = status["keystone_bionic_ussuri"] + keystone_ldap_focal_ussuri_status = status["keystone_ldap_focal_ussuri"] keystone_ussuri = Keystone( - "keystone", keystone_ussuri_status, config["openstack_ussuri"], model, "keystone" + "keystone", + keystone_focal_ussuri_status, + config["openstack_ussuri"], + model, + "keystone", + apps_machines["keystone"], ) keystone_wallaby = Keystone( - "keystone", keystone_wallaby_status, config["openstack_wallaby"], model, "keystone" - ) - keystone_bionic_ussuri = OpenStackApplication( - "keystone", keystone_bionic_ussuri_status, config["openstack_ussuri"], model, "keystone" + "keystone", + keystone_focal_wallaby_status, + config["openstack_wallaby"], + model, + "keystone", + apps_machines["keystone"], ) cinder_ussuri = OpenStackApplication( - "cinder", cinder_ussuri_status, config["openstack_ussuri"], model, "cinder" - ) - rmq_ussuri = AuxiliaryApplication( - "rabbitmq-server", rmq_status, config["auxiliary_ussuri"], model, "rabbitmq-server" + "cinder", + cinder_focal_ussuri_status, + config["openstack_ussuri"], + model, + "cinder", + apps_machines["cinder"], + ) + rmq = AuxiliaryApplication( + "rabbitmq-server", + rmq_status, + config["auxiliary_ussuri"], + model, + "rabbitmq-server", + apps_machines["rmq"], ) rmq_wallaby = AuxiliaryApplication( - "rabbitmq-server", rmq_status, config["auxiliary_wallaby"], model, "rabbitmq-server" + "rabbitmq-server", + rmq_status, + config["auxiliary_wallaby"], + model, + "rabbitmq-server", + apps_machines["rmq"], ) keystone_ldap = SubordinateApplication( - "keystone-ldap", keystone_ldap_status, {}, model, "keystone-ldap" + "keystone-ldap", keystone_ldap_focal_ussuri_status, {}, model, "keystone-ldap", {} ) keystone_mysql_router = AuxiliarySubordinateApplication( - "keystone-mysql-router", status["mysql_router"], {}, model, "mysql-router" + "keystone-mysql-router", status["mysql_router"], {}, model, "mysql-router", {} ) - nova_ussuri = OpenStackApplication( - "nova-compute", status["nova_ussuri"], config["openstack_ussuri"], model, "nova-compute" + nova_focal_ussuri = OpenStackApplication( + "nova-compute", + status["nova_focal_ussuri"], + config["openstack_ussuri"], + model, + "nova-compute", + apps_machines["nova-compute"], ) return { - "keystone_ussuri": keystone_ussuri, - "keystone_wallaby": keystone_wallaby, - "keystone_bionic_ussuri": keystone_bionic_ussuri, - "cinder_ussuri": cinder_ussuri, - "rmq_ussuri": rmq_ussuri, + "keystone_focal_ussuri": keystone_ussuri, + "keystone_focal_wallaby": keystone_wallaby, + "cinder_focal_ussuri": cinder_ussuri, + "rmq": rmq, "rmq_wallaby": rmq_wallaby, - "keystone_ldap": keystone_ldap, - "nova_ussuri": nova_ussuri, + "keystone_ldap_focal_ussuri": keystone_ldap, + "nova_focal_ussuri": nova_focal_ussuri, "keystone_mysql_router": keystone_mysql_router, } diff --git a/tests/unit/steps/test_steps_analyze.py b/tests/unit/steps/test_steps_analyze.py index 97c25e5c..8ca823a1 100644 --- a/tests/unit/steps/test_steps_analyze.py +++ b/tests/unit/steps/test_steps_analyze.py @@ -18,6 +18,7 @@ from cou.apps.base import ApplicationUnit, OpenStackApplication from cou.steps import analyze from cou.steps.analyze import Analysis +from cou.utils.juju_utils import Machine def test_analysis_dump(apps, model): @@ -75,9 +76,9 @@ def test_analysis_dump(apps, model): result = analyze.Analysis( model=model, apps_control_plane=[ - apps["keystone_ussuri"], - apps["cinder_ussuri"], - apps["rmq_ussuri"], + apps["keystone_focal_ussuri"], + apps["cinder_focal_ussuri"], + apps["rmq"], ], apps_data_plane=[], ) @@ -85,10 +86,16 @@ def test_analysis_dump(apps, model): @pytest.mark.asyncio -async def test_populate_model(full_status, config, model): +async def test_populate_model(full_status, config, model, apps_machines): model.get_status = AsyncMock(return_value=full_status) model.get_application_config = AsyncMock(return_value=config["openstack_ussuri"]) + machines = {} + for sub_dict in apps_machines.values(): + machines.update(sub_dict) + + model.get_machines = AsyncMock(return_value=machines) + # Initially, 6 applications are in the status: keystone, cinder, rabbitmq-server, my-app, # ceph-osd and nova-compute. my-app it's not on the lookup table, so won't be instantiated. assert len(full_status.applications) == 6 @@ -108,7 +115,7 @@ async def test_populate_model(full_status, config, model): @patch.object(analyze.Analysis, "_populate", new_callable=AsyncMock) async def test_analysis_create(mock_populate, apps, model): """Test analysis object.""" - exp_apps = [apps["keystone_ussuri"], apps["cinder_ussuri"], apps["rmq_ussuri"]] + exp_apps = [apps["keystone_focal_ussuri"], apps["cinder_focal_ussuri"], apps["rmq"]] expected_result = analyze.Analysis( model=model, apps_control_plane=exp_apps, apps_data_plane=[] ) @@ -123,7 +130,11 @@ async def test_analysis_create(mock_populate, apps, model): async def test_analysis_detect_current_cloud_os_release_different_releases(apps, model): result = analyze.Analysis( model=model, - apps_control_plane=[apps["rmq_ussuri"], apps["keystone_wallaby"], apps["cinder_ussuri"]], + apps_control_plane=[ + apps["rmq"], + apps["keystone_focal_wallaby"], + apps["cinder_focal_ussuri"], + ], apps_data_plane=[], ) @@ -135,7 +146,7 @@ async def test_analysis_detect_current_cloud_os_release_different_releases(apps, async def test_analysis_detect_current_cloud_os_release_same_release(apps, model): result = analyze.Analysis( model=model, - apps_control_plane=[apps["cinder_ussuri"], apps["keystone_ussuri"]], + apps_control_plane=[apps["cinder_focal_ussuri"], apps["keystone_focal_ussuri"]], apps_data_plane=[], ) @@ -147,7 +158,11 @@ async def test_analysis_detect_current_cloud_os_release_same_release(apps, model async def test_analysis_detect_current_cloud_series_same_series(apps, model): result = analyze.Analysis( model=model, - apps_control_plane=[apps["rmq_ussuri"], apps["keystone_wallaby"], apps["cinder_ussuri"]], + apps_control_plane=[ + apps["rmq"], + apps["keystone_focal_wallaby"], + apps["cinder_focal_ussuri"], + ], apps_data_plane=[], ) @@ -157,9 +172,13 @@ async def test_analysis_detect_current_cloud_series_same_series(apps, model): @pytest.mark.asyncio async def test_analysis_detect_current_cloud_series_different_series(apps, model): + # change keystone to bionic + keystone_bionic_ussuri = apps["keystone_focal_ussuri"] + keystone_bionic_ussuri.status.series = "bionic" + result = analyze.Analysis( model=model, - apps_control_plane=[apps["cinder_ussuri"], apps["keystone_bionic_ussuri"]], + apps_control_plane=[apps["cinder_focal_ussuri"], keystone_bionic_ussuri], apps_data_plane=[], ) @@ -174,9 +193,9 @@ def _app(name, units): return app -def _unit(machine): +def _unit(machine_id): unit = MagicMock(spec_set=ApplicationUnit).return_value - unit.machine = machine + unit.machine = Machine(machine_id, (), "zone-1") return unit diff --git a/tests/unit/steps/test_steps_plan.py b/tests/unit/steps/test_steps_plan.py index 324d1f27..55632f46 100644 --- a/tests/unit/steps/test_steps_plan.py +++ b/tests/unit/steps/test_steps_plan.py @@ -140,9 +140,9 @@ def generate_expected_upgrade_plan_subordinate(app, target, model): @pytest.mark.asyncio async def test_generate_plan(apps, model, cli_args): target = OpenStackRelease("victoria") - app_keystone = apps["keystone_ussuri"] - app_cinder = apps["cinder_ussuri"] - app_keystone_ldap = apps["keystone_ldap"] + app_keystone = apps["keystone_focal_ussuri"] + app_cinder = apps["cinder_focal_ussuri"] + app_keystone_ldap = apps["keystone_ldap_focal_ussuri"] analysis_result = Analysis( model=model, apps_control_plane=[app_keystone, app_cinder, app_keystone_ldap], @@ -304,8 +304,8 @@ async def test_create_upgrade_plan_failed(): def test_plan_print_warn_manually_upgrade(mock_print, model, apps): result = Analysis( model=model, - apps_control_plane=[apps["keystone_wallaby"]], - apps_data_plane=[apps["nova_ussuri"]], + apps_control_plane=[apps["keystone_focal_wallaby"]], + apps_data_plane=[apps["nova_focal_ussuri"]], ) manually_upgrade_data_plane(result) mock_print.assert_called_with( @@ -317,8 +317,8 @@ def test_plan_print_warn_manually_upgrade(mock_print, model, apps): def test_analysis_not_print_warn_manually_upgrade(mock_print, model, apps): result = Analysis( model=model, - apps_control_plane=[apps["keystone_ussuri"]], - apps_data_plane=[apps["nova_ussuri"]], + apps_control_plane=[apps["keystone_focal_ussuri"]], + apps_data_plane=[apps["nova_focal_ussuri"]], ) manually_upgrade_data_plane(result) mock_print.assert_not_called() diff --git a/tests/unit/utils/test_juju_utils.py b/tests/unit/utils/test_juju_utils.py index 2bcfb742..d4a56e88 100644 --- a/tests/unit/utils/test_juju_utils.py +++ b/tests/unit/utils/test_juju_utils.py @@ -18,6 +18,7 @@ from juju.action import Action from juju.application import Application from juju.client.connector import NoConnectionException +from juju.machine import Machine from juju.model import Model from juju.unit import Unit @@ -33,6 +34,35 @@ from cou.utils import juju_utils +@pytest.fixture +def mocked_model(mocker): + """Fixture providing mocked juju.model.Model object.""" + mocker.patch("cou.utils.juju_utils.FileJujuData") + model_mocker = mocker.patch("cou.utils.juju_utils.Model", return_value=AsyncMock(Model)) + model = model_mocker.return_value + model.connection.return_value.is_open = True # simulate already connected model + yield model + + +def _generate_juju_unit(app: str, machine_id: str) -> MagicMock: + unit = MagicMock(set=Unit)() + unit.application = app + unit.machine.id = machine_id + return unit + + +def _generate_juju_machine(machine_id: str) -> MagicMock: + machine = MagicMock(set=Machine)() + machine.id = machine_id + machine.hardware_characteristics = { + "arch": "amd64", + "mem": 0, + "cpu-cores": 0, + "availability-zone": f"zone-{int(machine_id) + 1}", + } + return machine + + def test_normalize_action_results(): results = {"Stderr": "error", "stdout": "output"} expected = {"Stderr": "error", "Stdout": "output", "stderr": "error", "stdout": "output"} @@ -135,14 +165,26 @@ async def func(self): await test_model.func() -@pytest.fixture -def mocked_model(mocker): - """Fixture providing mocked juju.model.Model object.""" - mocker.patch("cou.utils.juju_utils.FileJujuData") - model_mocker = mocker.patch("cou.utils.juju_utils.Model", return_value=AsyncMock(Model)) - model = model_mocker.return_value - model.connection.return_value.is_open = True # simulate already connected model - yield model +@pytest.mark.parametrize( + "machine_id, az", + [ + # one field different is considered another machine + ("0", "zone-3"), + ("1", "zone-2"), + ], +) +def test_machine_not_eq(machine_id, az): + machine_0 = juju_utils.Machine("0", (), "zone-1") + machine_1 = juju_utils.Machine(machine_id, (), az) + + assert machine_0 != machine_1 + + +def test_machine_eq(): + machine_0 = juju_utils.Machine("0", (), "zone-1") + machine_1 = juju_utils.Machine("0", (), "zone-1") + + assert machine_0 == machine_1 @patch("cou.utils.juju_utils.FileJujuData") @@ -530,3 +572,36 @@ async def test_coumodel_wait_for_active_idle_timeout(mock_get_supported_apps, mo status="active", ) mock_get_supported_apps.assert_not_awaited() + + +@pytest.mark.asyncio +async def test_get_machines(mocked_model): + """Test COUModel getting machines from model.""" + expected_machines = { + "0": juju_utils.Machine( + "0", + ( + "app1", + "app2", + ), + "zone-1", + ), + "1": juju_utils.Machine("1", ("app1",), "zone-2"), + "2": juju_utils.Machine("2", ("app1",), "zone-3"), + } + mocked_model.machines = {f"{i}": _generate_juju_machine(f"{i}") for i in range(3)} + mocked_model.units = { + "app1/0": _generate_juju_unit("app1", "0"), + "app1/1": _generate_juju_unit("app1", "1"), + "app1/2": _generate_juju_unit("app1", "2"), + "app2/0": _generate_juju_unit("app2", "0"), + } + mocked_model.applications = { + "app1": MagicMock(spec_set=Application)(), + "app2": MagicMock(spec_set=Application)(), + } + + model = juju_utils.COUModel("test-model") + machines = await model.get_machines() + + assert machines == expected_machines