From 2288c8a9a7598c6a4752c5aa2ffafae23ac9c82d Mon Sep 17 00:00:00 2001 From: Sergiu Miclea Date: Sun, 7 May 2023 20:14:32 +0000 Subject: [PATCH] Add unit tests for Minion Manager RPC Server --- coriolis/tests/minion_manager/__init__.py | 0 coriolis/tests/minion_manager/rpc/__init__.py | 0 .../data/get_minion_pool_refresh_flow.yaml | 159 ++++++++ ...machine_allocation_subflow_for_action.yaml | 201 ++++++++++ ...ion_pool_selections_for_action_config.yaml | 352 ++++++++++++++++++ .../tests/minion_manager/rpc/test_server.py | 201 ++++++++++ coriolis/tests/testutils.py | 33 ++ 7 files changed, 946 insertions(+) create mode 100644 coriolis/tests/minion_manager/__init__.py create mode 100644 coriolis/tests/minion_manager/rpc/__init__.py create mode 100644 coriolis/tests/minion_manager/rpc/data/get_minion_pool_refresh_flow.yaml create mode 100644 coriolis/tests/minion_manager/rpc/data/make_minion_machine_allocation_subflow_for_action.yaml create mode 100644 coriolis/tests/minion_manager/rpc/data/validate_minion_pool_selections_for_action_config.yaml create mode 100644 coriolis/tests/minion_manager/rpc/test_server.py diff --git a/coriolis/tests/minion_manager/__init__.py b/coriolis/tests/minion_manager/__init__.py new file mode 100644 index 000000000..e69de29bb diff --git a/coriolis/tests/minion_manager/rpc/__init__.py b/coriolis/tests/minion_manager/rpc/__init__.py new file mode 100644 index 000000000..e69de29bb diff --git a/coriolis/tests/minion_manager/rpc/data/get_minion_pool_refresh_flow.yaml b/coriolis/tests/minion_manager/rpc/data/get_minion_pool_refresh_flow.yaml new file mode 100644 index 000000000..5bd1ff55a --- /dev/null +++ b/coriolis/tests/minion_manager/rpc/data/get_minion_pool_refresh_flow.yaml @@ -0,0 +1,159 @@ +# perform health checks for powered on machines +# skip health checks for non-powered on machines +- config: + minion_pool: + id: minion_pool1 + minimum_minions: 4 + minion_retention_strategy: delete + minion_machines: + - id: minion1 + allocation_status: AVAILABLE + power_status: POWERED_ON + - id: minion2 + allocation_status: ERROR + power_status: POWERED_ERROR + - id: minion3 + allocation_status: AVAILABLE + power_status: POWERED_ON + - id: minion4 + allocation_status: ERROR_DEPLOYING + power_status: POWERED_ON + expect: + result: + flow_tasks: + - pool-minion_pool1-machine-minion1-healthcheck + - pool-minion_pool1-machine-minion3-healthcheck + - pool-minion_pool1-machine-minion4-healthcheck + # "include" are the db calls that must be present + # "exclude" are the db calls that must not be present + db_calls: + include: + - set_minion_machine_allocation_status: + id: minion1 + allocation_status: HEALTHCHECKING + - set_minion_machine_allocation_status: + id: minion3 + allocation_status: HEALTHCHECKING + - set_minion_machine_allocation_status: + id: minion4 + allocation_status: HEALTHCHECKING + exclude: + - set_minion_machine_allocation_status: + id: minion2 + allocation_status: HEALTHCHECKING + +# power off machines that are not needed +- config: + minion_pool: + id: minion_pool1 + # minimum_minions 1 means we have too many minions + minimum_minions: 1 + minion_max_idle_time: 0 + minion_retention_strategy: poweroff + minion_machines: + - id: minion1 + allocation_status: AVAILABLE + power_status: POWERED_ON + - id: minion2 + allocation_status: ERROR + power_status: POWERED_ERROR + - id: minion3 + allocation_status: AVAILABLE + power_status: POWERED_ON + - id: minion4 + allocation_status: ERROR_DEPLOYING + power_status: POWERED_ON + expect: + result: + flow_tasks: + - pool-minion_pool1-machine-minion1-power-off + - pool-minion_pool1-machine-minion3-healthcheck + - pool-minion_pool1-machine-minion4-healthcheck + db_calls: + include: + - set_minion_machine_allocation_status: + id: minion1 + allocation_status: POWERING_OFF + - set_minion_machine_allocation_status: + id: minion3 + allocation_status: HEALTHCHECKING + - set_minion_machine_allocation_status: + id: minion4 + allocation_status: HEALTHCHECKING + exclude: + - set_minion_machine_allocation_status: + id: minion2 + allocation_status: HEALTHCHECKING + - set_minion_machine_allocation_status: + id: minion2 + allocation_status: POWERING_OFF + +# delete machines that are not needed +- config: + minion_pool: + id: minion_pool1 + # minimum_minions 1 means we have too many minions + minimum_minions: 1 + minion_max_idle_time: 0 + minion_retention_strategy: delete + minion_machines: + - id: minion1 + allocation_status: AVAILABLE + power_status: POWERING_OFF + - id: minion2 + allocation_status: ERROR + power_status: POWERED_ERROR + - id: minion3 + allocation_status: AVAILABLE + power_status: POWERED_ON + - id: minion4 + allocation_status: ERROR_DEPLOYING + power_status: POWERED_ON + expect: + result: + flow_tasks: + - pool-minion_pool1-machine-minion1-deallocation + - pool-minion_pool1-machine-minion3-healthcheck + - pool-minion_pool1-machine-minion4-healthcheck + db_calls: + include: + - set_minion_machine_allocation_status: + id: minion1 + allocation_status: DEALLOCATING + - set_minion_machine_allocation_status: + id: minion3 + allocation_status: HEALTHCHECKING + - set_minion_machine_allocation_status: + id: minion4 + allocation_status: HEALTHCHECKING + exclude: + - set_minion_machine_allocation_status: + id: minion2 + allocation_status: HEALTHCHECKING + - set_minion_machine_allocation_status: + id: minion2 + allocation_status: DEALLOCATING + +# invalid retention strategy +- config: + minion_pool: + id: minion_pool1 + # minimum_minions 1 means we have too many minions + minimum_minions: 1 + minion_max_idle_time: 0 + minion_retention_strategy: invalid + minion_machines: + - id: minion1 + allocation_status: AVAILABLE + power_status: POWERING_OFF + - id: minion2 + allocation_status: ERROR + power_status: POWERED_ERROR + - id: minion3 + allocation_status: AVAILABLE + power_status: POWERED_ON + - id: minion4 + allocation_status: ERROR_DEPLOYING + power_status: POWERED_ON + expect: + exception: InvalidMinionPoolState \ No newline at end of file diff --git a/coriolis/tests/minion_manager/rpc/data/make_minion_machine_allocation_subflow_for_action.yaml b/coriolis/tests/minion_manager/rpc/data/make_minion_machine_allocation_subflow_for_action.yaml new file mode 100644 index 000000000..b55765a16 --- /dev/null +++ b/coriolis/tests/minion_manager/rpc/data/make_minion_machine_allocation_subflow_for_action.yaml @@ -0,0 +1,201 @@ +## EXCEPTIONS + +# requires 2 machines, but only 1 is available +- config: + minion_pool: + id: minion_pool_1 + maximum_minions: 1 + minion_machines: + - allocation_status: AVAILABLE + action_instances: + instance_1: + name: Instance 1 + instance_2: + name: Instance 2 + expect: + exception: InvalidMinionPoolState + +# requires 2 machines, but only 1 is available +- config: + minion_pool: + id: minion_pool_1 + maximum_minions: 2 + minion_machines: + - allocation_status: IN_USE + - allocation_status: AVAILABLE + action_instances: + instance_1: + name: Instance 1 + instance_2: + name: Instance 2 + expect: + exception: InvalidMinionPoolState + +# maximum_minions is too low for 4 new machines +- config: + minion_pool: + id: minion_pool_1 + maximum_minions: 6 + minion_machines: + - id: machine_1 + allocation_status: IN_USE + - id: machine_2 + allocation_status: IN_USE + - id: machine_3 + allocation_status: IN_USE + action_instances: + instance_1: + name: Instance 1 + instance_2: + name: Instance 2 + instance_3: + name: Instance 3 + instance_4: + name: Instance 4 + expect: + exception: InvalidMinionPoolState + +## SUCCESS + +# no new machines need to be allocated +- config: + minion_pool: + id: minion_pool_1 + maximum_minions: 4 + minion_machines: + - id: machine_1 + allocation_status: AVAILABLE + - id: machine_2 + allocation_status: AVAILABLE + - id: machine_3 + allocation_status: AVAILABLE + - id: machine_4 + allocation_status: AVAILABLE + action_instances: + instance_1: + name: Instance 1 + instance_2: + name: Instance 2 + instance_3: + name: Instance 3 + instance_4: + name: Instance 4 + expect: + result: + mappings: + instance_1: machine_1 + instance_2: machine_2 + instance_3: machine_3 + instance_4: machine_4 + +# 2 new machines are allocated +- config: + minion_pool: + id: minion_pool_1 + maximum_minions: 5 + minion_machines: + - id: machine_1 + allocation_status: AVAILABLE + - id: machine_2 + allocation_status: IN_USE + - id: machine_3 + allocation_status: AVAILABLE + action_instances: + instance_1: + name: Instance 1 + instance_2: + name: Instance 2 + instance_3: + name: Instance 3 + instance_4: + name: Instance 4 + expect: + result: + mappings: + instance_1: machine_1 + instance_2: machine_3 + instance_3: new_machine + instance_4: new_machine + +# 3 new machines are allocated +- config: + minion_pool: + id: minion_pool_1 + maximum_minions: 6 + minion_machines: + - id: machine_1 + allocation_status: AVAILABLE + - id: machine_2 + allocation_status: IN_USE + - id: machine_3 + allocation_status: IN_USE + action_instances: + instance_1: + name: Instance 1 + instance_2: + name: Instance 2 + instance_3: + name: Instance 3 + instance_4: + name: Instance 4 + expect: + result: + mappings: + instance_1: machine_1 + instance_2: new_machine + instance_3: new_machine + instance_4: new_machine + +# 4 new machines are allocated +- config: + minion_pool: + id: minion_pool_1 + maximum_minions: 7 + minion_machines: + - id: machine_1 + allocation_status: IN_USE + - id: machine_2 + allocation_status: IN_USE + - id: machine_3 + allocation_status: IN_USE + action_instances: + instance_1: + name: Instance 1 + instance_2: + name: Instance 2 + instance_3: + name: Instance 3 + instance_4: + name: Instance 4 + expect: + result: + mappings: + instance_1: new_machine + instance_2: new_machine + instance_3: new_machine + instance_4: new_machine + +# 1 allocated +- config: + minion_pool: + id: minion_pool_1 + maximum_minions: 4 + minion_machines: + - id: machine_1 + allocation_status: AVAILABLE + action_instances: + instance_1: + name: Instance 1 + instance_2: + name: Instance 2 + instance_3: + name: Instance 3 + instance_4: + name: Instance 4 + expect: + result: + mappings: + instance_1: machine_1 + instance_2: new_machine + instance_3: new_machine + instance_4: new_machine \ No newline at end of file diff --git a/coriolis/tests/minion_manager/rpc/data/validate_minion_pool_selections_for_action_config.yaml b/coriolis/tests/minion_manager/rpc/data/validate_minion_pool_selections_for_action_config.yaml new file mode 100644 index 000000000..21cc7d80b --- /dev/null +++ b/coriolis/tests/minion_manager/rpc/data/validate_minion_pool_selections_for_action_config.yaml @@ -0,0 +1,352 @@ +# missing destination_endpoint_id +- config: + action: + id: "action-1" + origin_endpoint_id: "origin-endpoint-1" + origin_minion_pool_id: "origin-pool-1" + destination_minion_pool_id: "destination-pool-1" + instance_osmorphing_minion_pool_mappings: {} + instances: [] + expected_exception: InvalidInput + +# action must be a dict +- config: + action: 1 + expected_exception: InvalidInput + +# valid action +- config: + minion_pools: + - id: "origin-pool-1" + endpoint_id: "origin-endpoint-1" + platform: source + os_type: linux + status: ALLOCATED + maximum_minions: 2 + - id: "destination-pool-1" + endpoint_id: "destination-endpoint-1" + platform: destination + os_type: linux + status: ALLOCATED + maximum_minions: 3 + action: + id: "action-1" + origin_endpoint_id: "origin-endpoint-1" + destination_endpoint_id: "destination-endpoint-1" + origin_minion_pool_id: "origin-pool-1" + destination_minion_pool_id: "destination-pool-1" + instance_osmorphing_minion_pool_mappings: {} + instances: ["instance-1", "instance-2"] + expected_exception: ~ + +# could not find minion pool (1) +- config: + minion_pools: + - id: "invalid id" + action: + id: "action-1" + origin_endpoint_id: "origin-endpoint-1" + destination_endpoint_id: "destination-endpoint-1" + origin_minion_pool_id: "origin-pool-1" + destination_minion_pool_id: "destination-pool-1" + instance_osmorphing_minion_pool_mappings: {} + instances: ["instance-1", "instance-2"] + expected_exception: NotFound + +# could not find minion pool (2) +- config: + action: + id: "action-1" + origin_endpoint_id: "origin-endpoint-1" + destination_endpoint_id: "destination-endpoint-1" + origin_minion_pool_id: "origin-pool-1" + destination_minion_pool_id: "destination-pool-1" + instance_osmorphing_minion_pool_mappings: {} + instances: ["instance-1", "instance-2"] + expected_exception: NotFound + +# maxmimum minions is less than number of instances +- config: + minion_pools: + - id: "origin-pool-1" + endpoint_id: "origin-endpoint-1" + platform: source + os_type: linux + status: ALLOCATED + maximum_minions: 1 + - id: "destination-pool-1" + endpoint_id: "destination-endpoint-1" + platform: destination + os_type: linux + status: ALLOCATED + maximum_minions: 3 + action: + id: "action-1" + origin_endpoint_id: "origin-endpoint-1" + destination_endpoint_id: "destination-endpoint-1" + origin_minion_pool_id: "origin-pool-1" + destination_minion_pool_id: "destination-pool-1" + instance_osmorphing_minion_pool_mappings: {} + instances: ["instance-1", "instance-2"] + expected_exception: InvalidMinionPoolSelection + +# minion pool is not allocated +- config: + minion_pools: + - id: "origin-pool-1" + endpoint_id: "origin-endpoint-1" + platform: source + os_type: linux + status: ALLOCATED + maximum_minions: 2 + - id: "destination-pool-1" + endpoint_id: "destination-endpoint-1" + platform: destination + os_type: linux + status: DEALLOCATED + maximum_minions: 3 + action: + id: "action-1" + origin_endpoint_id: "origin-endpoint-1" + destination_endpoint_id: "destination-endpoint-1" + origin_minion_pool_id: "origin-pool-1" + destination_minion_pool_id: "destination-pool-1" + instance_osmorphing_minion_pool_mappings: {} + instances: ["instance-1", "instance-2"] + expected_exception: InvalidMinionPoolState + +## ORIGIN TESTS ## + +# origin endpoint id does not match with minion pool endpoint id +- config: + minion_pools: + - id: "origin-pool-1" + endpoint_id: "invalid id" + platform: source + os_type: linux + status: ALLOCATED + maximum_minions: 2 + - id: "destination-pool-1" + endpoint_id: "destination-endpoint-1" + platform: destination + os_type: linux + status: ALLOCATED + maximum_minions: 3 + action: + id: "action-1" + origin_endpoint_id: "origin-endpoint-1" + destination_endpoint_id: "destination-endpoint-1" + origin_minion_pool_id: "origin-pool-1" + destination_minion_pool_id: "destination-pool-1" + instance_osmorphing_minion_pool_mappings: {} + instances: ["instance-1", "instance-2"] + expected_exception: InvalidMinionPoolSelection + +# origin pool platform is not source +- config: + minion_pools: + - id: "origin-pool-1" + endpoint_id: "origin-endpoint-1" + platform: destination + os_type: linux + status: ALLOCATED + maximum_minions: 2 + - id: "destination-pool-1" + endpoint_id: "destination-endpoint-1" + platform: destination + os_type: linux + status: ALLOCATED + maximum_minions: 3 + action: + id: "action-1" + origin_endpoint_id: "origin-endpoint-1" + destination_endpoint_id: "destination-endpoint-1" + origin_minion_pool_id: "origin-pool-1" + destination_minion_pool_id: "destination-pool-1" + instance_osmorphing_minion_pool_mappings: {} + instances: ["instance-1", "instance-2"] + expected_exception: InvalidMinionPoolSelection + +# origin minion pool is not linux +- config: + minion_pools: + - id: "origin-pool-1" + endpoint_id: "origin-endpoint-1" + platform: source + os_type: windows + status: ALLOCATED + maximum_minions: 2 + - id: "destination-pool-1" + endpoint_id: "destination-endpoint-1" + platform: destination + os_type: linux + status: ALLOCATED + maximum_minions: 3 + action: + id: "action-1" + origin_endpoint_id: "origin-endpoint-1" + destination_endpoint_id: "destination-endpoint-1" + origin_minion_pool_id: "origin-pool-1" + destination_minion_pool_id: "destination-pool-1" + instance_osmorphing_minion_pool_mappings: {} + instances: ["instance-1", "instance-2"] + expected_exception: InvalidMinionPoolSelection + +## DESTINATION TESTS ## + +# destination endpoint id does not match with minion pool endpoint id +- config: + minion_pools: + - id: "origin-pool-1" + endpoint_id: "origin-endpoint-1" + platform: source + os_type: linux + status: ALLOCATED + maximum_minions: 2 + - id: "destination-pool-1" + endpoint_id: "invalid id" + platform: destination + os_type: linux + status: ALLOCATED + maximum_minions: 3 + action: + id: "action-1" + origin_endpoint_id: "origin-endpoint-1" + destination_endpoint_id: "destination-endpoint-1" + origin_minion_pool_id: "origin-pool-1" + destination_minion_pool_id: "destination-pool-1" + instance_osmorphing_minion_pool_mappings: {} + instances: ["instance-1", "instance-2"] + expected_exception: InvalidMinionPoolSelection + +# destination pool platform is not destination +- config: + minion_pools: + - id: "origin-pool-1" + endpoint_id: "origin-endpoint-1" + platform: source + os_type: linux + status: ALLOCATED + maximum_minions: 2 + - id: "destination-pool-1" + endpoint_id: "destination-endpoint-1" + platform: source + os_type: linux + status: ALLOCATED + maximum_minions: 3 + action: + id: "action-1" + origin_endpoint_id: "origin-endpoint-1" + destination_endpoint_id: "destination-endpoint-1" + origin_minion_pool_id: "origin-pool-1" + destination_minion_pool_id: "destination-pool-1" + instance_osmorphing_minion_pool_mappings: {} + instances: ["instance-1", "instance-2"] + expected_exception: InvalidMinionPoolSelection + +# destination minion pool is not linux +- config: + minion_pools: + - id: "origin-pool-1" + endpoint_id: "origin-endpoint-1" + platform: source + os_type: linux + status: ALLOCATED + maximum_minions: 2 + - id: "destination-pool-1" + endpoint_id: "destination-endpoint-1" + platform: destination + os_type: windows + status: ALLOCATED + maximum_minions: 3 + action: + id: "action-1" + origin_endpoint_id: "origin-endpoint-1" + destination_endpoint_id: "destination-endpoint-1" + origin_minion_pool_id: "origin-pool-1" + destination_minion_pool_id: "destination-pool-1" + instance_osmorphing_minion_pool_mappings: {} + instances: ["instance-1", "instance-2"] + expected_exception: InvalidMinionPoolSelection + +## OSMORPHING TESTS ## + +# valid action with os morphing mappings (1) +- config: + minion_pools: + - id: "origin-pool-1" + endpoint_id: "origin-endpoint-1" + platform: source + os_type: linux + status: ALLOCATED + maximum_minions: 2 + - id: "destination-pool-1" + endpoint_id: "destination-endpoint-1" + platform: destination + os_type: linux + status: ALLOCATED + maximum_minions: 3 + action: + id: "action-1" + origin_endpoint_id: "origin-endpoint-1" + destination_endpoint_id: "destination-endpoint-1" + origin_minion_pool_id: "origin-pool-1" + destination_minion_pool_id: "destination-pool-1" + instance_osmorphing_minion_pool_mappings: + instance-1: "destination-pool-1" + instances: ["instance-1", "instance-2"] + expected_exception: ~ + +# valid action with os morphing mappings (2) +- config: + minion_pools: + - id: "origin-pool-1" + endpoint_id: "origin-endpoint-1" + platform: source + os_type: linux + status: ALLOCATED + maximum_minions: 2 + - id: "destination-pool-1" + endpoint_id: "destination-endpoint-1" + platform: destination + os_type: linux + status: ALLOCATED + maximum_minions: 3 + action: + id: "action-1" + origin_endpoint_id: "origin-endpoint-1" + destination_endpoint_id: "destination-endpoint-1" + origin_minion_pool_id: "origin-pool-1" + destination_minion_pool_id: "destination-pool-1" + instance_osmorphing_minion_pool_mappings: + instance-1: "destination-pool-1" + instance-2: "destination-pool-1" + instances: ["instance-1", "instance-2"] + expected_exception: ~ + +# pool belongs to origin endpoint +- config: + minion_pools: + - id: "origin-pool-1" + endpoint_id: "origin-endpoint-1" + platform: source + os_type: linux + status: ALLOCATED + maximum_minions: 2 + - id: "destination-pool-1" + endpoint_id: "destination-endpoint-1" + platform: destination + os_type: linux + status: ALLOCATED + maximum_minions: 3 + action: + id: "action-1" + origin_endpoint_id: "origin-endpoint-1" + destination_endpoint_id: "destination-endpoint-1" + origin_minion_pool_id: "origin-pool-1" + destination_minion_pool_id: "destination-pool-1" + instance_osmorphing_minion_pool_mappings: + instance-1: "origin-pool-1" + instance-2: "destination-pool-1" + instances: ["instance-1", "instance-2"] + expected_exception: InvalidMinionPoolSelection \ No newline at end of file diff --git a/coriolis/tests/minion_manager/rpc/test_server.py b/coriolis/tests/minion_manager/rpc/test_server.py new file mode 100644 index 000000000..a93826e28 --- /dev/null +++ b/coriolis/tests/minion_manager/rpc/test_server.py @@ -0,0 +1,201 @@ +# Copyright 2023 Cloudbase Solutions Srl +# All Rights Reserved. + +from unittest import mock + +import ddt +import uuid + +from coriolis import constants +from coriolis import exception +from coriolis.db import api as db_api +from coriolis.minion_manager.rpc import server +from coriolis.tests import test_base +from coriolis.tests import testutils + +@ddt.ddt +class MinionManagerServerEndpointTestCase(test_base.CoriolisBaseTestCase): + """Test suite for the Coriolis Minion Manager RPC server.""" + + def setUp(self, *_, **__): + super(MinionManagerServerEndpointTestCase, self).setUp() + self.server = server.MinionManagerServerEndpoint() + + @mock.patch.object(db_api, "get_minion_pools") + @ddt.file_data( + "data/validate_minion_pool_selections_for_action_config.yaml" + ) + @ddt.unpack + def test_validate_minion_pool_selections_for_action( + self, + mock_get_minion_pools, + config, + expected_exception, + ): + action = config.get("action") + minion_pools = config.get("minion_pools", []) + + mock_get_minion_pools.return_value = [ + mock.MagicMock( + **pool, + ) for pool in minion_pools + ] + + if expected_exception: + exception_type = getattr(exception, expected_exception) + self.assertRaises( + exception_type, + self.server.validate_minion_pool_selections_for_action, + mock.sentinel.context, + action, + ) + return + + self.server.validate_minion_pool_selections_for_action( + mock.sentinel.context, + action, + ) + + @mock.patch.object(uuid, "uuid4", return_value="new_machine") + @mock.patch.object( + server.MinionManagerServerEndpoint, + "_add_minion_pool_event") + @mock.patch.object(db_api, "add_minion_machine") + @mock.patch.object(db_api, "set_minion_machines_allocation_statuses") + @ddt.file_data( + "data/make_minion_machine_allocation_subflow_for_action.yaml" + ) + @ddt.unpack + def test_make_minion_machine_allocation_subflow_for_action( + self, + mock_set_minion_machines_allocation_statuses, + mock_add_minion_machine, + mock_add_minion_pool_event, + mock_uuid4, + config, + expect + ): + expected_exception = expect.get("exception") + expected_result = expect.get("result") + + minion_pool = testutils.DictToObject(config.get("minion_pool")) + action_instances = config.get("action_instances") + + args = [ + mock.sentinel.context, + minion_pool, + mock.sentinel.action_id, + action_instances, + mock.sentinel.subflow_name, + ] + + if expected_exception: + exception_type = getattr(exception, expected_exception) + self.assertRaises( + exception_type, + self.server._make_minion_machine_allocation_subflow_for_action, + *args) + return + + result = self.server\ + ._make_minion_machine_allocation_subflow_for_action( + *args) + + mappings = expected_result.get("mappings") + + num_new_machines = list(mappings.values()).count("new_machine") + + # db_api.add_minion_machine is called once for each new machine + self.assertEqual( + num_new_machines, + mock_add_minion_machine.call_count) + + num_non_new_machines = len(mappings) - num_new_machines + if num_non_new_machines: + # db_api.set_minion_machines_allocation_statuses is called once + # with the non-new machines + mock_set_minion_machines_allocation_statuses\ + .assert_called_once_with( + mock.sentinel.context, + list(mappings.values())[:num_non_new_machines], + mock.sentinel.action_id, + constants.MINION_MACHINE_STATUS_RESERVED, + refresh_allocation_time=True) + + self.assertEqual( + mappings, + result.get("action_instance_minion_allocation_mappings")) + + @mock.patch.object(db_api, "set_minion_machine_allocation_status") + @mock.patch.object( + server.MinionManagerServerEndpoint, + "_add_minion_pool_event") + @mock.patch.object( + server.MinionManagerServerEndpoint, + "_get_minion_pool") + @ddt.file_data( + "data/get_minion_pool_refresh_flow.yaml" + ) + @ddt.unpack + def test_get_minion_pool_refresh_flow( + self, + mock_get_minion_pool, + mock_add_minion_pool_event, + mock_set_minion_machine_allocation_status, + config, + expect, + ): + minion_pool = testutils.DictToObject(config.get("minion_pool"), {}) + expected_result = expect.get("result", {}) + expect_exception = expect.get("exception") + exptected_flow_tasks = expected_result.get("flow_tasks", []) + expected_db_calls = expected_result.get("db_calls", {}) + expected_db_calls_include = expected_db_calls.get("include", []) + expected_db_calls_exclude = expected_db_calls.get("exclude", []) + + mock_get_minion_pool.return_value = minion_pool + + if expect_exception: + exception_type = getattr(exception, expect_exception) + self.assertRaises( + exception_type, + self.server._get_minion_pool_refresh_flow, + mock.sentinel.context, + minion_pool, + requery=False, + ) + return + + flow = self.server._get_minion_pool_refresh_flow( + mock.sentinel.context, + minion_pool, + requery=False, + ) + + # Test the flow returned by the function + flow_tasks = [node.name for node, _ in flow.iter_nodes()] + self.assertEqual( + exptected_flow_tasks, + flow_tasks) + + # Test DB calls that should be made + for call in expected_db_calls_include: + for method, args in call.items(): + if method == "set_minion_machine_allocation_status": + mock_set_minion_machine_allocation_status\ + .assert_any_call( + mock.sentinel.context, + args.get("id"), + args.get("allocation_status")) + + # Test DB calls that should not be made + for call in expected_db_calls_exclude: + for method, args in call.items(): + if method == "set_minion_machine_allocation_status": + assert mock.call( + mock.sentinel.context, + args.get("id"), + args.get("allocation_status"))\ + not in mock_set_minion_machine_allocation_status\ + .mock_calls, f"Unexpected call to {method}, " \ + f"args: {args}" diff --git a/coriolis/tests/testutils.py b/coriolis/tests/testutils.py index f98d914df..c822e3a07 100644 --- a/coriolis/tests/testutils.py +++ b/coriolis/tests/testutils.py @@ -34,3 +34,36 @@ def _get_wrapped_function(function): return function return _get_wrapped_function(function) + + +class DictToObject: + """Converts a dictionary to an object with attributes. + + This is useful for mocking objects that are used as configuration + objects. + """ + + def __init__(self, dictionary, skip_attrs=None): + if skip_attrs is None: + skip_attrs = [] + + for key, value in dictionary.items(): + if key in skip_attrs: + setattr(self, key, value) + elif isinstance(value, dict): + setattr(self, key, DictToObject(value, skip_attrs=skip_attrs)) + elif isinstance(value, list): + setattr( + self, key, + [DictToObject(item, skip_attrs=skip_attrs) if isinstance( + item, dict) else item for item in value]) + else: + setattr(self, key, value) + + def __getattr__(self, item): + return None + + def __repr__(self): + attrs = [f"{k}={v!r}" for k, v in self.__dict__.items()] + attrs_str = ', '.join(attrs) + return f"{self.__class__.__name__}({attrs_str})"