senlin/senlin/tests/unit/engine/actions/test_replace_nodes.py

297 lines
12 KiB
Python

# Licensed under the Apache License, Version 2.0 (the "License"); you may
# not use this file except in compliance with the License. You may obtain
# a copy of the License at
#
# http://www.apache.org/licenses/LICENSE-2.0
#
# Unless required by applicable law or agreed to in writing, software
# distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
# WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
# License for the specific language governing permissions and limitations
# under the License.
from unittest import mock
from senlin.common import consts
from senlin.engine.actions import base as ab
from senlin.engine.actions import cluster_action as ca
from senlin.engine import cluster as cm
from senlin.engine import dispatcher
from senlin.objects import action as ao
from senlin.objects import dependency as dobj
from senlin.objects import node as no
from senlin.tests.unit.common import base
from senlin.tests.unit.common import utils
@mock.patch.object(cm.Cluster, 'load')
class ClusterReplaceNodesTest(base.SenlinTestCase):
def setUp(self):
super(ClusterReplaceNodesTest, self).setUp()
self.ctx = utils.dummy_context()
@mock.patch.object(ao.Action, 'update')
@mock.patch.object(ab.Action, 'create')
@mock.patch.object(no.Node, 'get')
@mock.patch.object(dobj.Dependency, 'create')
@mock.patch.object(dispatcher, 'start_action')
@mock.patch.object(ca.ClusterAction, '_wait_for_dependents')
def test_do_replace_nodes(self, mock_wait, mock_start, mock_dep,
mock_get_node, mock_action, mock_update,
mock_load):
cluster = mock.Mock(id='CLUSTER_ID', desired_capacity=10)
mock_load.return_value = cluster
action = ca.ClusterAction(cluster.id, 'CLUSTER_ACTION', self.ctx)
action.id = 'CLUSTER_ACTION_ID'
action.inputs = {'candidates': {'O_NODE_1': 'R_NODE_1'}, 'foo': 'bar'}
action.outputs = {}
origin_node = mock.Mock(id='O_NODE_1', cluster_id='CLUSTER_ID',
ACTIVE='ACTIVE', status='ACTIVE')
replace_node = mock.Mock(id='R_NODE_1', cluster_id='',
ACTIVE='ACTIVE', status='ACTIVE')
mock_get_node.side_effect = [origin_node, replace_node]
mock_action.side_effect = ['NODE_LEAVE_1', 'NODE_JOIN_1']
mock_wait.return_value = (action.RES_OK, 'Free to fly!')
# do the action
res_code, res_msg = action.do_replace_nodes()
# assertions
self.assertEqual(action.RES_OK, res_code)
self.assertEqual('Completed replacing nodes.', res_msg)
mock_get_node.assert_has_calls([
mock.call(action.context, 'O_NODE_1'),
mock.call(action.context, 'R_NODE_1')])
mock_load.assert_called_once_with(
action.context,
'CLUSTER_ID')
mock_action.assert_has_calls([
mock.call(action.context, 'O_NODE_1', 'NODE_LEAVE',
name='node_leave_O_NODE_1',
cluster_id='CLUSTER_ID',
cause='Derived Action'),
mock.call(action.context, 'R_NODE_1', 'NODE_JOIN',
name='node_join_R_NODE_1',
cluster_id='CLUSTER_ID',
cause='Derived Action',
inputs={'cluster_id': 'CLUSTER_ID'})])
mock_dep.assert_has_calls([
mock.call(action.context,
['NODE_JOIN_1'],
'CLUSTER_ACTION_ID'),
mock.call(action.context,
['NODE_JOIN_1'],
'NODE_LEAVE_1')])
mock_update.assert_has_calls([
mock.call(action.context,
'NODE_JOIN_1',
{'status': 'READY'}),
mock.call(action.context,
'NODE_LEAVE_1',
{'status': 'READY'})])
mock_start.assert_called_once_with()
mock_wait.assert_called_once_with()
cluster.remove_node.assert_called_once_with(origin_node)
cluster.add_node.assert_called_once_with(replace_node)
cluster.eval_status.assert_called_once_with(
action.context, consts.CLUSTER_REPLACE_NODES)
@mock.patch.object(no.Node, 'get')
def test_do_replace_nodes_original_not_found(self, mock_get_node,
mock_load):
action = ca.ClusterAction('ID', 'CLUSTER_ACTION', self.ctx)
action.inputs = {'candidates': {'ORIGIN_NODE': 'REPLACE_NODE'}}
origin_node = None
replace_node = mock.Mock(id='REPLACE_NODE', cluster_id='',
ACTIVE='ACTIVE', status='ACTIVE')
mock_get_node.side_effect = [origin_node, replace_node]
# do the action
res_code, res_msg = action.do_replace_nodes()
# assertions
self.assertEqual(action.RES_ERROR, res_code)
self.assertEqual('Original node ORIGIN_NODE not found.',
res_msg)
def test_do_replace_nodes_empty_candidates(self, mock_load):
action = ca.ClusterAction('ID', 'CLUSTER_ACTION', self.ctx)
action.inputs = {'candidates': {}}
res_code, res_msg = action.do_replace_nodes()
self.assertEqual(action.RES_ERROR, res_code)
self.assertEqual(
'Candidates must be a non-empty dict. Instead got {}',
res_msg)
@mock.patch.object(no.Node, 'get')
def test_do_replace_nodes_replacement_not_found(self, mock_get_node,
mock_load):
action = ca.ClusterAction('ID', 'CLUSTER_ACTION', self.ctx)
action.inputs = {'candidates': {'ORIGIN_NODE': 'REPLACE_NODE'}}
origin_node = mock.Mock(id='ORIGIN_NODE', cluster_id='CLUSTER_ID',
ACTIVE='ACTIVE', status='ACTIVE')
replace_node = None
mock_get_node.side_effect = [origin_node, replace_node]
# do the action
res_code, res_msg = action.do_replace_nodes()
# assertions
self.assertEqual(action.RES_ERROR, res_code)
self.assertEqual('Replacement node REPLACE_NODE not found.',
res_msg)
@mock.patch.object(no.Node, 'get')
def test_do_replace_nodes_not_a_member(self, mock_get_node,
mock_load):
cluster = mock.Mock(id='FAKE_CLUSTER')
mock_load.return_value = cluster
action = ca.ClusterAction(cluster.id, 'CLUSTER_ACTION', self.ctx)
action.inputs = {'candidates': {'ORIGIN_NODE': 'REPLACE_NODE'}}
origin_node = mock.Mock(id='ORIGIN_NODE', cluster_id='')
mock_get_node.return_value = origin_node
# do action
res_code, res_msg = action.do_replace_nodes()
# assertions
self.assertEqual(action.RES_ERROR, res_code)
self.assertEqual('Node ORIGIN_NODE is not a member of the '
'cluster FAKE_CLUSTER.', res_msg)
@mock.patch.object(no.Node, 'get')
def test_do_replace_nodes_node_already_member(self, mock_get_node,
mock_load):
cluster = mock.Mock(id='FAKE_CLUSTER')
mock_load.return_value = cluster
action = ca.ClusterAction(cluster.id, 'CLUSTER_ACTION', self.ctx)
action.inputs = {'candidates': {'ORIGIN_NODE': 'REPLACE_NODE'}}
replace_node = mock.Mock(id='REPLACE_NODE',
cluster_id='FAKE_CLUSTER')
mock_get_node.return_value = replace_node
# do it
res_code, res_msg = action.do_replace_nodes()
# assertions
self.assertEqual(action.RES_ERROR, res_code)
self.assertEqual('Node REPLACE_NODE is already owned by cluster '
'FAKE_CLUSTER.', res_msg)
@mock.patch.object(no.Node, 'get')
def test_do_replace_nodes_in_other_cluster(self, mock_get_node,
mock_load):
cluster = mock.Mock(id='CLUSTER_ID', desired_capacity=10)
mock_load.return_value = cluster
action = ca.ClusterAction(cluster.id, 'CLUSTER_ACTION', self.ctx)
action.id = 'CLUSTER_ACTION_ID'
action.inputs = {'candidates': {'ORIGIN_NODE': 'REPLACE_NODE'}}
action.outputs = {}
origin_node = mock.Mock(id='ORIGIN_NODE', cluster_id='CLUSTER_ID',
ACTIVE='ACTIVE', status='ACTIVE')
replace_node = mock.Mock(id='REPLACE_NODE', cluster_id='FAKE_CLUSTER',
ACTIVE='ACTIVE', status='ACTIVE')
mock_get_node.side_effect = [origin_node, replace_node]
# do it
res_code, res_msg = action.do_replace_nodes()
# assertions
self.assertEqual(action.RES_ERROR, res_code)
self.assertEqual('Node REPLACE_NODE is already owned by cluster '
'FAKE_CLUSTER.', res_msg)
@mock.patch.object(no.Node, 'get')
def test_do_replace_nodes_node_not_active(self, mock_get_node, mock_load):
cluster = mock.Mock(id='CLUSTER_ID', desired_capacity=10)
mock_load.return_value = cluster
action = ca.ClusterAction(cluster.id, 'CLUSTER_ACTION', self.ctx)
action.id = 'CLUSTER_ACTION_ID'
action.inputs = {'candidates': {'ORIGIN_NODE': 'REPLACE_NODE'}}
action.outputs = {}
origin_node = mock.Mock(id='ORIGIN_NODE', cluster_id='CLUSTER_ID',
ACTIVE='ACTIVE', status='ACTIVE')
replace_node = mock.Mock(id='REPLACE_NODE', cluster_id='',
ACTIVE='ACTIVE', status='ERROR')
mock_get_node.side_effect = [origin_node, replace_node]
# do it
res_code, res_msg = action.do_replace_nodes()
# assertions
self.assertEqual(action.RES_ERROR, res_code)
self.assertEqual("Node REPLACE_NODE is not in ACTIVE status.", res_msg)
@mock.patch.object(ao.Action, 'update')
@mock.patch.object(ab.Action, 'create')
@mock.patch.object(no.Node, 'get')
@mock.patch.object(dobj.Dependency, 'create')
@mock.patch.object(dispatcher, 'start_action')
@mock.patch.object(ca.ClusterAction, '_wait_for_dependents')
def test_do_replace_failed_waiting(self, mock_wait, mock_start, mock_dep,
mock_get_node, mock_action,
mock_update, mock_load):
cluster = mock.Mock(id='CLUSTER_ID', desired_capacity=10)
mock_load.return_value = cluster
action = ca.ClusterAction(cluster.id, 'CLUSTER_ACTION', self.ctx)
action.id = 'CLUSTER_ACTION_ID'
action.inputs = {'candidates': {'O_NODE_1': 'R_NODE_1'}}
action.outputs = {}
origin_node = mock.Mock(id='O_NODE_1', cluster_id='CLUSTER_ID',
ACTIVE='ACTIVE', status='ACTIVE')
replace_node = mock.Mock(id='R_NODE_1', cluster_id='',
ACTIVE='ACTIVE', status='ACTIVE')
mock_get_node.side_effect = [origin_node, replace_node]
mock_action.side_effect = ['NODE_LEAVE_1', 'NODE_JOIN_1']
mock_wait.return_value = (action.RES_TIMEOUT, 'Timeout!')
# do the action
res_code, res_msg = action.do_replace_nodes()
# assertions
mock_action.assert_has_calls([
mock.call(action.context, 'O_NODE_1', 'NODE_LEAVE',
name='node_leave_O_NODE_1',
cluster_id='CLUSTER_ID',
cause='Derived Action'),
mock.call(action.context, 'R_NODE_1', 'NODE_JOIN',
name='node_join_R_NODE_1',
cluster_id='CLUSTER_ID',
cause='Derived Action',
inputs={'cluster_id': 'CLUSTER_ID'})])
mock_dep.assert_has_calls([
mock.call(action.context,
['NODE_JOIN_1'],
'CLUSTER_ACTION_ID'),
mock.call(action.context,
['NODE_JOIN_1'],
'NODE_LEAVE_1')])
mock_update.assert_has_calls([
mock.call(action.context,
'NODE_JOIN_1',
{'status': 'READY'}),
mock.call(action.context,
'NODE_LEAVE_1',
{'status': 'READY'})])
self.assertEqual(action.RES_TIMEOUT, res_code)
self.assertEqual('Timeout!', res_msg)