Adding pagination support to mistral python client
Closes-Bug: #1495604 Change-Id: Ibf26b9ef4f7ae655bcd88fad863985f7309a4d17
This commit is contained in:
@@ -12,9 +12,14 @@
|
|||||||
# See the License for the specific language governing permissions and
|
# See the License for the specific language governing permissions and
|
||||||
# limitations under the License.
|
# limitations under the License.
|
||||||
|
|
||||||
|
import six
|
||||||
|
|
||||||
from mistralclient.api import base
|
from mistralclient.api import base
|
||||||
|
|
||||||
|
|
||||||
|
urlparse = six.moves.urllib.parse
|
||||||
|
|
||||||
|
|
||||||
class Action(base.Resource):
|
class Action(base.Resource):
|
||||||
resource_name = 'Action'
|
resource_name = 'Action'
|
||||||
|
|
||||||
@@ -52,8 +57,28 @@ class ActionManager(base.ResourceManager):
|
|||||||
return [self.resource_class(self, resource_data)
|
return [self.resource_class(self, resource_data)
|
||||||
for resource_data in base.extract_json(resp, 'actions')]
|
for resource_data in base.extract_json(resp, 'actions')]
|
||||||
|
|
||||||
def list(self):
|
def list(self, marker='', limit=None, sort_keys='', sort_dirs=''):
|
||||||
return self._list('/actions', response_key='actions')
|
qparams = {}
|
||||||
|
|
||||||
|
if marker:
|
||||||
|
qparams['marker'] = marker
|
||||||
|
|
||||||
|
if limit:
|
||||||
|
qparams['limit'] = limit
|
||||||
|
|
||||||
|
if sort_keys:
|
||||||
|
qparams['sort_keys'] = sort_keys
|
||||||
|
|
||||||
|
if sort_dirs:
|
||||||
|
qparams['sort_dirs'] = sort_dirs
|
||||||
|
|
||||||
|
query_string = ("?%s" % urlparse.urlencode(list(qparams.items()))
|
||||||
|
if qparams else "")
|
||||||
|
|
||||||
|
return self._list(
|
||||||
|
'/actions%s' % query_string,
|
||||||
|
response_key='actions',
|
||||||
|
)
|
||||||
|
|
||||||
def get(self, name):
|
def get(self, name):
|
||||||
self._ensure_not_empty(name=name)
|
self._ensure_not_empty(name=name)
|
||||||
|
@@ -19,6 +19,9 @@ import six
|
|||||||
from mistralclient.api import base
|
from mistralclient.api import base
|
||||||
|
|
||||||
|
|
||||||
|
urlparse = six.moves.urllib.parse
|
||||||
|
|
||||||
|
|
||||||
class Execution(base.Resource):
|
class Execution(base.Resource):
|
||||||
resource_name = 'Execution'
|
resource_name = 'Execution'
|
||||||
|
|
||||||
@@ -64,8 +67,28 @@ class ExecutionManager(base.ResourceManager):
|
|||||||
|
|
||||||
return self._update('/executions/%s' % id, data)
|
return self._update('/executions/%s' % id, data)
|
||||||
|
|
||||||
def list(self):
|
def list(self, marker='', limit=None, sort_keys='', sort_dirs=''):
|
||||||
return self._list('/executions', response_key='executions')
|
qparams = {}
|
||||||
|
|
||||||
|
if marker:
|
||||||
|
qparams['marker'] = marker
|
||||||
|
|
||||||
|
if limit:
|
||||||
|
qparams['limit'] = limit
|
||||||
|
|
||||||
|
if sort_keys:
|
||||||
|
qparams['sort_keys'] = sort_keys
|
||||||
|
|
||||||
|
if sort_dirs:
|
||||||
|
qparams['sort_dirs'] = sort_dirs
|
||||||
|
|
||||||
|
query_string = ("?%s" % urlparse.urlencode(list(qparams.items()))
|
||||||
|
if qparams else "")
|
||||||
|
|
||||||
|
return self._list(
|
||||||
|
'/executions%s' % query_string,
|
||||||
|
response_key='executions',
|
||||||
|
)
|
||||||
|
|
||||||
def get(self, id):
|
def get(self, id):
|
||||||
self._ensure_not_empty(id=id)
|
self._ensure_not_empty(id=id)
|
||||||
|
@@ -13,9 +13,14 @@
|
|||||||
# See the License for the specific language governing permissions and
|
# See the License for the specific language governing permissions and
|
||||||
# limitations under the License.
|
# limitations under the License.
|
||||||
|
|
||||||
|
import six
|
||||||
|
|
||||||
from mistralclient.api import base
|
from mistralclient.api import base
|
||||||
|
|
||||||
|
|
||||||
|
urlparse = six.moves.urllib.parse
|
||||||
|
|
||||||
|
|
||||||
class Workflow(base.Resource):
|
class Workflow(base.Resource):
|
||||||
resource_name = 'Workflow'
|
resource_name = 'Workflow'
|
||||||
|
|
||||||
@@ -53,8 +58,28 @@ class WorkflowManager(base.ResourceManager):
|
|||||||
return [self.resource_class(self, resource_data)
|
return [self.resource_class(self, resource_data)
|
||||||
for resource_data in base.extract_json(resp, 'workflows')]
|
for resource_data in base.extract_json(resp, 'workflows')]
|
||||||
|
|
||||||
def list(self):
|
def list(self, marker='', limit=None, sort_keys='', sort_dirs=''):
|
||||||
return self._list('/workflows', response_key='workflows')
|
qparams = {}
|
||||||
|
|
||||||
|
if marker:
|
||||||
|
qparams['marker'] = marker
|
||||||
|
|
||||||
|
if limit:
|
||||||
|
qparams['limit'] = limit
|
||||||
|
|
||||||
|
if sort_keys:
|
||||||
|
qparams['sort_keys'] = sort_keys
|
||||||
|
|
||||||
|
if sort_dirs:
|
||||||
|
qparams['sort_dirs'] = sort_dirs
|
||||||
|
|
||||||
|
query_string = ("?%s" % urlparse.urlencode(list(qparams.items()))
|
||||||
|
if qparams else "")
|
||||||
|
|
||||||
|
return self._list(
|
||||||
|
'/workflows%s' % query_string,
|
||||||
|
response_key='workflows',
|
||||||
|
)
|
||||||
|
|
||||||
def get(self, name):
|
def get(self, name):
|
||||||
self._ensure_not_empty(name=name)
|
self._ensure_not_empty(name=name)
|
||||||
|
@@ -29,4 +29,5 @@ class BaseClientV2Test(base.BaseClientTest):
|
|||||||
self.workflows = self._client.workflows
|
self.workflows = self._client.workflows
|
||||||
self.environments = self._client.environments
|
self.environments = self._client.environments
|
||||||
self.action_executions = self._client.action_executions
|
self.action_executions = self._client.action_executions
|
||||||
|
self.actions = self._client.actions
|
||||||
self.services = self._client.services
|
self.services = self._client.services
|
||||||
|
123
mistralclient/tests/unit/v2/test_actions.py
Normal file
123
mistralclient/tests/unit/v2/test_actions.py
Normal file
@@ -0,0 +1,123 @@
|
|||||||
|
# Copyright 2015 Huawei Technologies Co., Ltd.
|
||||||
|
#
|
||||||
|
# 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 mistralclient.api.v2 import actions
|
||||||
|
from mistralclient.tests.unit.v2 import base
|
||||||
|
|
||||||
|
|
||||||
|
ACTION_DEF = """
|
||||||
|
---
|
||||||
|
version: 2.0
|
||||||
|
|
||||||
|
my_action:
|
||||||
|
base: std.echo
|
||||||
|
base-input:
|
||||||
|
output: 'Bye!'
|
||||||
|
output:
|
||||||
|
info: <% $.output %>
|
||||||
|
"""
|
||||||
|
|
||||||
|
ACTION = {
|
||||||
|
'id': '123',
|
||||||
|
'name': 'my_action',
|
||||||
|
'input': '',
|
||||||
|
'definition': ACTION_DEF
|
||||||
|
}
|
||||||
|
|
||||||
|
URL_TEMPLATE = '/actions'
|
||||||
|
URL_TEMPLATE_SCOPE = '/actions?scope=private'
|
||||||
|
URL_TEMPLATE_NAME = '/actions/%s'
|
||||||
|
|
||||||
|
|
||||||
|
class TestActionsV2(base.BaseClientV2Test):
|
||||||
|
def test_create(self):
|
||||||
|
mock = self.mock_http_post(content={'actions': [ACTION]})
|
||||||
|
|
||||||
|
actions = self.actions.create(ACTION_DEF)
|
||||||
|
|
||||||
|
self.assertIsNotNone(actions)
|
||||||
|
self.assertEqual(ACTION_DEF, actions[0].definition)
|
||||||
|
|
||||||
|
mock.assert_called_once_with(
|
||||||
|
URL_TEMPLATE_SCOPE,
|
||||||
|
ACTION_DEF,
|
||||||
|
headers={'content-type': 'text/plain'}
|
||||||
|
)
|
||||||
|
|
||||||
|
def test_update(self):
|
||||||
|
mock = self.mock_http_put(content={'actions': [ACTION]})
|
||||||
|
|
||||||
|
actions = self.actions.update(ACTION_DEF)
|
||||||
|
|
||||||
|
self.assertIsNotNone(actions)
|
||||||
|
self.assertEqual(ACTION_DEF, actions[0].definition)
|
||||||
|
|
||||||
|
mock.assert_called_once_with(
|
||||||
|
URL_TEMPLATE_SCOPE,
|
||||||
|
ACTION_DEF,
|
||||||
|
headers={'content-type': 'text/plain'}
|
||||||
|
)
|
||||||
|
|
||||||
|
def test_list(self):
|
||||||
|
mock = self.mock_http_get(content={'actions': [ACTION]})
|
||||||
|
|
||||||
|
action_list = self.actions.list()
|
||||||
|
|
||||||
|
self.assertEqual(1, len(action_list))
|
||||||
|
|
||||||
|
action = action_list[0]
|
||||||
|
|
||||||
|
self.assertEqual(
|
||||||
|
actions.Action(self.actions, ACTION).to_dict(),
|
||||||
|
action.to_dict()
|
||||||
|
)
|
||||||
|
|
||||||
|
mock.assert_called_once_with(URL_TEMPLATE)
|
||||||
|
|
||||||
|
def test_list_with_pagination(self):
|
||||||
|
mock = self.mock_http_get(
|
||||||
|
content={'actions': [ACTION], 'next': '/actions?fake'}
|
||||||
|
)
|
||||||
|
|
||||||
|
action_list = self.actions.list(
|
||||||
|
limit=1,
|
||||||
|
sort_keys='created_at',
|
||||||
|
sort_dirs='asc'
|
||||||
|
)
|
||||||
|
|
||||||
|
self.assertEqual(1, len(action_list))
|
||||||
|
|
||||||
|
# The url param order is unpredictable.
|
||||||
|
self.assertIn('limit=1', mock.call_args[0][0])
|
||||||
|
self.assertIn('sort_keys=created_at', mock.call_args[0][0])
|
||||||
|
self.assertIn('sort_dirs=asc', mock.call_args[0][0])
|
||||||
|
|
||||||
|
def test_get(self):
|
||||||
|
mock = self.mock_http_get(content=ACTION)
|
||||||
|
|
||||||
|
action = self.actions.get('action')
|
||||||
|
|
||||||
|
self.assertIsNotNone(action)
|
||||||
|
self.assertEqual(
|
||||||
|
actions.Action(self.actions, ACTION).to_dict(),
|
||||||
|
action.to_dict()
|
||||||
|
)
|
||||||
|
|
||||||
|
mock.assert_called_once_with(URL_TEMPLATE_NAME % 'action')
|
||||||
|
|
||||||
|
def test_delete(self):
|
||||||
|
mock = self.mock_http_delete(status_code=204)
|
||||||
|
|
||||||
|
self.actions.delete('action')
|
||||||
|
|
||||||
|
mock.assert_called_once_with(URL_TEMPLATE_NAME % 'action')
|
@@ -93,6 +93,24 @@ class TestExecutionsV2(base.BaseClientV2Test):
|
|||||||
ex.to_dict())
|
ex.to_dict())
|
||||||
mock.assert_called_once_with(URL_TEMPLATE)
|
mock.assert_called_once_with(URL_TEMPLATE)
|
||||||
|
|
||||||
|
def test_list_with_pagination(self):
|
||||||
|
mock = self.mock_http_get(
|
||||||
|
content={'executions': [EXEC], 'next': '/executions?fake'}
|
||||||
|
)
|
||||||
|
|
||||||
|
execution_list = self.executions.list(
|
||||||
|
limit=1,
|
||||||
|
sort_keys='created_at',
|
||||||
|
sort_dirs='asc'
|
||||||
|
)
|
||||||
|
|
||||||
|
self.assertEqual(1, len(execution_list))
|
||||||
|
|
||||||
|
# The url param order is unpredictable.
|
||||||
|
self.assertIn('limit=1', mock.call_args[0][0])
|
||||||
|
self.assertIn('sort_keys=created_at', mock.call_args[0][0])
|
||||||
|
self.assertIn('sort_dirs=asc', mock.call_args[0][0])
|
||||||
|
|
||||||
def test_get(self):
|
def test_get(self):
|
||||||
mock = self.mock_http_get(content=EXEC)
|
mock = self.mock_http_get(content=EXEC)
|
||||||
|
|
||||||
|
123
mistralclient/tests/unit/v2/test_workflows.py
Normal file
123
mistralclient/tests/unit/v2/test_workflows.py
Normal file
@@ -0,0 +1,123 @@
|
|||||||
|
# Copyright 2015 Huawei Technologies Co., Ltd.
|
||||||
|
#
|
||||||
|
# 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 mistralclient.api.v2 import workflows
|
||||||
|
from mistralclient.tests.unit.v2 import base
|
||||||
|
|
||||||
|
|
||||||
|
WF_DEF = """
|
||||||
|
---
|
||||||
|
version: 2.0
|
||||||
|
|
||||||
|
my_wf:
|
||||||
|
type: direct
|
||||||
|
|
||||||
|
tasks:
|
||||||
|
task1:
|
||||||
|
action: std.echo output="hello, world"
|
||||||
|
"""
|
||||||
|
|
||||||
|
WORKFLOW = {
|
||||||
|
'id': '123',
|
||||||
|
'name': 'my_wf',
|
||||||
|
'input': '',
|
||||||
|
'definition': WF_DEF
|
||||||
|
}
|
||||||
|
|
||||||
|
URL_TEMPLATE = '/workflows'
|
||||||
|
URL_TEMPLATE_SCOPE = '/workflows?scope=private'
|
||||||
|
URL_TEMPLATE_NAME = '/workflows/%s'
|
||||||
|
|
||||||
|
|
||||||
|
class TestWorkflowsV2(base.BaseClientV2Test):
|
||||||
|
def test_create(self):
|
||||||
|
mock = self.mock_http_post(content={'workflows': [WORKFLOW]})
|
||||||
|
|
||||||
|
wfs = self.workflows.create(WF_DEF)
|
||||||
|
|
||||||
|
self.assertIsNotNone(wfs)
|
||||||
|
self.assertEqual(WF_DEF, wfs[0].definition)
|
||||||
|
|
||||||
|
mock.assert_called_once_with(
|
||||||
|
URL_TEMPLATE_SCOPE,
|
||||||
|
WF_DEF,
|
||||||
|
headers={'content-type': 'text/plain'}
|
||||||
|
)
|
||||||
|
|
||||||
|
def test_update(self):
|
||||||
|
mock = self.mock_http_put(content={'workflows': [WORKFLOW]})
|
||||||
|
|
||||||
|
wfs = self.workflows.update(WF_DEF)
|
||||||
|
|
||||||
|
self.assertIsNotNone(wfs)
|
||||||
|
self.assertEqual(WF_DEF, wfs[0].definition)
|
||||||
|
|
||||||
|
mock.assert_called_once_with(
|
||||||
|
URL_TEMPLATE_SCOPE,
|
||||||
|
WF_DEF,
|
||||||
|
headers={'content-type': 'text/plain'}
|
||||||
|
)
|
||||||
|
|
||||||
|
def test_list(self):
|
||||||
|
mock = self.mock_http_get(content={'workflows': [WORKFLOW]})
|
||||||
|
|
||||||
|
workflows_list = self.workflows.list()
|
||||||
|
|
||||||
|
self.assertEqual(1, len(workflows_list))
|
||||||
|
|
||||||
|
wf = workflows_list[0]
|
||||||
|
|
||||||
|
self.assertEqual(
|
||||||
|
workflows.Workflow(self.workflows, WORKFLOW).to_dict(),
|
||||||
|
wf.to_dict()
|
||||||
|
)
|
||||||
|
|
||||||
|
mock.assert_called_once_with(URL_TEMPLATE)
|
||||||
|
|
||||||
|
def test_list_with_pagination(self):
|
||||||
|
mock = self.mock_http_get(
|
||||||
|
content={'workflows': [WORKFLOW], 'next': '/workflows?fake'}
|
||||||
|
)
|
||||||
|
|
||||||
|
workflows_list = self.workflows.list(
|
||||||
|
limit=1,
|
||||||
|
sort_keys='created_at',
|
||||||
|
sort_dirs='asc'
|
||||||
|
)
|
||||||
|
|
||||||
|
self.assertEqual(1, len(workflows_list))
|
||||||
|
|
||||||
|
# The url param order is unpredictable.
|
||||||
|
self.assertIn('limit=1', mock.call_args[0][0])
|
||||||
|
self.assertIn('sort_keys=created_at', mock.call_args[0][0])
|
||||||
|
self.assertIn('sort_dirs=asc', mock.call_args[0][0])
|
||||||
|
|
||||||
|
def test_get(self):
|
||||||
|
mock = self.mock_http_get(content=WORKFLOW)
|
||||||
|
|
||||||
|
wf = self.workflows.get('wf')
|
||||||
|
|
||||||
|
self.assertIsNotNone(wf)
|
||||||
|
self.assertEqual(
|
||||||
|
workflows.Workflow(self.workflows, WORKFLOW).to_dict(),
|
||||||
|
wf.to_dict()
|
||||||
|
)
|
||||||
|
|
||||||
|
mock.assert_called_once_with(URL_TEMPLATE_NAME % 'wf')
|
||||||
|
|
||||||
|
def test_delete(self):
|
||||||
|
mock = self.mock_http_delete(status_code=204)
|
||||||
|
|
||||||
|
self.workflows.delete('wf')
|
||||||
|
|
||||||
|
mock.assert_called_once_with(URL_TEMPLATE_NAME % 'wf')
|
Reference in New Issue
Block a user