Refactored tempest tests
Change-Id: I8d0a4a8ea7f1936b0a7f89b79b0ddda2c30644fb
This commit is contained in:
parent
b03f1f1f87
commit
48db3ebbc7
@ -1,4 +1,5 @@
|
|||||||
# Copyright 2013 Mirantis, Inc. All Rights Reserved.
|
# 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
|
# 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
|
# not use this file except in compliance with the License. You may obtain
|
||||||
@ -16,16 +17,12 @@ import json
|
|||||||
import os
|
import os
|
||||||
import time
|
import time
|
||||||
|
|
||||||
import mock
|
|
||||||
import six
|
import six
|
||||||
|
|
||||||
from oslo_utils import uuidutils
|
|
||||||
from tempest import clients
|
|
||||||
from tempest import config
|
from tempest import config
|
||||||
from tempest.lib import auth
|
from tempest.lib import auth
|
||||||
from tempest.lib.common import rest_client
|
from tempest.lib.common import rest_client
|
||||||
from tempest.lib import exceptions
|
from tempest.lib import exceptions
|
||||||
from tempest import test as test
|
|
||||||
|
|
||||||
CONF = config.CONF
|
CONF = config.CONF
|
||||||
|
|
||||||
@ -116,119 +113,6 @@ class MistralClientBase(rest_client.RestClient):
|
|||||||
return ex_body
|
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):
|
class AuthProv(auth.KeystoneV2AuthProvider):
|
||||||
def __init__(self):
|
def __init__(self):
|
||||||
self.alt_part = None
|
self.alt_part = None
|
||||||
@ -244,76 +128,3 @@ class AuthProv(auth.KeystoneV2AuthProvider):
|
|||||||
|
|
||||||
def base_url(self, *args, **kwargs):
|
def base_url(self, *args, **kwargs):
|
||||||
return ''
|
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()
|
|
||||||
|
0
mistral_tempest_tests/services/v2/__init__.py
Normal file
0
mistral_tempest_tests/services/v2/__init__.py
Normal file
139
mistral_tempest_tests/services/v2/mistral_client.py
Normal file
139
mistral_tempest_tests/services/v2/mistral_client.py
Normal file
@ -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)
|
136
mistral_tempest_tests/tests/api/v2/test_action_executions.py
Normal file
136
mistral_tempest_tests/tests/api/v2/test_action_executions.py
Normal file
@ -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'])
|
239
mistral_tempest_tests/tests/api/v2/test_actions.py
Normal file
239
mistral_tempest_tests/tests/api/v2/test_actions.py
Normal file
@ -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'
|
||||||
|
)
|
224
mistral_tempest_tests/tests/api/v2/test_cron_triggers.py
Normal file
224
mistral_tempest_tests/tests/api/v2/test_cron_triggers.py
Normal file
@ -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 * * * *')
|
269
mistral_tempest_tests/tests/api/v2/test_executions.py
Normal file
269
mistral_tempest_tests/tests/api/v2/test_executions.py
Normal file
@ -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')
|
File diff suppressed because it is too large
Load Diff
61
mistral_tempest_tests/tests/api/v2/test_tasks.py
Normal file
61
mistral_tempest_tests/tests/api/v2/test_tasks.py
Normal file
@ -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']
|
||||||
|
)
|
132
mistral_tempest_tests/tests/api/v2/test_workbooks.py
Normal file
132
mistral_tempest_tests/tests/api/v2/test_workbooks.py
Normal file
@ -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'
|
||||||
|
)
|
284
mistral_tempest_tests/tests/api/v2/test_workflows.py
Normal file
284
mistral_tempest_tests/tests/api/v2/test_workflows.py
Normal file
@ -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')
|
98
mistral_tempest_tests/tests/base.py
Normal file
98
mistral_tempest_tests/tests/base.py
Normal file
@ -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()
|
@ -14,7 +14,7 @@
|
|||||||
|
|
||||||
from tempest import test
|
from tempest import test
|
||||||
|
|
||||||
from mistral_tempest_tests.services import base
|
from mistral_tempest_tests.tests import base
|
||||||
|
|
||||||
|
|
||||||
class OpenStackActionsTestsV2(base.TestCase):
|
class OpenStackActionsTestsV2(base.TestCase):
|
||||||
|
@ -25,7 +25,7 @@ from tempest import test
|
|||||||
|
|
||||||
from mistral import utils
|
from mistral import utils
|
||||||
from mistral.utils import ssh_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__)
|
LOG = logging.getLogger(__name__)
|
||||||
|
Loading…
Reference in New Issue
Block a user