diff --git a/mistral_tempest_tests/services/base.py b/mistral_tempest_tests/services/base.py index 338cbee..a4e91dc 100644 --- a/mistral_tempest_tests/services/base.py +++ b/mistral_tempest_tests/services/base.py @@ -1,4 +1,5 @@ # Copyright 2013 Mirantis, Inc. All Rights Reserved. +# Copyright 2016 NEC Corporation. All rights reserved. # # 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 @@ -16,16 +17,12 @@ import json import os import time -import mock import six -from oslo_utils import uuidutils -from tempest import clients from tempest import config from tempest.lib import auth from tempest.lib.common import rest_client from tempest.lib import exceptions -from tempest import test as test CONF = config.CONF @@ -116,119 +113,6 @@ class MistralClientBase(rest_client.RestClient): return ex_body -class MistralClientV2(MistralClientBase): - - def post_request(self, url, file_name): - headers = {"headers": "Content-Type:text/plain"} - - return self.post(url, get_resource(file_name), headers=headers) - - def post_json(self, url, obj): - headers = {"Content-Type": "application/json"} - - return self.post(url, json.dumps(obj), headers=headers) - - def update_request(self, url, file_name): - headers = {"headers": "Content-Type:text/plain"} - - resp, body = self.put(url, get_resource(file_name), headers=headers) - - return resp, json.loads(body) - - def get_definition(self, item, name): - resp, body = self.get("%s/%s" % (item, name)) - - return resp, json.loads(body)['definition'] - - def create_workbook(self, yaml_file): - resp, body = self.post_request('workbooks', yaml_file) - - wb_name = json.loads(body)['name'] - self.workbooks.append(wb_name) - - _, wfs = self.get_list_obj('workflows') - - for wf in wfs['workflows']: - if wf['name'].startswith(wb_name): - self.workflows.append(wf['name']) - - return resp, json.loads(body) - - def create_workflow(self, yaml_file, scope=None): - if scope: - resp, body = self.post_request('workflows?scope=public', yaml_file) - else: - resp, body = self.post_request('workflows', yaml_file) - - for wf in json.loads(body)['workflows']: - self.workflows.append(wf['name']) - - return resp, json.loads(body) - - def create_execution(self, identifier, wf_input=None, params=None): - if uuidutils.is_uuid_like(identifier): - body = {"workflow_id": "%s" % identifier} - else: - body = {"workflow_name": "%s" % identifier} - - if wf_input: - body.update({'input': json.dumps(wf_input)}) - if params: - body.update({'params': json.dumps(params)}) - - resp, body = self.post('executions', json.dumps(body)) - - self.executions.append(json.loads(body)['id']) - - return resp, json.loads(body) - - def update_execution(self, execution_id, put_body): - resp, body = self.put('executions/%s' % execution_id, put_body) - - return resp, json.loads(body) - - def create_cron_trigger(self, name, wf_name, wf_input=None, pattern=None, - first_time=None, count=None): - post_body = { - 'name': name, - 'workflow_name': wf_name, - 'pattern': pattern, - 'remaining_executions': count, - 'first_execution_time': first_time - } - - if wf_input: - post_body.update({'workflow_input': json.dumps(wf_input)}) - - rest, body = self.post('cron_triggers', json.dumps(post_body)) - - self.triggers.append(name) - - return rest, json.loads(body) - - def create_action(self, yaml_file): - resp, body = self.post_request('actions', yaml_file) - - self.actions.extend( - [action['name'] for action in json.loads(body)['actions']]) - - return resp, json.loads(body) - - def get_wf_tasks(self, wf_name): - all_tasks = self.get_list_obj('tasks')[1]['tasks'] - - 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): def __init__(self): self.alt_part = None @@ -244,76 +128,3 @@ class AuthProv(auth.KeystoneV2AuthProvider): def base_url(self, *args, **kwargs): return '' - - -class TestCase(test.BaseTestCase): - credentials = ['primary', 'alt'] - - @classmethod - def skip_checks(cls): - super(TestCase, cls).skip_checks() - - if not CONF.service_available.mistral: - raise cls.skipException("Mistral support is required.") - - @classmethod - def resource_setup(cls): - """Client authentication. - - This method allows to initialize authentication before - each test case and define parameters of Mistral API Service. - """ - super(TestCase, cls).resource_setup() - - if 'WITHOUT_AUTH' in os.environ: - cls.mgr = mock.MagicMock() - cls.mgr.auth_provider = AuthProv() - cls.alt_mgr = cls.mgr - else: - cls.mgr = cls.manager - cls.alt_mgr = cls.alt_manager - - if cls._service == 'workflowv2': - cls.client = MistralClientV2( - cls.mgr.auth_provider, cls._service) - cls.alt_client = MistralClientV2( - cls.alt_mgr.auth_provider, cls._service) - - def setUp(self): - super(TestCase, self).setUp() - - def tearDown(self): - super(TestCase, self).tearDown() - - for wb in self.client.workbooks: - self.client.delete_obj('workbooks', wb) - - self.client.workbooks = [] - - -class TestCaseAdvanced(TestCase): - @classmethod - def resource_setup(cls): - super(TestCaseAdvanced, cls).resource_setup() - - cls.server_client = clients.ServersClient( - cls.mgr.auth_provider, - "compute", - region=CONF.identity.region - ) - - cls.image_ref = CONF.compute.image_ref - cls.flavor_ref = CONF.compute.flavor_ref - - def tearDown(self): - for wb in self.client.workbooks: - self.client.delete_obj('workbooks', wb) - - self.client.workbooks = [] - - for ex in self.client.executions: - self.client.delete_obj('executions', ex) - - self.client.executions = [] - - super(TestCaseAdvanced, self).tearDown() diff --git a/mistral_tempest_tests/services/v2/__init__.py b/mistral_tempest_tests/services/v2/__init__.py new file mode 100644 index 0000000..e69de29 diff --git a/mistral_tempest_tests/services/v2/mistral_client.py b/mistral_tempest_tests/services/v2/mistral_client.py new file mode 100644 index 0000000..f9c8c4b --- /dev/null +++ b/mistral_tempest_tests/services/v2/mistral_client.py @@ -0,0 +1,139 @@ +# Copyright 2016 NEC Corporation. All rights reserved. +# +# 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 json + +from oslo_utils import uuidutils +from tempest import config + +from mistral_tempest_tests.services import base + +CONF = config.CONF + + +class MistralClientV2(base.MistralClientBase): + + def post_request(self, url, file_name): + headers = {"headers": "Content-Type:text/plain"} + + return self.post(url, base.get_resource(file_name), headers=headers) + + def post_json(self, url, obj): + headers = {"Content-Type": "application/json"} + + return self.post(url, json.dumps(obj), headers=headers) + + def update_request(self, url, file_name): + headers = {"headers": "Content-Type:text/plain"} + + resp, body = self.put( + url, + base.get_resource(file_name), + headers=headers + ) + + return resp, json.loads(body) + + def get_definition(self, item, name): + resp, body = self.get("%s/%s" % (item, name)) + + return resp, json.loads(body)['definition'] + + def create_workbook(self, yaml_file): + resp, body = self.post_request('workbooks', yaml_file) + + wb_name = json.loads(body)['name'] + self.workbooks.append(wb_name) + + _, wfs = self.get_list_obj('workflows') + + for wf in wfs['workflows']: + if wf['name'].startswith(wb_name): + self.workflows.append(wf['name']) + + return resp, json.loads(body) + + def create_workflow(self, yaml_file, scope=None): + if scope: + resp, body = self.post_request('workflows?scope=public', yaml_file) + else: + resp, body = self.post_request('workflows', yaml_file) + + for wf in json.loads(body)['workflows']: + self.workflows.append(wf['name']) + + return resp, json.loads(body) + + def create_execution(self, identifier, wf_input=None, params=None): + if uuidutils.is_uuid_like(identifier): + body = {"workflow_id": "%s" % identifier} + else: + body = {"workflow_name": "%s" % identifier} + + if wf_input: + body.update({'input': json.dumps(wf_input)}) + if params: + body.update({'params': json.dumps(params)}) + + resp, body = self.post('executions', json.dumps(body)) + + self.executions.append(json.loads(body)['id']) + + return resp, json.loads(body) + + def update_execution(self, execution_id, put_body): + resp, body = self.put('executions/%s' % execution_id, put_body) + + return resp, json.loads(body) + + def create_cron_trigger(self, name, wf_name, wf_input=None, pattern=None, + first_time=None, count=None): + post_body = { + 'name': name, + 'workflow_name': wf_name, + 'pattern': pattern, + 'remaining_executions': count, + 'first_execution_time': first_time + } + + if wf_input: + post_body.update({'workflow_input': json.dumps(wf_input)}) + + rest, body = self.post('cron_triggers', json.dumps(post_body)) + + self.triggers.append(name) + + return rest, json.loads(body) + + def create_action(self, yaml_file): + resp, body = self.post_request('actions', yaml_file) + + self.actions.extend( + [action['name'] for action in json.loads(body)['actions']]) + + return resp, json.loads(body) + + def get_wf_tasks(self, wf_name): + all_tasks = self.get_list_obj('tasks')[1]['tasks'] + + 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) diff --git a/mistral_tempest_tests/tests/api/v2/test_action_executions.py b/mistral_tempest_tests/tests/api/v2/test_action_executions.py new file mode 100644 index 0000000..6297e5a --- /dev/null +++ b/mistral_tempest_tests/tests/api/v2/test_action_executions.py @@ -0,0 +1,136 @@ +# Copyright 2016 NEC Corporation. All rights reserved. +# +# 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 json +import six + +from oslo_log import log as logging +from tempest.lib import exceptions +from tempest import test + +from mistral_tempest_tests.tests import base + + +LOG = logging.getLogger(__name__) + + +class ActionExecutionTestsV2(base.TestCase): + _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') + def test_run_action_execution(self): + resp, body = self.client.create_action_execution( + { + 'name': 'std.echo', + 'input': '{"output": "Hello, Mistral!"}' + } + ) + + self.assertEqual(201, resp.status) + output = json.loads(body['output']) + self.assertDictEqual( + {'result': 'Hello, Mistral!'}, + output + ) + + @test.attr(type='sanity') + def test_run_action_std_http(self): + resp, body = self.client.create_action_execution( + { + 'name': 'std.http', + 'input': '{"url": "http://wiki.openstack.org"}' + } + ) + + self.assertEqual(201, resp.status) + output = json.loads(body['output']) + self.assertTrue(output['result']['status'] in range(200, 307)) + + @test.attr(type='sanity') + def test_run_action_std_http_error(self): + resp, body = self.client.create_action_execution( + { + 'name': 'std.http', + 'input': '{"url": "http://www.google.ru/not-found-test"}' + } + ) + + self.assertEqual(201, resp.status) + output = json.loads(body['output']) + self.assertEqual(404, output['result']['status']) + + @test.attr(type='sanity') + def test_create_action_execution(self): + resp, body = self.client.create_action_execution( + { + 'name': 'std.echo', + 'input': '{"output": "Hello, Mistral!"}', + 'params': '{"save_result": true}' + } + ) + + self.assertEqual(201, resp.status) + self.assertEqual('RUNNING', body['state']) + + # We must reread action execution in order to get actual + # state and output. + body = self.client.wait_execution_success( + body, + url='action_executions' + ) + output = json.loads(body['output']) + + self.assertEqual('SUCCESS', body['state']) + self.assertDictEqual( + {'result': 'Hello, Mistral!'}, + output + ) + + @test.attr(type='negative') + def test_delete_nonexistent_action_execution(self): + self.assertRaises( + exceptions.NotFound, + self.client.delete_obj, + 'action_executions', + 'nonexist' + ) + + @test.attr(type='sanity') + def test_create_action_execution_sync(self): + token = self.client.auth_provider.get_token() + resp, body = self.client.create_action_execution( + { + 'name': 'std.http', + 'input': '{{"url": "http://localhost:8989/v2/workflows",\ + "headers": {{"X-Auth-Token": "{}"}}}}'.format(token) + } + ) + + self.assertEqual(201, resp.status) + output = json.loads(body['output']) + self.assertEqual(200, output['result']['status']) diff --git a/mistral_tempest_tests/tests/api/v2/test_actions.py b/mistral_tempest_tests/tests/api/v2/test_actions.py new file mode 100644 index 0000000..b095500 --- /dev/null +++ b/mistral_tempest_tests/tests/api/v2/test_actions.py @@ -0,0 +1,239 @@ +# Copyright 2016 NEC Corporation. All rights reserved. +# +# 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 oslo_log import log as logging +from tempest.lib import exceptions +from tempest import test + +from mistral import utils +from mistral_tempest_tests.tests import base + + +LOG = logging.getLogger(__name__) + + +class ActionTestsV2(base.TestCase): + + _service = 'workflowv2' + + def get_field_value(self, body, act_name, field): + return [body['actions'][i][field] + for i in range(len(body['actions'])) + if body['actions'][i]['name'] == act_name][0] + + def tearDown(self): + for act in self.client.actions: + self.client.delete_obj('actions', act) + self.client.actions = [] + + super(ActionTestsV2, self).tearDown() + + @test.attr(type='smoke') + def test_get_list_actions(self): + resp, body = self.client.get_list_obj('actions') + + self.assertEqual(200, resp.status) + self.assertNotEqual([], body['actions']) + self.assertNotIn('next', body) + + @test.attr(type='smoke') + def test_get_list_actions_with_pagination(self): + resp, body = self.client.get_list_obj( + 'actions?limit=1&sort_keys=name&sort_dirs=desc' + ) + + self.assertEqual(200, resp.status) + self.assertEqual(1, len(body['actions'])) + self.assertIn('next', body) + + name_1 = body['actions'][0].get('name') + next = body.get('next') + + param_dict = utils.get_dict_from_string( + next.split('?')[1], + delimiter='&' + ) + + expected_sub_dict = { + 'limit': 1, + 'sort_keys': 'name', + 'sort_dirs': 'desc' + } + + self.assertDictContainsSubset(expected_sub_dict, param_dict) + + # Query again using 'next' hint + url_param = next.split('/')[-1] + resp, body = self.client.get_list_obj(url_param) + + self.assertEqual(200, resp.status) + self.assertEqual(1, len(body['actions'])) + + name_2 = body['actions'][0].get('name') + + self.assertGreater(name_1, name_2) + + @test.attr(type='negative') + def test_get_list_actions_nonexist_sort_dirs(self): + context = self.assertRaises( + exceptions.BadRequest, + self.client.get_list_obj, + 'actions?limit=1&sort_keys=id&sort_dirs=nonexist' + ) + + self.assertIn( + 'Unknown sort direction', + context.resp_body.get('faultstring') + ) + + @test.attr(type='negative') + def test_get_list_actions_invalid_limit(self): + context = self.assertRaises( + exceptions.BadRequest, + self.client.get_list_obj, + 'actions?limit=-1&sort_keys=id&sort_dirs=asc' + ) + + self.assertIn( + 'Limit must be positive', + context.resp_body.get('faultstring') + ) + + @test.attr(type='negative') + def test_get_list_actions_duplicate_sort_keys(self): + context = self.assertRaises( + exceptions.BadRequest, + self.client.get_list_obj, + 'actions?limit=1&sort_keys=id,id&sort_dirs=asc,asc' + ) + + self.assertIn( + 'Length of sort_keys must be equal or greater than sort_dirs', + context.resp_body.get('faultstring') + ) + + @test.attr(type='sanity') + def test_create_and_delete_few_actions(self): + resp, body = self.client.create_action('action_v2.yaml') + self.assertEqual(201, resp.status) + + created_acts = [action['name'] for action in body['actions']] + + resp, body = self.client.get_list_obj('actions') + self.assertEqual(200, resp.status) + + actions = [action['name'] for action in body['actions']] + + for act in created_acts: + self.assertIn(act, actions) + self.client.delete_obj('actions', act) + + _, body = self.client.get_list_obj('actions') + actions = [action['name'] for action in body['actions']] + + for act in created_acts: + self.assertNotIn(act, actions) + self.client.actions.remove(act) + + @test.attr(type='sanity') + def test_get_action(self): + _, body = self.client.create_action('action_v2.yaml') + action_name = body['actions'][0]['name'] + resp, body = self.client.get_object('actions', action_name) + + self.assertEqual(200, resp.status) + self.assertEqual(action_name, body['name']) + + @test.attr(type='sanity') + def test_update_action(self): + _, body = self.client.create_action('action_v2.yaml') + action = body['actions'][0]['name'] + + act_created_at = self.get_field_value( + body=body, act_name=action, field='created_at') + + self.assertNotIn('updated at', body['actions']) + + resp, body = self.client.update_request('actions', 'action_v2.yaml') + self.assertEqual(200, resp.status) + + actions = [act['name'] for act in body['actions']] + self.assertIn(action, actions) + + updated_act_created_at = self.get_field_value( + body=body, act_name=action, field='created_at') + + self.assertEqual(act_created_at.split(".")[0], updated_act_created_at) + self.assertTrue(all(['updated_at' in item + for item in body['actions']])) + + @test.attr(type='sanity') + def test_get_action_definition(self): + _, body = self.client.create_action('action_v2.yaml') + act_name = body['actions'][0]['name'] + + resp, body = self.client.get_definition('actions', act_name) + self.assertEqual(200, resp.status) + self.assertIsNotNone(body) + self.assertIn(act_name, body) + + @test.attr(type='negative') + def test_get_nonexistent_action(self): + self.assertRaises( + exceptions.NotFound, + self.client.get_object, + 'actions', 'nonexist' + ) + + @test.attr(type='negative') + def test_double_creation(self): + self.client.create_action('action_v2.yaml') + + self.assertRaises( + exceptions.Conflict, + self.client.create_action, + 'action_v2.yaml' + ) + + @test.attr(type='negative') + def test_create_action_invalid_def(self): + self.assertRaises( + exceptions.BadRequest, + self.client.create_action, + 'wb_v2.yaml' + ) + + @test.attr(type='negative') + def test_update_action_invalid_def(self): + self.assertRaises( + exceptions.BadRequest, + self.client.update_request, + 'actions', 'wb_v2.yaml' + ) + + @test.attr(type='negative') + def test_delete_nonexistent_action(self): + self.assertRaises( + exceptions.NotFound, + self.client.delete_obj, + 'actions', 'nonexist' + ) + + @test.attr(type='negative') + def test_delete_standard_action(self): + self.assertRaises( + exceptions.BadRequest, + self.client.delete_obj, + 'actions', 'nova.servers_create' + ) diff --git a/mistral_tempest_tests/tests/api/v2/test_cron_triggers.py b/mistral_tempest_tests/tests/api/v2/test_cron_triggers.py new file mode 100644 index 0000000..4dcfb29 --- /dev/null +++ b/mistral_tempest_tests/tests/api/v2/test_cron_triggers.py @@ -0,0 +1,224 @@ +# Copyright 2016 NEC Corporation. All rights reserved. +# +# 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 oslo_concurrency.fixture import lockutils +from oslo_log import log as logging +from tempest.lib import exceptions +from tempest import test + +from mistral_tempest_tests.tests import base + + +LOG = logging.getLogger(__name__) + + +class CronTriggerTestsV2(base.TestCase): + + _service = 'workflowv2' + + def setUp(self): + super(CronTriggerTestsV2, self).setUp() + self.useFixture(lockutils.LockFixture('mistral-workflow')) + _, body = self.client.create_workflow('wf_v2.yaml') + self.wf_name = body['workflows'][0]['name'] + + def tearDown(self): + + for tr in self.client.triggers: + self.client.delete_obj('cron_triggers', tr) + self.client.triggers = [] + + for wf in self.client.workflows: + self.client.delete_obj('workflows', wf) + self.client.workflows = [] + + super(CronTriggerTestsV2, self).tearDown() + + @test.attr(type='smoke') + def test_get_list_cron_triggers(self): + resp, body = self.client.get_list_obj('cron_triggers') + + self.assertEqual(200, resp.status) + self.assertEqual([], body['cron_triggers']) + + @test.attr(type='sanity') + def test_create_and_delete_cron_triggers(self): + tr_name = 'trigger' + + resp, body = self.client.create_cron_trigger( + tr_name, self.wf_name, None, '5 * * * *') + self.assertEqual(201, resp.status) + self.assertEqual(tr_name, body['name']) + + resp, body = self.client.get_list_obj('cron_triggers') + self.assertEqual(200, resp.status) + + trs_names = [tr['name'] for tr in body['cron_triggers']] + self.assertIn(tr_name, trs_names) + + self.client.delete_obj('cron_triggers', tr_name) + self.client.triggers.remove(tr_name) + + _, body = self.client.get_list_obj('cron_triggers') + + trs_names = [tr['name'] for tr in body['cron_triggers']] + self.assertNotIn(tr_name, trs_names) + + @test.attr(type='sanity') + def test_create_and_delete_oneshot_cron_triggers(self): + tr_name = 'trigger' + + resp, body = self.client.create_cron_trigger( + tr_name, self.wf_name, None, None, "4242-12-25 13:37") + self.assertEqual(201, resp.status) + self.assertEqual(tr_name, body['name']) + self.assertEqual("4242-12-25 13:37:00", body['next_execution_time']) + + resp, body = self.client.get_list_obj('cron_triggers') + self.assertEqual(200, resp.status) + + trs_names = [tr['name'] for tr in body['cron_triggers']] + self.assertIn(tr_name, trs_names) + + self.client.delete_obj('cron_triggers', tr_name) + self.client.triggers.remove(tr_name) + + _, body = self.client.get_list_obj('cron_triggers') + + trs_names = [tr['name'] for tr in body['cron_triggers']] + self.assertNotIn(tr_name, trs_names) + + @test.attr(type='sanity') + def test_create_two_cron_triggers_for_one_wf(self): + tr_name_1 = 'trigger1' + tr_name_2 = 'trigger2' + + resp, body = self.client.create_cron_trigger( + tr_name_1, self.wf_name, None, '5 * * * *') + self.assertEqual(201, resp.status) + self.assertEqual(tr_name_1, body['name']) + + resp, body = self.client.create_cron_trigger( + tr_name_2, self.wf_name, None, '15 * * * *') + self.assertEqual(201, resp.status) + self.assertEqual(tr_name_2, body['name']) + + resp, body = self.client.get_list_obj('cron_triggers') + self.assertEqual(200, resp.status) + + trs_names = [tr['name'] for tr in body['cron_triggers']] + self.assertIn(tr_name_1, trs_names) + self.assertIn(tr_name_2, trs_names) + + @test.attr(type='sanity') + def test_get_cron_trigger(self): + tr_name = 'trigger' + self.client.create_cron_trigger( + tr_name, self.wf_name, None, '5 * * * *') + + resp, body = self.client.get_object('cron_triggers', tr_name) + + self.assertEqual(200, resp.status) + self.assertEqual(tr_name, body['name']) + + @test.attr(type='negative') + def test_create_cron_trigger_nonexistent_wf(self): + self.assertRaises(exceptions.NotFound, + self.client.create_cron_trigger, + 'trigger', 'nonexist', None, '5 * * * *') + + @test.attr(type='negative') + def test_create_cron_trigger_invalid_count(self): + self.assertRaises(exceptions.BadRequest, + self.client.create_cron_trigger, + 'trigger', 'nonexist', None, '5 * * * *', None, "q") + + @test.attr(type='negative') + def test_create_cron_trigger_negative_count(self): + self.assertRaises(exceptions.BadRequest, + self.client.create_cron_trigger, + 'trigger', 'nonexist', None, '5 * * * *', None, -1) + + @test.attr(type='negative') + def test_create_cron_trigger_invalid_first_date(self): + self.assertRaises(exceptions.BadRequest, + self.client.create_cron_trigger, + 'trigger', 'nonexist', None, '5 * * * *', "q") + + @test.attr(type='negative') + def test_create_cron_trigger_count_only(self): + self.assertRaises(exceptions.BadRequest, + self.client.create_cron_trigger, + 'trigger', 'nonexist', None, None, None, "42") + + @test.attr(type='negative') + def test_create_cron_trigger_date_and_count_without_pattern(self): + self.assertRaises(exceptions.BadRequest, + self.client.create_cron_trigger, + 'trigger', 'nonexist', None, None, + "4242-12-25 13:37", "42") + + @test.attr(type='negative') + def test_get_nonexistent_cron_trigger(self): + self.assertRaises(exceptions.NotFound, + self.client.get_object, + 'cron_triggers', 'trigger') + + @test.attr(type='negative') + def test_delete_nonexistent_trigger(self): + self.assertRaises(exceptions.NotFound, + self.client.delete_obj, + 'cron_triggers', 'trigger') + + @test.attr(type='negative') + def test_create_two_cron_triggers_with_same_name(self): + tr_name = 'trigger' + self.client.create_cron_trigger( + tr_name, self.wf_name, None, '5 * * * *') + self.assertRaises(exceptions.Conflict, + self.client.create_cron_trigger, + tr_name, self.wf_name, None, '5 * * * *') + + @test.attr(type='negative') + def test_create_two_cron_triggers_with_same_pattern(self): + self.client.create_cron_trigger( + 'trigger1', + self.wf_name, + None, + '5 * * * *', + "4242-12-25 13:37", + "42" + ) + self.assertRaises( + exceptions.Conflict, + self.client.create_cron_trigger, + 'trigger2', + self.wf_name, + None, + '5 * * * *', + "4242-12-25 13:37", + "42" + ) + + @test.attr(type='negative') + def test_invalid_cron_pattern_not_enough_params(self): + self.assertRaises(exceptions.BadRequest, + self.client.create_cron_trigger, + 'trigger', self.wf_name, None, '5 *') + + @test.attr(type='negative') + def test_invalid_cron_pattern_out_of_range(self): + self.assertRaises(exceptions.BadRequest, + self.client.create_cron_trigger, + 'trigger', self.wf_name, None, '88 * * * *') diff --git a/mistral_tempest_tests/tests/api/v2/test_executions.py b/mistral_tempest_tests/tests/api/v2/test_executions.py new file mode 100644 index 0000000..d7fc100 --- /dev/null +++ b/mistral_tempest_tests/tests/api/v2/test_executions.py @@ -0,0 +1,269 @@ +# Copyright 2016 NEC Corporation. All rights reserved. +# +# 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 oslo_concurrency.fixture import lockutils +from oslo_log import log as logging +from tempest.lib import exceptions +from tempest import test + +from mistral import utils +from mistral_tempest_tests.tests import base + + +LOG = logging.getLogger(__name__) + + +class ExecutionTestsV2(base.TestCase): + + _service = 'workflowv2' + + def setUp(self): + super(ExecutionTestsV2, self).setUp() + self.useFixture(lockutils.LockFixture('mistral-workflow')) + _, body = self.client.create_workflow('wf_v2.yaml') + + self.direct_wf_name = 'wf' + self.direct_wf2_name = 'wf2' + self.direct_wf_id = body['workflows'][0]['id'] + reverse_wfs = [wf for wf in body['workflows'] if wf['name'] == 'wf1'] + self.reverse_wf = reverse_wfs[0] + + def tearDown(self): + for wf in self.client.workflows: + self.client.delete_obj('workflows', wf) + self.client.workflows = [] + + for ex in self.client.executions: + self.client.delete_obj('executions', ex) + self.client.executions = [] + + super(ExecutionTestsV2, self).tearDown() + + @test.attr(type='smoke') + def test_get_list_executions(self): + resp, body = self.client.get_list_obj('executions') + self.assertEqual(200, resp.status) + self.assertNotIn('next', body) + + @test.attr(type='smoke') + def test_get_list_executions_with_pagination(self): + resp, body = self.client.create_execution(self.direct_wf_name) + exec_id_1 = body['id'] + + self.assertEqual(201, resp.status) + + resp, body = self.client.create_execution(self.direct_wf2_name) + exec_id_2 = body['id'] + + self.assertEqual(201, resp.status) + + resp, body = self.client.get_list_obj('executions') + + self.assertIn(exec_id_1, [ex['id'] for ex in body['executions']]) + self.assertIn(exec_id_2, [ex['id'] for ex in body['executions']]) + + resp, body = self.client.get_list_obj( + 'executions?limit=1&sort_keys=workflow_name&sort_dirs=asc' + ) + + self.assertEqual(200, resp.status) + self.assertEqual(1, len(body['executions'])) + self.assertIn('next', body) + + workflow_name_1 = body['executions'][0].get('workflow_name') + next = body.get('next') + param_dict = utils.get_dict_from_string( + next.split('?')[1], + delimiter='&' + ) + + expected_dict = { + 'limit': 1, + 'sort_keys': 'workflow_name', + 'sort_dirs': 'asc', + } + + self.assertTrue( + set(expected_dict.items()).issubset(set(param_dict.items())) + ) + + # Query again using 'next' link + url_param = next.split('/')[-1] + resp, body = self.client.get_list_obj(url_param) + + self.assertEqual(200, resp.status) + self.assertEqual(1, len(body['executions'])) + + workflow_name_2 = body['executions'][0].get('workflow_name') + + self.assertGreater(workflow_name_2, workflow_name_1) + + @test.attr(type='sanity') + def test_create_execution_for_direct_wf(self): + resp, body = self.client.create_execution(self.direct_wf_name) + exec_id = body['id'] + self.assertEqual(201, resp.status) + self.assertEqual(self.direct_wf_name, body['workflow_name']) + self.assertEqual('RUNNING', body['state']) + + resp, body = self.client.get_list_obj('executions') + self.assertIn(exec_id, + [ex_id['id'] for ex_id in body['executions']]) + + @test.attr(type='sanity') + def test_create_execution_for_reverse_wf(self): + resp, body = self.client.create_execution( + self.reverse_wf['name'], + {self.reverse_wf['input']: "Bye"}, + {"task_name": "goodbye"}) + + exec_id = body['id'] + self.assertEqual(201, resp.status) + self.assertEqual(self.reverse_wf['name'], body['workflow_name']) + self.assertEqual('RUNNING', body['state']) + + resp, body = self.client.get_list_obj('executions') + self.assertIn(exec_id, + [ex_id['id'] for ex_id in body['executions']]) + + resp, body = self.client.get_object('executions', exec_id) + # TODO(nmakhotkin): Fix this loop. It is infinite now. + while body['state'] != 'SUCCESS': + resp, body = self.client.get_object('executions', exec_id) + self.assertEqual(200, resp.status) + self.assertEqual('SUCCESS', body['state']) + + @test.attr(type='sanity') + def test_create_execution_by_wf_id(self): + resp, body = self.client.create_execution(self.direct_wf_id) + exec_id = body['id'] + self.assertEqual(201, resp.status) + self.assertEqual(self.direct_wf_id, body['workflow_id']) + self.assertEqual('RUNNING', body['state']) + + resp, body = self.client.get_list_obj('executions') + self.assertIn( + exec_id, + [ex_id['id'] for ex_id in body['executions']] + ) + + @test.attr(type='sanity') + def test_get_execution(self): + _, execution = self.client.create_execution(self.direct_wf_name) + + resp, body = self.client.get_object('executions', execution['id']) + + del execution['state'] + del body['state'] + + self.assertEqual(200, resp.status) + self.assertEqual(execution['id'], body['id']) + + @test.attr(type='sanity') + def test_update_execution_pause(self): + _, execution = self.client.create_execution(self.direct_wf_name) + resp, body = self.client.update_execution( + execution['id'], '{"state": "PAUSED"}') + + self.assertEqual(200, resp.status) + self.assertEqual('PAUSED', body['state']) + + @test.attr(type='sanity') + def test_update_execution_description(self): + _, execution = self.client.create_execution(self.direct_wf_name) + resp, body = self.client.update_execution( + execution['id'], '{"description": "description"}') + + self.assertEqual(200, resp.status) + self.assertEqual('description', body['description']) + + @test.attr(type='sanity') + def test_update_execution_fail(self): + _, execution = self.client.create_execution(self.direct_wf_name) + resp, body = self.client.update_execution( + execution['id'], '{"state": "ERROR", "state_info": "Forced"}') + + self.assertEqual(200, resp.status) + self.assertEqual('ERROR', body['state']) + self.assertEqual('Forced', body['state_info']) + + @test.attr(type='negative') + def test_get_nonexistent_execution(self): + self.assertRaises(exceptions.NotFound, self.client.get_object, + 'executions', '1a2b3c') + + @test.attr(type='negative') + def test_update_nonexistent_execution(self): + put_body = '{"state": "STOPPED"}' + + self.assertRaises(exceptions.NotFound, + self.client.update_execution, + '1a2b3c', put_body) + + @test.attr(type='negative') + def test_delete_nonexistent_execution(self): + self.assertRaises(exceptions.NotFound, + self.client.delete_obj, + 'executions', 'nonexist') + + @test.attr(type='negative') + def test_create_ex_for_nonexistent_wf(self): + self.assertRaises(exceptions.NotFound, + self.client.create_execution, + 'nonexist') + + @test.attr(type='negative') + def test_create_execution_for_reverse_wf_invalid_start_task(self): + self.assertRaises( + exceptions.BadRequest, + self.client.create_execution, + self.reverse_wf['name'], + {self.reverse_wf['input']: "Bye"}, + {"task_name": "nonexist"} + ) + + @test.attr(type='negative') + def test_create_execution_forgot_input_params(self): + self.assertRaises( + exceptions.BadRequest, + self.client.create_execution, + self.reverse_wf['name'], + params={"task_name": "nonexist"} + ) + + @test.attr(type='sanity') + def test_action_ex_concurrency(self): + resp, wf = self.client.create_workflow("wf_action_ex_concurrency.yaml") + self.assertEqual(201, resp.status) + + wf_name = wf['workflows'][0]['name'] + resp, execution = self.client.create_execution(wf_name) + + self.assertEqual(201, resp.status) + self.assertEqual('RUNNING', execution['state']) + + self.client.wait_execution_success(execution) + + @test.attr(type='sanity') + def test_task_ex_concurrency(self): + resp, wf = self.client.create_workflow("wf_task_ex_concurrency.yaml") + self.assertEqual(201, resp.status) + + wf_name = wf['workflows'][0]['name'] + resp, execution = self.client.create_execution(wf_name) + + self.assertEqual(201, resp.status) + self.assertEqual('RUNNING', execution['state']) + + self.client.wait_execution(execution, target_state='ERROR') diff --git a/mistral_tempest_tests/tests/api/v2/test_mistral_basic_v2.py b/mistral_tempest_tests/tests/api/v2/test_mistral_basic_v2.py deleted file mode 100644 index 6aa9022..0000000 --- a/mistral_tempest_tests/tests/api/v2/test_mistral_basic_v2.py +++ /dev/null @@ -1,1208 +0,0 @@ -# Copyright 2014 Mirantis, Inc. -# -# 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 json -import six - -from oslo_concurrency.fixture import lockutils -from oslo_log import log as logging -from tempest.lib import exceptions -from tempest import test - -from mistral import utils -from mistral_tempest_tests.services import base - - -LOG = logging.getLogger(__name__) - - -class WorkbookTestsV2(base.TestCase): - - _service = 'workflowv2' - - def tearDown(self): - for wf in self.client.workflows: - self.client.delete_obj('workflows', wf) - self.client.workflows = [] - - super(WorkbookTestsV2, self).tearDown() - - @test.attr(type='smoke') - def test_get_list_workbooks(self): - resp, body = self.client.get_list_obj('workbooks') - - self.assertEqual(200, resp.status) - self.assertEqual([], body['workbooks']) - - @test.attr(type='sanity') - def test_create_and_delete_workbook(self): - self.useFixture(lockutils.LockFixture('mistral-workflow')) - resp, body = self.client.create_workbook('wb_v2.yaml') - name = body['name'] - self.assertEqual(201, resp.status) - - resp, body = self.client.get_list_obj('workbooks') - self.assertEqual(200, resp.status) - self.assertEqual(name, body['workbooks'][0]['name']) - - self.client.delete_obj('workbooks', name) - self.client.workbooks.remove(name) - - _, body = self.client.get_list_obj('workbooks') - self.assertEqual([], body['workbooks']) - - @test.attr(type='sanity') - def test_get_workbook(self): - self.useFixture(lockutils.LockFixture('mistral-workflow')) - _, body = self.client.create_workbook('wb_v2.yaml') - name = body['name'] - - resp, body = self.client.get_object('workbooks', name) - self.assertEqual(200, resp.status) - self.assertEqual(name, body['name']) - - @test.attr(type='sanity') - def test_update_workbook(self): - self.useFixture(lockutils.LockFixture('mistral-workflow')) - _, body = self.client.create_workbook('wb_v2.yaml') - name = body['name'] - resp, body = self.client.update_request('workbooks', 'wb_v2.yaml') - - self.assertEqual(200, resp.status) - self.assertEqual(name, body['name']) - - @test.attr(type='sanity') - def test_get_workbook_definition(self): - self.useFixture(lockutils.LockFixture('mistral-workflow')) - _, body = self.client.create_workbook('wb_v2.yaml') - name = body['name'] - resp, body = self.client.get_definition('workbooks', name) - - self.assertEqual(200, resp.status) - self.assertIsNotNone(body) - - @test.attr(type='negative') - def test_get_nonexistent_workbook_definition(self): - self.assertRaises(exceptions.NotFound, - self.client.get_definition, - 'workbooks', 'nonexist') - - @test.attr(type='negative') - def test_get_nonexistent_workbook(self): - self.assertRaises(exceptions.NotFound, self.client.get_object, - 'workbooks', 'nonexist') - - @test.attr(type='negative') - def test_double_create_workbook(self): - self.useFixture(lockutils.LockFixture('mistral-workflow')) - _, body = self.client.create_workbook('wb_v2.yaml') - name = body['name'] - self.assertRaises(exceptions.Conflict, - self.client.create_workbook, - 'wb_v2.yaml') - - self.client.delete_obj('workbooks', name) - self.client.workbooks.remove(name) - _, body = self.client.get_list_obj('workbooks') - - self.assertEqual([], body['workbooks']) - - @test.attr(type='negative') - def test_create_wb_with_invalid_def(self): - self.assertRaises( - exceptions.BadRequest, - self.client.create_workbook, - 'wb_v1.yaml' - ) - - @test.attr(type='negative') - def test_update_wb_with_invalid_def(self): - self.assertRaises( - exceptions.BadRequest, - self.client.update_request, - 'workbooks', - 'wb_v1.yaml' - ) - - -class WorkflowTestsV2(base.TestCase): - - _service = 'workflowv2' - - def tearDown(self): - for wf in self.client.workflows: - self.client.delete_obj('workflows', wf) - self.client.workflows = [] - - super(WorkflowTestsV2, self).tearDown() - - @test.attr(type='smoke') - def test_get_list_workflows(self): - resp, body = self.client.get_list_obj('workflows') - self.assertEqual(200, resp.status) - - names = [wf['name'] for wf in body['workflows']] - - self.assertIn('std.create_instance', names) - - self.assertNotIn('next', body) - - @test.attr(type='smoke') - def test_get_list_workflows_with_fields(self): - resp, body = self.client.get_list_obj('workflows?fields=name') - - self.assertEqual(200, resp.status) - - for wf in body['workflows']: - self.assertListEqual(sorted(['id', 'name']), sorted(list(wf))) - - @test.attr(type='smoke') - def test_get_list_workflows_with_pagination(self): - resp, body = self.client.get_list_obj( - 'workflows?limit=1&sort_keys=name&sort_dirs=desc' - ) - - self.assertEqual(200, resp.status) - self.assertEqual(1, len(body['workflows'])) - self.assertIn('next', body) - - name_1 = body['workflows'][0].get('name') - next = body.get('next') - - param_dict = utils.get_dict_from_string( - next.split('?')[1], - delimiter='&' - ) - - expected_sub_dict = { - 'limit': 1, - 'sort_keys': 'name', - 'sort_dirs': 'desc' - } - - self.assertDictContainsSubset(expected_sub_dict, param_dict) - - # Query again using 'next' hint - url_param = next.split('/')[-1] - resp, body = self.client.get_list_obj(url_param) - - self.assertEqual(200, resp.status) - self.assertEqual(1, len(body['workflows'])) - - name_2 = body['workflows'][0].get('name') - - self.assertGreater(name_1, name_2) - - @test.attr(type='negative') - def test_get_list_workflows_nonexist_sort_dirs(self): - context = self.assertRaises( - exceptions.BadRequest, - self.client.get_list_obj, - 'workflows?limit=1&sort_keys=id&sort_dirs=nonexist' - ) - - self.assertIn( - 'Unknown sort direction', - context.resp_body.get('faultstring') - ) - - @test.attr(type='negative') - def test_get_list_workflows_invalid_limit(self): - context = self.assertRaises( - exceptions.BadRequest, - self.client.get_list_obj, - 'workflows?limit=-1&sort_keys=id&sort_dirs=asc' - ) - - self.assertIn( - 'Limit must be positive', - context.resp_body.get('faultstring') - ) - - @test.attr(type='negative') - def test_get_list_workflows_duplicate_sort_keys(self): - context = self.assertRaises( - exceptions.BadRequest, - self.client.get_list_obj, - 'workflows?limit=1&sort_keys=id,id&sort_dirs=asc,asc' - ) - - self.assertIn( - 'Length of sort_keys must be equal or greater than sort_dirs', - context.resp_body.get('faultstring') - ) - - @test.attr(type='sanity') - def test_create_and_delete_workflow(self): - self.useFixture(lockutils.LockFixture('mistral-workflow')) - resp, body = self.client.create_workflow('wf_v2.yaml') - name = body['workflows'][0]['name'] - - self.assertEqual(201, resp.status) - - resp, body = self.client.get_list_obj('workflows') - self.assertEqual(200, resp.status) - - names = [wf['name'] for wf in body['workflows']] - self.assertIn(name, names) - - self.client.delete_obj('workflows', name) - self.client.workflows.remove(name) - - _, body = self.client.get_list_obj('workflows') - - names = [wf['name'] for wf in body['workflows']] - self.assertNotIn(name, names) - - @test.attr(type='sanity') - def test_get_workflow(self): - self.useFixture(lockutils.LockFixture('mistral-workflow')) - _, body = self.client.create_workflow('wf_v2.yaml') - name = body['workflows'][0]['name'] - - resp, body = self.client.get_object('workflows', name) - - self.assertEqual(200, resp.status) - self.assertEqual(name, body['name']) - - @test.attr(type='sanity') - def test_update_workflow(self): - self.useFixture(lockutils.LockFixture('mistral-workflow')) - _, body = self.client.create_workflow('wf_v2.yaml') - name = body['workflows'][0]['name'] - - resp, body = self.client.update_request('workflows', 'wf_v2.yaml') - - self.assertEqual(200, resp.status) - self.assertEqual(name, body['workflows'][0]['name']) - - @test.attr(type='sanity') - def test_get_workflow_definition(self): - self.useFixture(lockutils.LockFixture('mistral-workflow')) - _, body = self.client.create_workflow('wf_v2.yaml') - name = body['workflows'][0]['name'] - - resp, body = self.client.get_definition('workflows', name) - - self.assertEqual(200, resp.status) - self.assertIsNotNone(body) - - @test.attr(type='sanity') - def test_get_workflow_uploaded_in_wb(self): - self.useFixture(lockutils.LockFixture('mistral-workflow')) - _, body = self.client.create_workbook('wb_v2.yaml') - wb_name = body['name'] - - _, body = self.client.get_list_obj('workflows') - wf_names = [wf['name'] for wf in body['workflows'] - if wf['name'].startswith(wb_name)] - - self.assertNotEmpty(wf_names) - - @test.attr(type='negative') - def test_get_nonexistent_workflow_definition(self): - self.assertRaises(exceptions.NotFound, - self.client.get_definition, - 'workflows', 'nonexist') - - @test.attr(type='negative') - def test_get_nonexistent_workflow(self): - self.assertRaises(exceptions.NotFound, self.client.get_object, - 'workflows', 'nonexist') - - @test.attr(type='negative') - def test_double_create_workflows(self): - self.useFixture(lockutils.LockFixture('mistral-workflow')) - _, body = self.client.create_workflow('wf_v2.yaml') - self.assertRaises(exceptions.Conflict, - self.client.create_workflow, - 'wf_v2.yaml') - - @test.attr(type='negative') - def test_create_wf_with_invalid_def(self): - self.assertRaises(exceptions.BadRequest, - self.client.create_workflow, - 'wb_v1.yaml') - - @test.attr(type='negative') - def test_update_wf_with_invalid_def(self): - self.assertRaises(exceptions.BadRequest, - self.client.update_request, - 'workflows', 'wb_v1.yaml') - - @test.attr(type='negative') - def test_delete_wf_with_trigger_associate(self): - tr_name = 'trigger' - resp, body = self.client.create_workflow('wf_v2.yaml') - name = body['workflows'][0]['name'] - resp, body = self.client.create_cron_trigger( - tr_name, name, None, '5 * * * *') - - try: - self.assertRaises( - exceptions.BadRequest, - self.client.delete_obj, - 'workflows', - name - ) - finally: - self.client.delete_obj('cron_triggers', tr_name) - self.client.triggers.remove(tr_name) - - @test.attr(type='negative') - def test_delete_wf_with_trigger_associate_in_other_tenant(self): - self.useFixture(lockutils.LockFixture('mistral-workflow')) - tr_name = 'trigger' - _, body = self.client.create_workflow('wf_v2.yaml', scope='public') - name = body['workflows'][0]['name'] - resp, body = self.alt_client.create_cron_trigger( - tr_name, - name, - None, - '5 * * * *' - ) - - try: - exception = self.assertRaises( - exceptions.BadRequest, - self.client.delete_obj, - 'workflows', - name - ) - - self.assertIn( - "Can't delete workflow that has triggers associated", - exception.resp_body['faultstring'] - ) - finally: - self.alt_client.delete_obj('cron_triggers', tr_name) - self.alt_client.triggers.remove(tr_name) - - @test.attr(type='negative') - def test_delete_nonexistent_wf(self): - self.assertRaises(exceptions.NotFound, - self.client.delete_obj, - 'workflows', 'nonexist') - - -class ExecutionTestsV2(base.TestCase): - - _service = 'workflowv2' - - def setUp(self): - super(ExecutionTestsV2, self).setUp() - self.useFixture(lockutils.LockFixture('mistral-workflow')) - _, body = self.client.create_workflow('wf_v2.yaml') - - self.direct_wf_name = 'wf' - self.direct_wf2_name = 'wf2' - self.direct_wf_id = body['workflows'][0]['id'] - reverse_wfs = [wf for wf in body['workflows'] if wf['name'] == 'wf1'] - self.reverse_wf = reverse_wfs[0] - - def tearDown(self): - for wf in self.client.workflows: - self.client.delete_obj('workflows', wf) - self.client.workflows = [] - - for ex in self.client.executions: - self.client.delete_obj('executions', ex) - self.client.executions = [] - - super(ExecutionTestsV2, self).tearDown() - - @test.attr(type='smoke') - def test_get_list_executions(self): - resp, body = self.client.get_list_obj('executions') - self.assertEqual(200, resp.status) - self.assertNotIn('next', body) - - @test.attr(type='smoke') - def test_get_list_executions_with_pagination(self): - resp, body = self.client.create_execution(self.direct_wf_name) - exec_id_1 = body['id'] - - self.assertEqual(201, resp.status) - - resp, body = self.client.create_execution(self.direct_wf2_name) - exec_id_2 = body['id'] - - self.assertEqual(201, resp.status) - - resp, body = self.client.get_list_obj('executions') - - self.assertIn(exec_id_1, [ex['id'] for ex in body['executions']]) - self.assertIn(exec_id_2, [ex['id'] for ex in body['executions']]) - - resp, body = self.client.get_list_obj( - 'executions?limit=1&sort_keys=workflow_name&sort_dirs=asc' - ) - - self.assertEqual(200, resp.status) - self.assertEqual(1, len(body['executions'])) - self.assertIn('next', body) - - workflow_name_1 = body['executions'][0].get('workflow_name') - next = body.get('next') - param_dict = utils.get_dict_from_string( - next.split('?')[1], - delimiter='&' - ) - - expected_dict = { - 'limit': 1, - 'sort_keys': 'workflow_name', - 'sort_dirs': 'asc', - } - - self.assertTrue( - set(expected_dict.items()).issubset(set(param_dict.items())) - ) - - # Query again using 'next' link - url_param = next.split('/')[-1] - resp, body = self.client.get_list_obj(url_param) - - self.assertEqual(200, resp.status) - self.assertEqual(1, len(body['executions'])) - - workflow_name_2 = body['executions'][0].get('workflow_name') - - self.assertGreater(workflow_name_2, workflow_name_1) - - @test.attr(type='sanity') - def test_create_execution_for_direct_wf(self): - resp, body = self.client.create_execution(self.direct_wf_name) - exec_id = body['id'] - self.assertEqual(201, resp.status) - self.assertEqual(self.direct_wf_name, body['workflow_name']) - self.assertEqual('RUNNING', body['state']) - - resp, body = self.client.get_list_obj('executions') - self.assertIn(exec_id, - [ex_id['id'] for ex_id in body['executions']]) - - @test.attr(type='sanity') - def test_create_execution_for_reverse_wf(self): - resp, body = self.client.create_execution( - self.reverse_wf['name'], - {self.reverse_wf['input']: "Bye"}, - {"task_name": "goodbye"}) - - exec_id = body['id'] - self.assertEqual(201, resp.status) - self.assertEqual(self.reverse_wf['name'], body['workflow_name']) - self.assertEqual('RUNNING', body['state']) - - resp, body = self.client.get_list_obj('executions') - self.assertIn(exec_id, - [ex_id['id'] for ex_id in body['executions']]) - - resp, body = self.client.get_object('executions', exec_id) - # TODO(nmakhotkin): Fix this loop. It is infinite now. - while body['state'] != 'SUCCESS': - resp, body = self.client.get_object('executions', exec_id) - self.assertEqual(200, resp.status) - self.assertEqual('SUCCESS', body['state']) - - @test.attr(type='sanity') - def test_create_execution_by_wf_id(self): - resp, body = self.client.create_execution(self.direct_wf_id) - exec_id = body['id'] - self.assertEqual(201, resp.status) - self.assertEqual(self.direct_wf_id, body['workflow_id']) - self.assertEqual('RUNNING', body['state']) - - resp, body = self.client.get_list_obj('executions') - self.assertIn( - exec_id, - [ex_id['id'] for ex_id in body['executions']] - ) - - @test.attr(type='sanity') - def test_get_execution(self): - _, execution = self.client.create_execution(self.direct_wf_name) - - resp, body = self.client.get_object('executions', execution['id']) - - del execution['state'] - del body['state'] - - self.assertEqual(200, resp.status) - self.assertEqual(execution['id'], body['id']) - - @test.attr(type='sanity') - def test_update_execution_pause(self): - _, execution = self.client.create_execution(self.direct_wf_name) - resp, body = self.client.update_execution( - execution['id'], '{"state": "PAUSED"}') - - self.assertEqual(200, resp.status) - self.assertEqual('PAUSED', body['state']) - - @test.attr(type='sanity') - def test_update_execution_description(self): - _, execution = self.client.create_execution(self.direct_wf_name) - resp, body = self.client.update_execution( - execution['id'], '{"description": "description"}') - - self.assertEqual(200, resp.status) - self.assertEqual('description', body['description']) - - @test.attr(type='sanity') - def test_update_execution_fail(self): - _, execution = self.client.create_execution(self.direct_wf_name) - resp, body = self.client.update_execution( - execution['id'], '{"state": "ERROR", "state_info": "Forced"}') - - self.assertEqual(200, resp.status) - self.assertEqual('ERROR', body['state']) - self.assertEqual('Forced', body['state_info']) - - @test.attr(type='negative') - def test_get_nonexistent_execution(self): - self.assertRaises(exceptions.NotFound, self.client.get_object, - 'executions', '1a2b3c') - - @test.attr(type='negative') - def test_update_nonexistent_execution(self): - put_body = '{"state": "STOPPED"}' - - self.assertRaises(exceptions.NotFound, - self.client.update_execution, - '1a2b3c', put_body) - - @test.attr(type='negative') - def test_delete_nonexistent_execution(self): - self.assertRaises(exceptions.NotFound, - self.client.delete_obj, - 'executions', 'nonexist') - - @test.attr(type='negative') - def test_create_ex_for_nonexistent_wf(self): - self.assertRaises(exceptions.NotFound, - self.client.create_execution, - 'nonexist') - - @test.attr(type='negative') - def test_create_execution_for_reverse_wf_invalid_start_task(self): - self.assertRaises( - exceptions.BadRequest, - self.client.create_execution, - self.reverse_wf['name'], - {self.reverse_wf['input']: "Bye"}, - {"task_name": "nonexist"} - ) - - @test.attr(type='negative') - def test_create_execution_forgot_input_params(self): - self.assertRaises( - exceptions.BadRequest, - self.client.create_execution, - self.reverse_wf['name'], - params={"task_name": "nonexist"} - ) - - @test.attr(type='sanity') - def test_action_ex_concurrency(self): - resp, wf = self.client.create_workflow("wf_action_ex_concurrency.yaml") - self.assertEqual(201, resp.status) - - wf_name = wf['workflows'][0]['name'] - resp, execution = self.client.create_execution(wf_name) - - self.assertEqual(201, resp.status) - self.assertEqual('RUNNING', execution['state']) - - self.client.wait_execution_success(execution) - - @test.attr(type='sanity') - def test_task_ex_concurrency(self): - resp, wf = self.client.create_workflow("wf_task_ex_concurrency.yaml") - self.assertEqual(201, resp.status) - - wf_name = wf['workflows'][0]['name'] - resp, execution = self.client.create_execution(wf_name) - - self.assertEqual(201, resp.status) - self.assertEqual('RUNNING', execution['state']) - - self.client.wait_execution(execution, target_state='ERROR') - - -class CronTriggerTestsV2(base.TestCase): - - _service = 'workflowv2' - - def setUp(self): - super(CronTriggerTestsV2, self).setUp() - self.useFixture(lockutils.LockFixture('mistral-workflow')) - _, body = self.client.create_workflow('wf_v2.yaml') - self.wf_name = body['workflows'][0]['name'] - - def tearDown(self): - - for tr in self.client.triggers: - self.client.delete_obj('cron_triggers', tr) - self.client.triggers = [] - - for wf in self.client.workflows: - self.client.delete_obj('workflows', wf) - self.client.workflows = [] - - super(CronTriggerTestsV2, self).tearDown() - - @test.attr(type='smoke') - def test_get_list_cron_triggers(self): - resp, body = self.client.get_list_obj('cron_triggers') - - self.assertEqual(200, resp.status) - self.assertEqual([], body['cron_triggers']) - - @test.attr(type='sanity') - def test_create_and_delete_cron_triggers(self): - tr_name = 'trigger' - - resp, body = self.client.create_cron_trigger( - tr_name, self.wf_name, None, '5 * * * *') - self.assertEqual(201, resp.status) - self.assertEqual(tr_name, body['name']) - - resp, body = self.client.get_list_obj('cron_triggers') - self.assertEqual(200, resp.status) - - trs_names = [tr['name'] for tr in body['cron_triggers']] - self.assertIn(tr_name, trs_names) - - self.client.delete_obj('cron_triggers', tr_name) - self.client.triggers.remove(tr_name) - - _, body = self.client.get_list_obj('cron_triggers') - - trs_names = [tr['name'] for tr in body['cron_triggers']] - self.assertNotIn(tr_name, trs_names) - - @test.attr(type='sanity') - def test_create_and_delete_oneshot_cron_triggers(self): - tr_name = 'trigger' - - resp, body = self.client.create_cron_trigger( - tr_name, self.wf_name, None, None, "4242-12-25 13:37") - self.assertEqual(201, resp.status) - self.assertEqual(tr_name, body['name']) - self.assertEqual("4242-12-25 13:37:00", body['next_execution_time']) - - resp, body = self.client.get_list_obj('cron_triggers') - self.assertEqual(200, resp.status) - - trs_names = [tr['name'] for tr in body['cron_triggers']] - self.assertIn(tr_name, trs_names) - - self.client.delete_obj('cron_triggers', tr_name) - self.client.triggers.remove(tr_name) - - _, body = self.client.get_list_obj('cron_triggers') - - trs_names = [tr['name'] for tr in body['cron_triggers']] - self.assertNotIn(tr_name, trs_names) - - @test.attr(type='sanity') - def test_create_two_cron_triggers_for_one_wf(self): - tr_name_1 = 'trigger1' - tr_name_2 = 'trigger2' - - resp, body = self.client.create_cron_trigger( - tr_name_1, self.wf_name, None, '5 * * * *') - self.assertEqual(201, resp.status) - self.assertEqual(tr_name_1, body['name']) - - resp, body = self.client.create_cron_trigger( - tr_name_2, self.wf_name, None, '15 * * * *') - self.assertEqual(201, resp.status) - self.assertEqual(tr_name_2, body['name']) - - resp, body = self.client.get_list_obj('cron_triggers') - self.assertEqual(200, resp.status) - - trs_names = [tr['name'] for tr in body['cron_triggers']] - self.assertIn(tr_name_1, trs_names) - self.assertIn(tr_name_2, trs_names) - - @test.attr(type='sanity') - def test_get_cron_trigger(self): - tr_name = 'trigger' - self.client.create_cron_trigger( - tr_name, self.wf_name, None, '5 * * * *') - - resp, body = self.client.get_object('cron_triggers', tr_name) - - self.assertEqual(200, resp.status) - self.assertEqual(tr_name, body['name']) - - @test.attr(type='negative') - def test_create_cron_trigger_nonexistent_wf(self): - self.assertRaises(exceptions.NotFound, - self.client.create_cron_trigger, - 'trigger', 'nonexist', None, '5 * * * *') - - @test.attr(type='negative') - def test_create_cron_trigger_invalid_count(self): - self.assertRaises(exceptions.BadRequest, - self.client.create_cron_trigger, - 'trigger', 'nonexist', None, '5 * * * *', None, "q") - - @test.attr(type='negative') - def test_create_cron_trigger_negative_count(self): - self.assertRaises(exceptions.BadRequest, - self.client.create_cron_trigger, - 'trigger', 'nonexist', None, '5 * * * *', None, -1) - - @test.attr(type='negative') - def test_create_cron_trigger_invalid_first_date(self): - self.assertRaises(exceptions.BadRequest, - self.client.create_cron_trigger, - 'trigger', 'nonexist', None, '5 * * * *', "q") - - @test.attr(type='negative') - def test_create_cron_trigger_count_only(self): - self.assertRaises(exceptions.BadRequest, - self.client.create_cron_trigger, - 'trigger', 'nonexist', None, None, None, "42") - - @test.attr(type='negative') - def test_create_cron_trigger_date_and_count_without_pattern(self): - self.assertRaises(exceptions.BadRequest, - self.client.create_cron_trigger, - 'trigger', 'nonexist', None, None, - "4242-12-25 13:37", "42") - - @test.attr(type='negative') - def test_get_nonexistent_cron_trigger(self): - self.assertRaises(exceptions.NotFound, - self.client.get_object, - 'cron_triggers', 'trigger') - - @test.attr(type='negative') - def test_delete_nonexistent_trigger(self): - self.assertRaises(exceptions.NotFound, - self.client.delete_obj, - 'cron_triggers', 'trigger') - - @test.attr(type='negative') - def test_create_two_cron_triggers_with_same_name(self): - tr_name = 'trigger' - self.client.create_cron_trigger( - tr_name, self.wf_name, None, '5 * * * *') - self.assertRaises(exceptions.Conflict, - self.client.create_cron_trigger, - tr_name, self.wf_name, None, '5 * * * *') - - @test.attr(type='negative') - def test_create_two_cron_triggers_with_same_pattern(self): - self.client.create_cron_trigger( - 'trigger1', - self.wf_name, - None, - '5 * * * *', - "4242-12-25 13:37", - "42" - ) - self.assertRaises( - exceptions.Conflict, - self.client.create_cron_trigger, - 'trigger2', - self.wf_name, - None, - '5 * * * *', - "4242-12-25 13:37", - "42" - ) - - @test.attr(type='negative') - def test_invalid_cron_pattern_not_enough_params(self): - self.assertRaises(exceptions.BadRequest, - self.client.create_cron_trigger, - 'trigger', self.wf_name, None, '5 *') - - @test.attr(type='negative') - def test_invalid_cron_pattern_out_of_range(self): - self.assertRaises(exceptions.BadRequest, - self.client.create_cron_trigger, - 'trigger', self.wf_name, None, '88 * * * *') - - -class ActionTestsV2(base.TestCase): - - _service = 'workflowv2' - - def get_field_value(self, body, act_name, field): - return [body['actions'][i][field] - for i in range(len(body['actions'])) - if body['actions'][i]['name'] == act_name][0] - - def tearDown(self): - for act in self.client.actions: - self.client.delete_obj('actions', act) - self.client.actions = [] - - super(ActionTestsV2, self).tearDown() - - @test.attr(type='smoke') - def test_get_list_actions(self): - resp, body = self.client.get_list_obj('actions') - - self.assertEqual(200, resp.status) - self.assertNotEqual([], body['actions']) - self.assertNotIn('next', body) - - @test.attr(type='smoke') - def test_get_list_actions_with_pagination(self): - resp, body = self.client.get_list_obj( - 'actions?limit=1&sort_keys=name&sort_dirs=desc' - ) - - self.assertEqual(200, resp.status) - self.assertEqual(1, len(body['actions'])) - self.assertIn('next', body) - - name_1 = body['actions'][0].get('name') - next = body.get('next') - - param_dict = utils.get_dict_from_string( - next.split('?')[1], - delimiter='&' - ) - - expected_sub_dict = { - 'limit': 1, - 'sort_keys': 'name', - 'sort_dirs': 'desc' - } - - self.assertDictContainsSubset(expected_sub_dict, param_dict) - - # Query again using 'next' hint - url_param = next.split('/')[-1] - resp, body = self.client.get_list_obj(url_param) - - self.assertEqual(200, resp.status) - self.assertEqual(1, len(body['actions'])) - - name_2 = body['actions'][0].get('name') - - self.assertGreater(name_1, name_2) - - @test.attr(type='negative') - def test_get_list_actions_nonexist_sort_dirs(self): - context = self.assertRaises( - exceptions.BadRequest, - self.client.get_list_obj, - 'actions?limit=1&sort_keys=id&sort_dirs=nonexist' - ) - - self.assertIn( - 'Unknown sort direction', - context.resp_body.get('faultstring') - ) - - @test.attr(type='negative') - def test_get_list_actions_invalid_limit(self): - context = self.assertRaises( - exceptions.BadRequest, - self.client.get_list_obj, - 'actions?limit=-1&sort_keys=id&sort_dirs=asc' - ) - - self.assertIn( - 'Limit must be positive', - context.resp_body.get('faultstring') - ) - - @test.attr(type='negative') - def test_get_list_actions_duplicate_sort_keys(self): - context = self.assertRaises( - exceptions.BadRequest, - self.client.get_list_obj, - 'actions?limit=1&sort_keys=id,id&sort_dirs=asc,asc' - ) - - self.assertIn( - 'Length of sort_keys must be equal or greater than sort_dirs', - context.resp_body.get('faultstring') - ) - - @test.attr(type='sanity') - def test_create_and_delete_few_actions(self): - resp, body = self.client.create_action('action_v2.yaml') - self.assertEqual(201, resp.status) - - created_acts = [action['name'] for action in body['actions']] - - resp, body = self.client.get_list_obj('actions') - self.assertEqual(200, resp.status) - - actions = [action['name'] for action in body['actions']] - - for act in created_acts: - self.assertIn(act, actions) - self.client.delete_obj('actions', act) - - _, body = self.client.get_list_obj('actions') - actions = [action['name'] for action in body['actions']] - - for act in created_acts: - self.assertNotIn(act, actions) - self.client.actions.remove(act) - - @test.attr(type='sanity') - def test_get_action(self): - _, body = self.client.create_action('action_v2.yaml') - action_name = body['actions'][0]['name'] - resp, body = self.client.get_object('actions', action_name) - - self.assertEqual(200, resp.status) - self.assertEqual(action_name, body['name']) - - @test.attr(type='sanity') - def test_update_action(self): - _, body = self.client.create_action('action_v2.yaml') - action = body['actions'][0]['name'] - - act_created_at = self.get_field_value( - body=body, act_name=action, field='created_at') - - self.assertNotIn('updated at', body['actions']) - - resp, body = self.client.update_request('actions', 'action_v2.yaml') - self.assertEqual(200, resp.status) - - actions = [act['name'] for act in body['actions']] - self.assertIn(action, actions) - - updated_act_created_at = self.get_field_value( - body=body, act_name=action, field='created_at') - - self.assertEqual(act_created_at.split(".")[0], updated_act_created_at) - self.assertTrue(all(['updated_at' in item - for item in body['actions']])) - - @test.attr(type='sanity') - def test_get_action_definition(self): - _, body = self.client.create_action('action_v2.yaml') - act_name = body['actions'][0]['name'] - - resp, body = self.client.get_definition('actions', act_name) - self.assertEqual(200, resp.status) - self.assertIsNotNone(body) - self.assertIn(act_name, body) - - @test.attr(type='negative') - def test_get_nonexistent_action(self): - self.assertRaises( - exceptions.NotFound, - self.client.get_object, - 'actions', 'nonexist' - ) - - @test.attr(type='negative') - def test_double_creation(self): - self.client.create_action('action_v2.yaml') - - self.assertRaises( - exceptions.Conflict, - self.client.create_action, - 'action_v2.yaml' - ) - - @test.attr(type='negative') - def test_create_action_invalid_def(self): - self.assertRaises( - exceptions.BadRequest, - self.client.create_action, - 'wb_v2.yaml' - ) - - @test.attr(type='negative') - def test_update_action_invalid_def(self): - self.assertRaises( - exceptions.BadRequest, - self.client.update_request, - 'actions', 'wb_v2.yaml' - ) - - @test.attr(type='negative') - def test_delete_nonexistent_action(self): - self.assertRaises( - exceptions.NotFound, - self.client.delete_obj, - 'actions', 'nonexist' - ) - - @test.attr(type='negative') - def test_delete_standard_action(self): - self.assertRaises( - exceptions.BadRequest, - self.client.delete_obj, - 'actions', 'nova.servers_create' - ) - - -class TasksTestsV2(base.TestCase): - - _service = 'workflowv2' - - def setUp(self): - super(TasksTestsV2, self).setUp() - self.useFixture(lockutils.LockFixture('mistral-workflow')) - _, body = self.client.create_workflow('wf_v2.yaml') - self.direct_wf_name = body['workflows'][0]['name'] - _, execution = self.client.create_execution(self.direct_wf_name) - - def tearDown(self): - for wf in self.client.workflows: - self.client.delete_obj('workflows', wf) - self.client.workflows = [] - - for wf in self.client.executions: - self.client.delete_obj('executions', wf) - self.client.executions = [] - - super(TasksTestsV2, self).tearDown() - - @test.attr(type='smoke') - def test_get_tasks_list(self): - resp, body = self.client.get_list_obj('tasks') - - self.assertEqual(200, resp.status) - self.assertNotEmpty(body['tasks']) - - @test.attr(type='sanity') - def test_get_task(self): - resp, body = self.client.get_list_obj('tasks') - - self.assertEqual(200, resp.status) - self.assertEqual( - self.direct_wf_name, body['tasks'][-1]['workflow_name'] - ) - - -class ActionExecutionTestsV2(base.TestCase): - _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') - def test_run_action_execution(self): - resp, body = self.client.create_action_execution( - { - 'name': 'std.echo', - 'input': '{"output": "Hello, Mistral!"}' - } - ) - - self.assertEqual(201, resp.status) - output = json.loads(body['output']) - self.assertDictEqual( - {'result': 'Hello, Mistral!'}, - output - ) - - @test.attr(type='sanity') - def test_run_action_std_http(self): - resp, body = self.client.create_action_execution( - { - 'name': 'std.http', - 'input': '{"url": "http://wiki.openstack.org"}' - } - ) - - self.assertEqual(201, resp.status) - output = json.loads(body['output']) - self.assertTrue(output['result']['status'] in range(200, 307)) - - @test.attr(type='sanity') - def test_run_action_std_http_error(self): - resp, body = self.client.create_action_execution( - { - 'name': 'std.http', - 'input': '{"url": "http://www.google.ru/not-found-test"}' - } - ) - - self.assertEqual(201, resp.status) - output = json.loads(body['output']) - self.assertEqual(404, output['result']['status']) - - @test.attr(type='sanity') - def test_create_action_execution(self): - resp, body = self.client.create_action_execution( - { - 'name': 'std.echo', - 'input': '{"output": "Hello, Mistral!"}', - 'params': '{"save_result": true}' - } - ) - - self.assertEqual(201, resp.status) - self.assertEqual('RUNNING', body['state']) - - # We must reread action execution in order to get actual - # state and output. - body = self.client.wait_execution_success( - body, - url='action_executions' - ) - output = json.loads(body['output']) - - self.assertEqual('SUCCESS', body['state']) - self.assertDictEqual( - {'result': 'Hello, Mistral!'}, - output - ) - - @test.attr(type='negative') - def test_delete_nonexistent_action_execution(self): - self.assertRaises( - exceptions.NotFound, - self.client.delete_obj, - 'action_executions', - 'nonexist' - ) - - @test.attr(type='sanity') - def test_create_action_execution_sync(self): - token = self.client.auth_provider.get_token() - resp, body = self.client.create_action_execution( - { - 'name': 'std.http', - 'input': '{{"url": "http://localhost:8989/v2/workflows",\ - "headers": {{"X-Auth-Token": "{}"}}}}'.format(token) - } - ) - - self.assertEqual(201, resp.status) - output = json.loads(body['output']) - self.assertEqual(200, output['result']['status']) diff --git a/mistral_tempest_tests/tests/api/v2/test_tasks.py b/mistral_tempest_tests/tests/api/v2/test_tasks.py new file mode 100644 index 0000000..8936129 --- /dev/null +++ b/mistral_tempest_tests/tests/api/v2/test_tasks.py @@ -0,0 +1,61 @@ +# Copyright 2016 NEC Corporation. All rights reserved. +# +# 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 oslo_concurrency.fixture import lockutils +from oslo_log import log as logging +from tempest import test + +from mistral_tempest_tests.tests import base + + +LOG = logging.getLogger(__name__) + + +class TasksTestsV2(base.TestCase): + + _service = 'workflowv2' + + def setUp(self): + super(TasksTestsV2, self).setUp() + self.useFixture(lockutils.LockFixture('mistral-workflow')) + _, body = self.client.create_workflow('wf_v2.yaml') + self.direct_wf_name = body['workflows'][0]['name'] + _, execution = self.client.create_execution(self.direct_wf_name) + + def tearDown(self): + for wf in self.client.workflows: + self.client.delete_obj('workflows', wf) + self.client.workflows = [] + + for wf in self.client.executions: + self.client.delete_obj('executions', wf) + self.client.executions = [] + + super(TasksTestsV2, self).tearDown() + + @test.attr(type='smoke') + def test_get_tasks_list(self): + resp, body = self.client.get_list_obj('tasks') + + self.assertEqual(200, resp.status) + self.assertNotEmpty(body['tasks']) + + @test.attr(type='sanity') + def test_get_task(self): + resp, body = self.client.get_list_obj('tasks') + + self.assertEqual(200, resp.status) + self.assertEqual( + self.direct_wf_name, body['tasks'][-1]['workflow_name'] + ) diff --git a/mistral_tempest_tests/tests/api/v2/test_workbooks.py b/mistral_tempest_tests/tests/api/v2/test_workbooks.py new file mode 100644 index 0000000..a9469df --- /dev/null +++ b/mistral_tempest_tests/tests/api/v2/test_workbooks.py @@ -0,0 +1,132 @@ +# Copyright 2016 NEC Corporation. All rights reserved. +# +# 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 oslo_concurrency.fixture import lockutils +from oslo_log import log as logging +from tempest.lib import exceptions +from tempest import test + +from mistral_tempest_tests.tests import base + + +LOG = logging.getLogger(__name__) + + +class WorkbookTestsV2(base.TestCase): + + _service = 'workflowv2' + + def tearDown(self): + for wf in self.client.workflows: + self.client.delete_obj('workflows', wf) + self.client.workflows = [] + + super(WorkbookTestsV2, self).tearDown() + + @test.attr(type='smoke') + def test_get_list_workbooks(self): + resp, body = self.client.get_list_obj('workbooks') + + self.assertEqual(200, resp.status) + self.assertEqual([], body['workbooks']) + + @test.attr(type='sanity') + def test_create_and_delete_workbook(self): + self.useFixture(lockutils.LockFixture('mistral-workflow')) + resp, body = self.client.create_workbook('wb_v2.yaml') + name = body['name'] + self.assertEqual(201, resp.status) + + resp, body = self.client.get_list_obj('workbooks') + self.assertEqual(200, resp.status) + self.assertEqual(name, body['workbooks'][0]['name']) + + self.client.delete_obj('workbooks', name) + self.client.workbooks.remove(name) + + _, body = self.client.get_list_obj('workbooks') + self.assertEqual([], body['workbooks']) + + @test.attr(type='sanity') + def test_get_workbook(self): + self.useFixture(lockutils.LockFixture('mistral-workflow')) + _, body = self.client.create_workbook('wb_v2.yaml') + name = body['name'] + + resp, body = self.client.get_object('workbooks', name) + self.assertEqual(200, resp.status) + self.assertEqual(name, body['name']) + + @test.attr(type='sanity') + def test_update_workbook(self): + self.useFixture(lockutils.LockFixture('mistral-workflow')) + _, body = self.client.create_workbook('wb_v2.yaml') + name = body['name'] + resp, body = self.client.update_request('workbooks', 'wb_v2.yaml') + + self.assertEqual(200, resp.status) + self.assertEqual(name, body['name']) + + @test.attr(type='sanity') + def test_get_workbook_definition(self): + self.useFixture(lockutils.LockFixture('mistral-workflow')) + _, body = self.client.create_workbook('wb_v2.yaml') + name = body['name'] + resp, body = self.client.get_definition('workbooks', name) + + self.assertEqual(200, resp.status) + self.assertIsNotNone(body) + + @test.attr(type='negative') + def test_get_nonexistent_workbook_definition(self): + self.assertRaises(exceptions.NotFound, + self.client.get_definition, + 'workbooks', 'nonexist') + + @test.attr(type='negative') + def test_get_nonexistent_workbook(self): + self.assertRaises(exceptions.NotFound, self.client.get_object, + 'workbooks', 'nonexist') + + @test.attr(type='negative') + def test_double_create_workbook(self): + self.useFixture(lockutils.LockFixture('mistral-workflow')) + _, body = self.client.create_workbook('wb_v2.yaml') + name = body['name'] + self.assertRaises(exceptions.Conflict, + self.client.create_workbook, + 'wb_v2.yaml') + + self.client.delete_obj('workbooks', name) + self.client.workbooks.remove(name) + _, body = self.client.get_list_obj('workbooks') + + self.assertEqual([], body['workbooks']) + + @test.attr(type='negative') + def test_create_wb_with_invalid_def(self): + self.assertRaises( + exceptions.BadRequest, + self.client.create_workbook, + 'wb_v1.yaml' + ) + + @test.attr(type='negative') + def test_update_wb_with_invalid_def(self): + self.assertRaises( + exceptions.BadRequest, + self.client.update_request, + 'workbooks', + 'wb_v1.yaml' + ) diff --git a/mistral_tempest_tests/tests/api/v2/test_workflows.py b/mistral_tempest_tests/tests/api/v2/test_workflows.py new file mode 100644 index 0000000..890e4dd --- /dev/null +++ b/mistral_tempest_tests/tests/api/v2/test_workflows.py @@ -0,0 +1,284 @@ +# Copyright 2016 NEC Corporation. All rights reserved. +# +# 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 oslo_concurrency.fixture import lockutils +from oslo_log import log as logging +from tempest.lib import exceptions +from tempest import test + +from mistral import utils +from mistral_tempest_tests.tests import base + + +LOG = logging.getLogger(__name__) + + +class WorkflowTestsV2(base.TestCase): + + _service = 'workflowv2' + + def tearDown(self): + for wf in self.client.workflows: + self.client.delete_obj('workflows', wf) + self.client.workflows = [] + + super(WorkflowTestsV2, self).tearDown() + + @test.attr(type='smoke') + def test_get_list_workflows(self): + resp, body = self.client.get_list_obj('workflows') + self.assertEqual(200, resp.status) + + names = [wf['name'] for wf in body['workflows']] + + self.assertIn('std.create_instance', names) + + self.assertNotIn('next', body) + + @test.attr(type='smoke') + def test_get_list_workflows_with_fields(self): + resp, body = self.client.get_list_obj('workflows?fields=name') + + self.assertEqual(200, resp.status) + + for wf in body['workflows']: + self.assertListEqual(sorted(['id', 'name']), sorted(list(wf))) + + @test.attr(type='smoke') + def test_get_list_workflows_with_pagination(self): + resp, body = self.client.get_list_obj( + 'workflows?limit=1&sort_keys=name&sort_dirs=desc' + ) + + self.assertEqual(200, resp.status) + self.assertEqual(1, len(body['workflows'])) + self.assertIn('next', body) + + name_1 = body['workflows'][0].get('name') + next = body.get('next') + + param_dict = utils.get_dict_from_string( + next.split('?')[1], + delimiter='&' + ) + + expected_sub_dict = { + 'limit': 1, + 'sort_keys': 'name', + 'sort_dirs': 'desc' + } + + self.assertDictContainsSubset(expected_sub_dict, param_dict) + + # Query again using 'next' hint + url_param = next.split('/')[-1] + resp, body = self.client.get_list_obj(url_param) + + self.assertEqual(200, resp.status) + self.assertEqual(1, len(body['workflows'])) + + name_2 = body['workflows'][0].get('name') + + self.assertGreater(name_1, name_2) + + @test.attr(type='negative') + def test_get_list_workflows_nonexist_sort_dirs(self): + context = self.assertRaises( + exceptions.BadRequest, + self.client.get_list_obj, + 'workflows?limit=1&sort_keys=id&sort_dirs=nonexist' + ) + + self.assertIn( + 'Unknown sort direction', + context.resp_body.get('faultstring') + ) + + @test.attr(type='negative') + def test_get_list_workflows_invalid_limit(self): + context = self.assertRaises( + exceptions.BadRequest, + self.client.get_list_obj, + 'workflows?limit=-1&sort_keys=id&sort_dirs=asc' + ) + + self.assertIn( + 'Limit must be positive', + context.resp_body.get('faultstring') + ) + + @test.attr(type='negative') + def test_get_list_workflows_duplicate_sort_keys(self): + context = self.assertRaises( + exceptions.BadRequest, + self.client.get_list_obj, + 'workflows?limit=1&sort_keys=id,id&sort_dirs=asc,asc' + ) + + self.assertIn( + 'Length of sort_keys must be equal or greater than sort_dirs', + context.resp_body.get('faultstring') + ) + + @test.attr(type='sanity') + def test_create_and_delete_workflow(self): + self.useFixture(lockutils.LockFixture('mistral-workflow')) + resp, body = self.client.create_workflow('wf_v2.yaml') + name = body['workflows'][0]['name'] + + self.assertEqual(201, resp.status) + + resp, body = self.client.get_list_obj('workflows') + self.assertEqual(200, resp.status) + + names = [wf['name'] for wf in body['workflows']] + self.assertIn(name, names) + + self.client.delete_obj('workflows', name) + self.client.workflows.remove(name) + + _, body = self.client.get_list_obj('workflows') + + names = [wf['name'] for wf in body['workflows']] + self.assertNotIn(name, names) + + @test.attr(type='sanity') + def test_get_workflow(self): + self.useFixture(lockutils.LockFixture('mistral-workflow')) + _, body = self.client.create_workflow('wf_v2.yaml') + name = body['workflows'][0]['name'] + + resp, body = self.client.get_object('workflows', name) + + self.assertEqual(200, resp.status) + self.assertEqual(name, body['name']) + + @test.attr(type='sanity') + def test_update_workflow(self): + self.useFixture(lockutils.LockFixture('mistral-workflow')) + _, body = self.client.create_workflow('wf_v2.yaml') + name = body['workflows'][0]['name'] + + resp, body = self.client.update_request('workflows', 'wf_v2.yaml') + + self.assertEqual(200, resp.status) + self.assertEqual(name, body['workflows'][0]['name']) + + @test.attr(type='sanity') + def test_get_workflow_definition(self): + self.useFixture(lockutils.LockFixture('mistral-workflow')) + _, body = self.client.create_workflow('wf_v2.yaml') + name = body['workflows'][0]['name'] + + resp, body = self.client.get_definition('workflows', name) + + self.assertEqual(200, resp.status) + self.assertIsNotNone(body) + + @test.attr(type='sanity') + def test_get_workflow_uploaded_in_wb(self): + self.useFixture(lockutils.LockFixture('mistral-workflow')) + _, body = self.client.create_workbook('wb_v2.yaml') + wb_name = body['name'] + + _, body = self.client.get_list_obj('workflows') + wf_names = [wf['name'] for wf in body['workflows'] + if wf['name'].startswith(wb_name)] + + self.assertNotEmpty(wf_names) + + @test.attr(type='negative') + def test_get_nonexistent_workflow_definition(self): + self.assertRaises(exceptions.NotFound, + self.client.get_definition, + 'workflows', 'nonexist') + + @test.attr(type='negative') + def test_get_nonexistent_workflow(self): + self.assertRaises(exceptions.NotFound, self.client.get_object, + 'workflows', 'nonexist') + + @test.attr(type='negative') + def test_double_create_workflows(self): + self.useFixture(lockutils.LockFixture('mistral-workflow')) + _, body = self.client.create_workflow('wf_v2.yaml') + self.assertRaises(exceptions.Conflict, + self.client.create_workflow, + 'wf_v2.yaml') + + @test.attr(type='negative') + def test_create_wf_with_invalid_def(self): + self.assertRaises(exceptions.BadRequest, + self.client.create_workflow, + 'wb_v1.yaml') + + @test.attr(type='negative') + def test_update_wf_with_invalid_def(self): + self.assertRaises(exceptions.BadRequest, + self.client.update_request, + 'workflows', 'wb_v1.yaml') + + @test.attr(type='negative') + def test_delete_wf_with_trigger_associate(self): + tr_name = 'trigger' + resp, body = self.client.create_workflow('wf_v2.yaml') + name = body['workflows'][0]['name'] + resp, body = self.client.create_cron_trigger( + tr_name, name, None, '5 * * * *') + + try: + self.assertRaises( + exceptions.BadRequest, + self.client.delete_obj, + 'workflows', + name + ) + finally: + self.client.delete_obj('cron_triggers', tr_name) + self.client.triggers.remove(tr_name) + + @test.attr(type='negative') + def test_delete_wf_with_trigger_associate_in_other_tenant(self): + self.useFixture(lockutils.LockFixture('mistral-workflow')) + tr_name = 'trigger' + _, body = self.client.create_workflow('wf_v2.yaml', scope='public') + name = body['workflows'][0]['name'] + resp, body = self.alt_client.create_cron_trigger( + tr_name, + name, + None, + '5 * * * *' + ) + + try: + exception = self.assertRaises( + exceptions.BadRequest, + self.client.delete_obj, + 'workflows', + name + ) + + self.assertIn( + "Can't delete workflow that has triggers associated", + exception.resp_body['faultstring'] + ) + finally: + self.alt_client.delete_obj('cron_triggers', tr_name) + self.alt_client.triggers.remove(tr_name) + + @test.attr(type='negative') + def test_delete_nonexistent_wf(self): + self.assertRaises(exceptions.NotFound, + self.client.delete_obj, + 'workflows', 'nonexist') diff --git a/mistral_tempest_tests/tests/base.py b/mistral_tempest_tests/tests/base.py new file mode 100644 index 0000000..56a2e56 --- /dev/null +++ b/mistral_tempest_tests/tests/base.py @@ -0,0 +1,98 @@ +# Copyright 2016 NEC Corporation. All rights reserved. +# +# 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 os + +import mock +from tempest import clients +from tempest import config +from tempest import test as test + +from mistral_tempest_tests.services import base as service_base +from mistral_tempest_tests.services.v2 import mistral_client + +CONF = config.CONF + + +class TestCase(test.BaseTestCase): + credentials = ['primary', 'alt'] + + @classmethod + def skip_checks(cls): + super(TestCase, cls).skip_checks() + + if not CONF.service_available.mistral: + raise cls.skipException("Mistral support is required.") + + @classmethod + def resource_setup(cls): + """Client authentication. + + This method allows to initialize authentication before + each test case and define parameters of Mistral API Service. + """ + super(TestCase, cls).resource_setup() + + if 'WITHOUT_AUTH' in os.environ: + cls.mgr = mock.MagicMock() + cls.mgr.auth_provider = service_base.AuthProv() + cls.alt_mgr = cls.mgr + else: + cls.mgr = cls.manager + cls.alt_mgr = cls.alt_manager + + if cls._service == 'workflowv2': + cls.client = mistral_client.MistralClientV2( + cls.mgr.auth_provider, cls._service) + cls.alt_client = mistral_client.MistralClientV2( + cls.alt_mgr.auth_provider, cls._service) + + def setUp(self): + super(TestCase, self).setUp() + + def tearDown(self): + super(TestCase, self).tearDown() + + for wb in self.client.workbooks: + self.client.delete_obj('workbooks', wb) + + self.client.workbooks = [] + + +class TestCaseAdvanced(TestCase): + @classmethod + def resource_setup(cls): + super(TestCaseAdvanced, cls).resource_setup() + + cls.server_client = clients.ServersClient( + cls.mgr.auth_provider, + "compute", + region=CONF.identity.region + ) + + cls.image_ref = CONF.compute.image_ref + cls.flavor_ref = CONF.compute.flavor_ref + + def tearDown(self): + for wb in self.client.workbooks: + self.client.delete_obj('workbooks', wb) + + self.client.workbooks = [] + + for ex in self.client.executions: + self.client.delete_obj('executions', ex) + + self.client.executions = [] + + super(TestCaseAdvanced, self).tearDown() diff --git a/mistral_tempest_tests/tests/scenario/engine/actions/v2/test_openstack_actions.py b/mistral_tempest_tests/tests/scenario/engine/actions/v2/test_openstack_actions.py index 69f1069..3518251 100644 --- a/mistral_tempest_tests/tests/scenario/engine/actions/v2/test_openstack_actions.py +++ b/mistral_tempest_tests/tests/scenario/engine/actions/v2/test_openstack_actions.py @@ -14,7 +14,7 @@ from tempest import test -from mistral_tempest_tests.services import base +from mistral_tempest_tests.tests import base class OpenStackActionsTestsV2(base.TestCase): diff --git a/mistral_tempest_tests/tests/scenario/engine/actions/v2/test_ssh_actions.py b/mistral_tempest_tests/tests/scenario/engine/actions/v2/test_ssh_actions.py index 7697cf9..e417d5a 100644 --- a/mistral_tempest_tests/tests/scenario/engine/actions/v2/test_ssh_actions.py +++ b/mistral_tempest_tests/tests/scenario/engine/actions/v2/test_ssh_actions.py @@ -25,7 +25,7 @@ from tempest import test from mistral import utils from mistral.utils import ssh_utils -from mistral_tempest_tests.services import base +from mistral_tempest_tests.tests import base LOG = logging.getLogger(__name__)