senlin/senlin/tests/unit/engine/service/test_actions.py

242 lines
9.7 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.
import mock
from oslo_messaging.rpc import dispatcher as rpc
import six
from senlin.common import consts
from senlin.common import exception as exc
from senlin.engine.actions import base as ab
from senlin.engine import service
from senlin.objects import action as ao
from senlin.objects import cluster as co
from senlin.objects.requests import actions as orao
from senlin.tests.unit.common import base
from senlin.tests.unit.common import utils
class ActionTest(base.SenlinTestCase):
def setUp(self):
super(ActionTest, self).setUp()
self.ctx = utils.dummy_context(project='action_test_project')
self.eng = service.EngineService('host-a', 'topic-a')
self.eng.init_tgm()
@mock.patch.object(ao.Action, 'get_all')
def test_action_list(self, mock_get):
x_1 = mock.Mock()
x_1.to_dict.return_value = {'k': 'v1'}
x_2 = mock.Mock()
x_2.to_dict.return_value = {'k': 'v2'}
mock_get.return_value = [x_1, x_2]
req = orao.ActionListRequest()
result = self.eng.action_list(self.ctx, req.obj_to_primitive())
expected = [{'k': 'v1'}, {'k': 'v2'}]
self.assertEqual(expected, result)
mock_get.assert_called_once_with(self.ctx, project_safe=True)
@mock.patch.object(ao.Action, 'get_all')
def test_action_list_with_params(self, mock_get):
x_1 = mock.Mock()
x_1.to_dict.return_value = {'status': 'READY'}
x_2 = mock.Mock()
x_2.to_dict.return_value = {'status': 'SUCCESS'}
mock_get.return_value = [x_1, x_2]
req = orao.ActionListRequest(status=['READY', 'SUCCEEDED'],
limit=100,
sort='status',
project_safe=True)
result = self.eng.action_list(self.ctx, req.obj_to_primitive())
expected = [{'status': 'READY'}, {'status': 'SUCCESS'}]
self.assertEqual(expected, result)
filters = {'status': ['READY', 'SUCCEEDED']}
mock_get.assert_called_once_with(self.ctx,
filters=filters,
limit=100,
sort='status',
project_safe=True
)
def test_action_list_with_bad_params(self):
req = orao.ActionListRequest(project_safe=False)
ex = self.assertRaises(rpc.ExpectedException,
self.eng.action_list,
self.ctx, req.obj_to_primitive())
self.assertEqual(exc.Forbidden, ex.exc_info[0])
@mock.patch.object(ao.Action, 'get_all')
def test_action_list_with_Auth(self, mock_get):
mock_get.return_value = []
req = orao.ActionListRequest(project_safe=True)
result = self.eng.action_list(self.ctx, req.obj_to_primitive())
self.assertEqual([], result)
mock_get.assert_called_once_with(self.ctx, project_safe=True)
self.ctx.is_admin = True
mock_get.reset_mock()
req = orao.ActionListRequest(project_safe=True)
result = self.eng.action_list(self.ctx, req.obj_to_primitive())
self.assertEqual([], result)
mock_get.assert_called_once_with(self.ctx, project_safe=True)
mock_get.reset_mock()
req = orao.ActionListRequest(project_safe=False)
result = self.eng.action_list(self.ctx, req.obj_to_primitive())
self.assertEqual([], result)
mock_get.assert_called_once_with(self.ctx, project_safe=False)
@mock.patch.object(ab.Action, 'create')
@mock.patch.object(co.Cluster, 'find')
def test_action_create(self, mock_find, mock_action):
mock_find.return_value = mock.Mock(id='FAKE_CLUSTER')
mock_action.return_value = 'ACTION_ID'
req = orao.ActionCreateRequestBody(name='a1', cluster_id='C1',
action='CLUSTER_CREATE')
result = self.eng.action_create(self.ctx, req.obj_to_primitive())
self.assertEqual({'action': 'ACTION_ID'}, result)
mock_find.assert_called_once_with(self.ctx, 'C1')
mock_action.assert_called_once_with(
self.ctx, 'FAKE_CLUSTER', 'CLUSTER_CREATE',
name='a1',
cause=consts.CAUSE_RPC,
status=ab.Action.READY,
inputs={})
@mock.patch.object(co.Cluster, 'find')
def test_action_create_cluster_not_found(self, mock_find):
mock_find.side_effect = exc.ResourceNotFound(type='cluster', id='C1')
req = orao.ActionCreateRequestBody(name='NODE1',
cluster_id='C1')
ex = self.assertRaises(rpc.ExpectedException,
self.eng.action_create,
self.ctx, req.obj_to_primitive())
self.assertEqual(exc.BadRequest, ex.exc_info[0])
self.assertEqual("Cannot find the given cluster: C1.",
six.text_type(ex.exc_info[1]))
mock_find.assert_called_once_with(self.ctx, 'C1')
@mock.patch.object(ao.Action, 'find')
def test_action_get(self, mock_find):
x_obj = mock.Mock()
mock_find.return_value = x_obj
x_obj.to_dict.return_value = {'k': 'v'}
req = orao.ActionGetRequest(identity='ACTION_ID')
result = self.eng.action_get(self.ctx, req.obj_to_primitive())
self.assertEqual({'k': 'v'}, result)
mock_find.assert_called_once_with(self.ctx, 'ACTION_ID')
@mock.patch.object(ao.Action, 'find')
def test_action_get_not_found(self, mock_find):
mock_find.side_effect = exc.ResourceNotFound(type='action', id='Bogus')
req = orao.ActionGetRequest(identity='Bogus')
ex = self.assertRaises(rpc.ExpectedException,
self.eng.action_get,
self.ctx, req.obj_to_primitive())
self.assertEqual(exc.ResourceNotFound, ex.exc_info[0])
mock_find.assert_called_once_with(self.ctx, 'Bogus')
@mock.patch.object(ab.Action, 'delete')
@mock.patch.object(ao.Action, 'find')
def test_action_delete(self, mock_find, mock_delete):
x_obj = mock.Mock()
x_obj.id = 'FAKE_ID'
mock_find.return_value = x_obj
mock_delete.return_value = None
req = orao.ActionDeleteRequest(identity='ACTION_ID')
result = self.eng.action_delete(self.ctx, req.obj_to_primitive())
self.assertIsNone(result)
mock_find.assert_called_once_with(self.ctx, 'ACTION_ID')
mock_delete.assert_called_once_with(self.ctx, 'FAKE_ID')
@mock.patch.object(ab.Action, 'delete')
@mock.patch.object(ao.Action, 'find')
def test_action_delete_resource_busy(self, mock_find, mock_delete):
x_obj = mock.Mock()
x_obj.id = 'FAKE_ID'
mock_find.return_value = x_obj
ex = exc.EResourceBusy(type='action', id='FAKE_ID')
mock_delete.side_effect = ex
req = orao.ActionDeleteRequest(identity='ACTION_ID')
ex = self.assertRaises(rpc.ExpectedException,
self.eng.action_delete,
self.ctx, req.obj_to_primitive())
self.assertEqual(exc.ResourceInUse, ex.exc_info[0])
self.assertEqual("The action 'ACTION_ID' cannot be deleted: still "
"in one of WAITING, RUNNING or SUSPENDED state.",
six.text_type(ex.exc_info[1]))
mock_find.assert_called_once_with(self.ctx, 'ACTION_ID')
mock_delete.assert_called_once_with(self.ctx, 'FAKE_ID')
@mock.patch.object(ao.Action, 'find')
def test_action_delete_not_found(self, mock_find):
mock_find.side_effect = exc.ResourceNotFound(type='action', id='Bogus')
req = orao.ActionDeleteRequest(identity='ACTION_ID')
ex = self.assertRaises(rpc.ExpectedException,
self.eng.action_delete,
self.ctx, req.obj_to_primitive())
self.assertEqual(exc.ResourceNotFound, ex.exc_info[0])
@mock.patch.object(ab.Action, 'load')
def test_action_update(self, mock_load):
x_obj = mock.Mock()
x_obj.id = 'FAKE_ID'
x_obj.signal_cancel = mock.Mock()
x_obj.SIG_CANCEL = 'CANCEL'
mock_load.return_value = x_obj
req = orao.ActionUpdateRequest(identity='ACTION_ID',
status='CANCELLED')
result = self.eng.action_update(self.ctx, req.obj_to_primitive())
self.assertIsNone(result)
mock_load.assert_called_with(self.ctx, 'ACTION_ID', project_safe=False)
x_obj.signal_cancel.assert_called_once_with()
@mock.patch.object(ab.Action, 'load')
def test_action_update_unknown_action(self, mock_load):
x_obj = mock.Mock()
x_obj.id = 'FAKE_ID'
x_obj.signal_cancel = mock.Mock()
mock_load.return_value = x_obj
req = orao.ActionUpdateRequest(identity='ACTION_ID',
status='FOO')
ex = self.assertRaises(rpc.ExpectedException, self.eng.action_update,
self.ctx, req.obj_to_primitive())
self.assertEqual(exc.BadRequest, ex.exc_info[0])
mock_load.assert_not_called()
x_obj.signal_cancel.assert_not_called()