Support action_execution deletion

With this patch, users can perform operation as following:
DELETE http://127.0.0.1:8989/v2/action_executions/<action_execution_id>

NOTE:
* Added a new config item 'allow_action_execution_deletion', with default
  value 'False'.
* The 'allow_action_execution_deletion' item is set to 'True' for
  functional testing.
* Only ad-hoc action-execution can be deleted.
* Only completed action execution can be deleted.

Implements: blueprint mistral-action-execution-deletion
Closes-Bug: #1488157
Change-Id: I3729636911a42c273c5a7b2d7fbdaae0da398e31
This commit is contained in:
Lingxian Kong 2015-08-25 10:54:51 +08:00 committed by Renat Akhmerov
parent 7a8c29a2f1
commit f2eaea5a12
8 changed files with 198 additions and 34 deletions

View File

@ -101,6 +101,7 @@ function configure_mistral {
iniset $MISTRAL_CONF_FILE keystone_authtoken admin_tenant_name $SERVICE_TENANT_NAME iniset $MISTRAL_CONF_FILE keystone_authtoken admin_tenant_name $SERVICE_TENANT_NAME
iniset $MISTRAL_CONF_FILE keystone_authtoken admin_user $MISTRAL_ADMIN_USER iniset $MISTRAL_CONF_FILE keystone_authtoken admin_user $MISTRAL_ADMIN_USER
iniset $MISTRAL_CONF_FILE keystone_authtoken admin_password $SERVICE_PASSWORD iniset $MISTRAL_CONF_FILE keystone_authtoken admin_password $SERVICE_PASSWORD
iniset $MISTRAL_CONF_FILE keystone_authtoken auth_uri "http://${KEYSTONE_AUTH_HOST}:5000/v3"
# Setup RabbitMQ credentials # Setup RabbitMQ credentials
iniset $MISTRAL_CONF_FILE DEFAULT rabbit_userid $RABBIT_USERID iniset $MISTRAL_CONF_FILE DEFAULT rabbit_userid $RABBIT_USERID
@ -111,8 +112,8 @@ function configure_mistral {
iniset $MISTRAL_CONF_FILE database max_overflow -1 iniset $MISTRAL_CONF_FILE database max_overflow -1
iniset $MISTRAL_CONF_FILE database max_pool_size 1000 iniset $MISTRAL_CONF_FILE database max_pool_size 1000
# Configure keystone auth url # Configure action execution deletion policy
iniset $MISTRAL_CONF_FILE keystone_authtoken auth_uri "http://${KEYSTONE_AUTH_HOST}:5000/v3" iniset $MISTRAL_CONF_FILE api allow_action_execution_deletion True
} }

View File

@ -14,6 +14,7 @@
# See the License for the specific language governing permissions and # See the License for the specific language governing permissions and
# limitations under the License. # limitations under the License.
from oslo_config import cfg
from oslo_log import log as logging from oslo_log import log as logging
from pecan import rest from pecan import rest
from wsme import types as wtypes from wsme import types as wtypes
@ -191,6 +192,29 @@ class ActionExecutionsController(rest.RestController):
return _get_action_executions() return _get_action_executions()
@rest_utils.wrap_wsme_controller_exception
@wsme_pecan.wsexpose(None, wtypes.text, status_code=204)
def delete(self, id):
"""Delete the specified action_execution."""
LOG.info("Delete action_execution [id=%s]" % id)
if not cfg.CONF.api.allow_action_execution_deletion:
raise exc.NotAllowedException("Action execution deletion is not "
"allowed.")
action_ex = db_api.get_action_execution(id)
if action_ex.task_execution_id:
raise exc.NotAllowedException("Only ad-hoc action execution can "
"be deleted.")
if not states.is_completed(action_ex.state):
raise exc.NotAllowedException("Only completed action execution "
"can be deleted.")
return db_api.delete_action_execution(id)
class TasksActionExecutionController(rest.RestController): class TasksActionExecutionController(rest.RestController):
@wsme_pecan.wsexpose(ActionExecutions, wtypes.text) @wsme_pecan.wsexpose(ActionExecutions, wtypes.text)

View File

@ -36,7 +36,10 @@ launch_opt = cfg.ListOpt(
api_opts = [ api_opts = [
cfg.StrOpt('host', default='0.0.0.0', help='Mistral API server host'), cfg.StrOpt('host', default='0.0.0.0', help='Mistral API server host'),
cfg.IntOpt('port', default=8989, help='Mistral API server port') cfg.IntOpt('port', default=8989, help='Mistral API server port'),
cfg.BoolOpt('allow_action_execution_deletion', default=False,
help='Enables the ability to delete action_execution which '
'has no relationship with workflows.'),
] ]
pecan_opts = [ pecan_opts = [

View File

@ -127,3 +127,8 @@ class SizeLimitExceededException(MistralException):
class CoordinationException(MistralException): class CoordinationException(MistralException):
http_code = 500 http_code = 500
class NotAllowedException(MistralException):
http_code = 403
message = "Operation not allowed"

View File

@ -14,6 +14,8 @@
import json import json
from oslo_log import log as logging
import six
from tempest import test from tempest import test
from tempest_lib import decorators from tempest_lib import decorators
from tempest_lib import exceptions from tempest_lib import exceptions
@ -22,6 +24,9 @@ from mistral.tests.functional import base
from mistral import utils from mistral import utils
LOG = logging.getLogger(__name__)
class WorkbookTestsV2(base.TestCase): class WorkbookTestsV2(base.TestCase):
_service = 'workflowv2' _service = 'workflowv2'
@ -976,15 +981,26 @@ class TasksTestsV2(base.TestCase):
) )
# TODO(namkhotkin) Need more tests on action executions.
class ActionExecutionTestsV2(base.TestCase): class ActionExecutionTestsV2(base.TestCase):
_service = 'workflowv2' _service = 'workflowv2'
@classmethod
def resource_cleanup(cls):
for action_ex in cls.client.action_executions:
try:
cls.client.delete_obj('action_executions', action_ex)
except Exception as e:
LOG.exception('Exception raised when deleting '
'action_executions %s, error message: %s.'
% (action_ex, six.text_type(e)))
cls.client.action_executions = []
super(ActionExecutionTestsV2, cls).resource_cleanup()
@test.attr(type='sanity') @test.attr(type='sanity')
def test_run_action_execution(self): def test_run_action_execution(self):
resp, body = self.client.post_json( resp, body = self.client.create_action_execution(
'action_executions',
{ {
'name': 'std.echo', 'name': 'std.echo',
'input': '{"output": "Hello, Mistral!"}' 'input': '{"output": "Hello, Mistral!"}'
@ -992,7 +1008,6 @@ class ActionExecutionTestsV2(base.TestCase):
) )
self.assertEqual(201, resp.status) self.assertEqual(201, resp.status)
body = json.loads(body)
output = json.loads(body['output']) output = json.loads(body['output'])
self.assertDictEqual( self.assertDictEqual(
{'result': 'Hello, Mistral!'}, {'result': 'Hello, Mistral!'},
@ -1001,8 +1016,7 @@ class ActionExecutionTestsV2(base.TestCase):
@test.attr(type='sanity') @test.attr(type='sanity')
def test_run_action_std_http(self): def test_run_action_std_http(self):
resp, body = self.client.post_json( resp, body = self.client.create_action_execution(
'action_executions',
{ {
'name': 'std.http', 'name': 'std.http',
'input': '{"url": "http://wiki.openstack.org"}' 'input': '{"url": "http://wiki.openstack.org"}'
@ -1010,14 +1024,12 @@ class ActionExecutionTestsV2(base.TestCase):
) )
self.assertEqual(201, resp.status) self.assertEqual(201, resp.status)
body = json.loads(body)
output = json.loads(body['output']) output = json.loads(body['output'])
self.assertTrue(output['result']['status'] in range(200, 307)) self.assertTrue(output['result']['status'] in range(200, 307))
@test.attr(type='sanity') @test.attr(type='sanity')
def test_run_action_std_http_error(self): def test_run_action_std_http_error(self):
resp, body = self.client.post_json( resp, body = self.client.create_action_execution(
'action_executions',
{ {
'name': 'std.http', 'name': 'std.http',
'input': '{"url": "http://www.google.ru/not-found-test"}' 'input': '{"url": "http://www.google.ru/not-found-test"}'
@ -1025,14 +1037,12 @@ class ActionExecutionTestsV2(base.TestCase):
) )
self.assertEqual(201, resp.status) self.assertEqual(201, resp.status)
body = json.loads(body)
output = json.loads(body['output']) output = json.loads(body['output'])
self.assertEqual(404, output['result']['status']) self.assertEqual(404, output['result']['status'])
@test.attr(type='sanity') @test.attr(type='sanity')
def test_create_action_execution(self): def test_create_action_execution(self):
resp, body = self.client.post_json( resp, body = self.client.create_action_execution(
'action_executions',
{ {
'name': 'std.echo', 'name': 'std.echo',
'input': '{"output": "Hello, Mistral!"}', 'input': '{"output": "Hello, Mistral!"}',
@ -1041,9 +1051,6 @@ class ActionExecutionTestsV2(base.TestCase):
) )
self.assertEqual(201, resp.status) self.assertEqual(201, resp.status)
body = json.loads(body)
self.assertEqual('RUNNING', body['state']) self.assertEqual('RUNNING', body['state'])
# We must reread action execution in order to get actual # We must reread action execution in order to get actual
@ -1059,3 +1066,12 @@ class ActionExecutionTestsV2(base.TestCase):
{'result': 'Hello, Mistral!'}, {'result': 'Hello, Mistral!'},
output output
) )
@test.attr(type='negative')
def test_delete_nonexistent_action_execution(self):
self.assertRaises(
exceptions.NotFound,
self.client.delete_obj,
'action_executions',
'nonexist'
)

View File

@ -70,6 +70,7 @@ class MistralClientBase(rest_client.RestClient):
self.workflows = [] self.workflows = []
self.triggers = [] self.triggers = []
self.actions = [] self.actions = []
self.action_executions = []
def get_list_obj(self, name): def get_list_obj(self, name):
resp, body = self.get(name) resp, body = self.get(name)
@ -204,6 +205,15 @@ class MistralClientV2(MistralClientBase):
return [t for t in all_tasks if t['workflow_name'] == wf_name] return [t for t in all_tasks if t['workflow_name'] == wf_name]
def create_action_execution(self, request_body):
resp, body = self.post_json('action_executions', request_body)
params = json.loads(request_body.get('params', '{}'))
if params.get('save_result', False):
self.action_executions.append(json.loads(body)['id'])
return resp, json.loads(body)
class AuthProv(auth.KeystoneV2AuthProvider): class AuthProv(auth.KeystoneV2AuthProvider):
def __init__(self): def __init__(self):

View File

@ -17,7 +17,9 @@
import copy import copy
import datetime import datetime
import json import json
import mock import mock
from oslo_config import cfg
from mistral.db.v2 import api as db_api from mistral.db.v2 import api as db_api
from mistral.db.v2.sqlalchemy import models from mistral.db.v2.sqlalchemy import models
@ -45,6 +47,34 @@ ACTION_EX_DB = models.ActionExecution(
updated_at=datetime.datetime(1970, 1, 1) updated_at=datetime.datetime(1970, 1, 1)
) )
AD_HOC_ACTION_EX_DB = models.ActionExecution(
id='123',
state=states.SUCCESS,
state_info=states.SUCCESS,
tags=['foo', 'fee'],
name='std.echo',
description='something',
accepted=True,
input={},
output={},
created_at=datetime.datetime(1970, 1, 1),
updated_at=datetime.datetime(1970, 1, 1)
)
ACTION_EX_DB_NOT_COMPLETE = models.ActionExecution(
id='123',
state=states.RUNNING,
state_info=states.RUNNING,
tags=['foo', 'fee'],
name='std.echo',
description='something',
accepted=False,
input={},
output={},
created_at=datetime.datetime(1970, 1, 1),
updated_at=datetime.datetime(1970, 1, 1)
)
ACTION_EX = { ACTION_EX = {
'id': '123', 'id': '123',
'workflow_name': 'flow', 'workflow_name': 'flow',
@ -80,24 +110,39 @@ BROKEN_ACTION = copy.copy(ACTION_EX)
BROKEN_ACTION['output'] = 'string not escaped' BROKEN_ACTION['output'] = 'string not escaped'
MOCK_ACTION = mock.MagicMock(return_value=ACTION_EX_DB) MOCK_ACTION = mock.MagicMock(return_value=ACTION_EX_DB)
MOCK_ACTION_NOT_COMPLETE = mock.MagicMock(
return_value=ACTION_EX_DB_NOT_COMPLETE
)
MOCK_AD_HOC_ACTION = mock.MagicMock(return_value=AD_HOC_ACTION_EX_DB)
MOCK_ACTIONS = mock.MagicMock(return_value=[ACTION_EX_DB]) MOCK_ACTIONS = mock.MagicMock(return_value=[ACTION_EX_DB])
MOCK_EMPTY = mock.MagicMock(return_value=[]) MOCK_EMPTY = mock.MagicMock(return_value=[])
MOCK_NOT_FOUND = mock.MagicMock(side_effect=exc.NotFoundException()) MOCK_NOT_FOUND = mock.MagicMock(side_effect=exc.NotFoundException())
MOCK_DELETE = mock.MagicMock(return_value=None)
class TestActionExecutionsController(base.FunctionalTest): class TestActionExecutionsController(base.FunctionalTest):
def setUp(self):
super(TestActionExecutionsController, self).setUp()
self.addCleanup(
cfg.CONF.set_default,
'allow_action_execution_deletion',
False,
group='api'
)
@mock.patch.object(db_api, 'get_action_execution', MOCK_ACTION) @mock.patch.object(db_api, 'get_action_execution', MOCK_ACTION)
def test_get(self): def test_get(self):
resp = self.app.get('/v2/action_executions/123') resp = self.app.get('/v2/action_executions/123')
self.assertEqual(resp.status_int, 200) self.assertEqual(200, resp.status_int)
self.assertDictEqual(ACTION_EX, resp.json) self.assertDictEqual(ACTION_EX, resp.json)
@mock.patch.object(db_api, 'get_action_execution', MOCK_NOT_FOUND) @mock.patch.object(db_api, 'get_action_execution', MOCK_NOT_FOUND)
def test_get_not_found(self): def test_get_not_found(self):
resp = self.app.get('/v2/action_executions/123', expect_errors=True) resp = self.app.get('/v2/action_executions/123', expect_errors=True)
self.assertEqual(resp.status_int, 404) self.assertEqual(404, resp.status_int)
@mock.patch.object(rpc.EngineClient, 'start_action') @mock.patch.object(rpc.EngineClient, 'start_action')
def test_post(self, f): def test_post(self, f):
@ -112,7 +157,7 @@ class TestActionExecutionsController(base.FunctionalTest):
} }
) )
self.assertEqual(resp.status_int, 201) self.assertEqual(201, resp.status_int)
action_exec = ACTION_EX action_exec = ACTION_EX
del action_exec['task_name'] del action_exec['task_name']
@ -136,7 +181,7 @@ class TestActionExecutionsController(base.FunctionalTest):
{'name': 'nova.servers_list'} {'name': 'nova.servers_list'}
) )
self.assertEqual(resp.status_int, 201) self.assertEqual(201, resp.status_int)
self.assertEqual('{"result": "123"}', resp.json['output']) self.assertEqual('{"result": "123"}', resp.json['output'])
f.assert_called_once_with('nova.servers_list', {}, description=None) f.assert_called_once_with('nova.servers_list', {}, description=None)
@ -148,7 +193,7 @@ class TestActionExecutionsController(base.FunctionalTest):
expect_errors=True expect_errors=True
) )
self.assertEqual(resp.status_int, 400) self.assertEqual(400, resp.status_int)
def test_post_bad_input(self): def test_post_bad_input(self):
resp = self.app.post_json( resp = self.app.post_json(
@ -157,7 +202,7 @@ class TestActionExecutionsController(base.FunctionalTest):
expect_errors=True expect_errors=True
) )
self.assertEqual(resp.status_int, 400) self.assertEqual(400, resp.status_int)
@mock.patch.object(rpc.EngineClient, 'on_action_complete') @mock.patch.object(rpc.EngineClient, 'on_action_complete')
def test_put(self, f): def test_put(self, f):
@ -165,7 +210,7 @@ class TestActionExecutionsController(base.FunctionalTest):
resp = self.app.put_json('/v2/action_executions/123', UPDATED_ACTION) resp = self.app.put_json('/v2/action_executions/123', UPDATED_ACTION)
self.assertEqual(resp.status_int, 200) self.assertEqual(200, resp.status_int)
self.assertDictEqual(UPDATED_ACTION, resp.json) self.assertDictEqual(UPDATED_ACTION, resp.json)
f.assert_called_once_with( f.assert_called_once_with(
@ -179,7 +224,7 @@ class TestActionExecutionsController(base.FunctionalTest):
resp = self.app.put_json('/v2/action_executions/123', ERROR_ACTION) resp = self.app.put_json('/v2/action_executions/123', ERROR_ACTION)
self.assertEqual(resp.status_int, 200) self.assertEqual(200, resp.status_int)
self.assertDictEqual(ERROR_ACTION, resp.json) self.assertDictEqual(ERROR_ACTION, resp.json)
f.assert_called_once_with( f.assert_called_once_with(
@ -199,7 +244,7 @@ class TestActionExecutionsController(base.FunctionalTest):
expect_errors=True expect_errors=True
) )
self.assertEqual(resp.status_int, 404) self.assertEqual(404, resp.status_int)
def test_put_bad_result(self): def test_put_bad_result(self):
resp = self.app.put_json( resp = self.app.put_json(
@ -208,7 +253,7 @@ class TestActionExecutionsController(base.FunctionalTest):
expect_errors=True expect_errors=True
) )
self.assertEqual(resp.status_int, 400) self.assertEqual(400, resp.status_int)
@mock.patch.object(rpc.EngineClient, 'on_action_complete') @mock.patch.object(rpc.EngineClient, 'on_action_complete')
def test_put_without_result(self, f): def test_put_without_result(self, f):
@ -219,21 +264,69 @@ class TestActionExecutionsController(base.FunctionalTest):
resp = self.app.put_json('/v2/action_executions/123', action_ex) resp = self.app.put_json('/v2/action_executions/123', action_ex)
self.assertEqual(resp.status_int, 200) self.assertEqual(200, resp.status_int)
@mock.patch.object(db_api, 'get_action_executions', MOCK_ACTIONS) @mock.patch.object(db_api, 'get_action_executions', MOCK_ACTIONS)
def test_get_all(self): def test_get_all(self):
resp = self.app.get('/v2/action_executions') resp = self.app.get('/v2/action_executions')
self.assertEqual(resp.status_int, 200) self.assertEqual(200, resp.status_int)
self.assertEqual(len(resp.json['action_executions']), 1) self.assertEqual(1, len(resp.json['action_executions']))
self.assertDictEqual(ACTION_EX, resp.json['action_executions'][0]) self.assertDictEqual(ACTION_EX, resp.json['action_executions'][0])
@mock.patch.object(db_api, 'get_action_executions', MOCK_EMPTY) @mock.patch.object(db_api, 'get_action_executions', MOCK_EMPTY)
def test_get_all_empty(self): def test_get_all_empty(self):
resp = self.app.get('/v2/action_executions') resp = self.app.get('/v2/action_executions')
self.assertEqual(resp.status_int, 200) self.assertEqual(200, resp.status_int)
self.assertEqual(len(resp.json['action_executions']), 0) self.assertEqual(0, len(resp.json['action_executions']))
@mock.patch.object(db_api, 'get_action_execution', MOCK_AD_HOC_ACTION)
@mock.patch.object(db_api, 'delete_action_execution', MOCK_DELETE)
def test_delete(self):
cfg.CONF.set_default('allow_action_execution_deletion', True, 'api')
resp = self.app.delete('/v2/action_executions/123')
self.assertEqual(204, resp.status_int)
@mock.patch.object(db_api, 'get_action_execution', MOCK_NOT_FOUND)
def test_delete_not_found(self):
cfg.CONF.set_default('allow_action_execution_deletion', True, 'api')
resp = self.app.delete('/v2/action_executions/123', expect_errors=True)
self.assertEqual(404, resp.status_int)
def test_delete_not_allowed(self):
resp = self.app.delete('/v2/action_executions/123', expect_errors=True)
self.assertEqual(403, resp.status_int)
self.assertIn("Action execution deletion is not allowed", resp.body)
@mock.patch.object(db_api, 'get_action_execution', MOCK_ACTION)
def test_delete_action_exeuction_with_task(self):
cfg.CONF.set_default('allow_action_execution_deletion', True, 'api')
resp = self.app.delete('/v2/action_executions/123', expect_errors=True)
self.assertEqual(403, resp.status_int)
self.assertIn("Only ad-hoc action execution can be deleted", resp.body)
@mock.patch.object(
db_api,
'get_action_execution',
MOCK_ACTION_NOT_COMPLETE
)
def test_delete_action_exeuction_not_complete(self):
cfg.CONF.set_default('allow_action_execution_deletion', True, 'api')
resp = self.app.delete('/v2/action_executions/123', expect_errors=True)
self.assertEqual(403, resp.status_int)
self.assertIn(
"Only completed action execution can be deleted",
resp.body
)

View File

@ -633,6 +633,18 @@ class ActionExecutionTest(SQLAlchemyTest):
created.id created.id
) )
def test_delete_other_tenant_action_execution(self):
created = db_api.create_action_execution(ACTION_EXECS[0])
# Create a new user.
auth_context.set_ctx(test_base.get_context(default=False))
self.assertRaises(
exc.NotFoundException,
db_api.delete_action_execution,
created.id
)
def test_trim_status_info(self): def test_trim_status_info(self):
created = db_api.create_action_execution(ACTION_EXECS[0]) created = db_api.create_action_execution(ACTION_EXECS[0])