Modify API to include cluster related operations

This patch adds new API /cluster that allows summary and detailed
listings, show and update operations.

It also updates service listings to return cluster_name for each
service.

DocImpact: 3 new policies have been added for cluster, "get", "get_all" and
           "update".
APIImpact: Return cluster_name in service listings and add /cluster endpoint.
Specs: https://review.openstack.org/327283
Implements: blueprint cinder-volume-active-active-support
Change-Id: If1ef3a80900ca6d117bf854ad3de142d93694adf
This commit is contained in:
Gorka Eguileor 2016-05-23 14:24:33 +02:00
parent 62f761ff16
commit 8b713e5327
13 changed files with 660 additions and 22 deletions

View File

@ -85,6 +85,11 @@ class ServiceController(wsgi.Controller):
'zone': svc.availability_zone, 'zone': svc.availability_zone,
'status': active, 'state': art, 'status': active, 'state': art,
'updated_at': updated_at} 'updated_at': updated_at}
# On V3.7 we added cluster support
if req.api_version_request.matches('3.7'):
ret_fields['cluster'] = svc.cluster_name
if detailed: if detailed:
ret_fields['disabled_reason'] = svc.disabled_reason ret_fields['disabled_reason'] = svc.disabled_reason
if svc.binary == "cinder-volume": if svc.binary == "cinder-volume":
@ -153,8 +158,7 @@ class ServiceController(wsgi.Controller):
try: try:
host = body['host'] host = body['host']
except (TypeError, KeyError): except (TypeError, KeyError):
msg = _("Missing required element 'host' in request body.") raise exception.MissingRequired(element='host')
raise webob.exc.HTTPBadRequest(explanation=msg)
ret_val['disabled'] = disabled ret_val['disabled'] = disabled
if id == "disable-log-reason" and ext_loaded: if id == "disable-log-reason" and ext_loaded:

View File

@ -54,6 +54,7 @@ REST_API_VERSION_HISTORY = """
* 3.5 - Add pagination support to messages API. * 3.5 - Add pagination support to messages API.
* 3.6 - Allows to set empty description and empty name for consistency * 3.6 - Allows to set empty description and empty name for consistency
group in consisgroup-update operation. group in consisgroup-update operation.
* 3.7 - Add cluster API and cluster_name field to service list API
""" """
@ -62,7 +63,7 @@ REST_API_VERSION_HISTORY = """
# minimum version of the API supported. # minimum version of the API supported.
# Explicitly using /v1 or /v2 enpoints will still work # Explicitly using /v1 or /v2 enpoints will still work
_MIN_API_VERSION = "3.0" _MIN_API_VERSION = "3.0"
_MAX_API_VERSION = "3.6" _MAX_API_VERSION = "3.7"
_LEGACY_API_VERSION1 = "1.0" _LEGACY_API_VERSION1 = "1.0"
_LEGACY_API_VERSION2 = "2.0" _LEGACY_API_VERSION2 = "2.0"

View File

@ -69,3 +69,80 @@ user documentation.
--- ---
Allowed to set empty description and empty name for consistency Allowed to set empty description and empty name for consistency
group in consisgroup-update operation. group in consisgroup-update operation.
3.7
---
Added ``cluster_name`` field to service list/detail.
Added /clusters endpoint to list/show/update clusters.
Show endpoint requires the cluster name and optionally the binary as a URL
paramter (default is "cinder-volume"). Returns:
.. code-block:: json
"cluster": {
"created_at": ...,
"disabled_reason": null,
"last_heartbeat": ...,
"name": "cluster_name",
"num_down_hosts": 4,
"num_hosts": 2,
"state": "up",
"status": "enabled",
"updated_at": ...
}
Update endpoint allows enabling and disabling a cluster in a similar way to
service's update endpoint, but in the body we must specify the name and
optionally the binary ("cinder-volume" is the default) and the disabled
reason. Returns:
.. code-block:: json
"cluster": {
"name": "cluster_name",
"state": "up",
"status": "enabled"
"disabled_reason": null
}
Index and detail accept filtering by `name`, `binary`, `disabled`,
`num_hosts` , `num_down_hosts`, and up/down status (`is_up`) as URL
parameters.
Index endpoint returns:
.. code-block:: json
"clusters": [
{
"name": "cluster_name",
"state": "up",
"status": "enabled"
},
{
...
}
]
Detail endpoint returns:
.. code-block:: json
"clusters": [
{
"created_at": ...,
"disabled_reason": null,
"last_heartbeat": ...,
"name": "cluster_name",
"num_down_hosts": 4,
"num_hosts": 2,
"state": "up",
"status": "enabled",
"updated_at": ...
},
{
...
}
]

View File

@ -33,6 +33,7 @@ from cinder.api.openstack import versioned_method
from cinder import exception from cinder import exception
from cinder import i18n from cinder import i18n
from cinder.i18n import _, _LE, _LI from cinder.i18n import _, _LE, _LI
from cinder import policy
from cinder import utils from cinder import utils
from cinder.wsgi import common as wsgi from cinder.wsgi import common as wsgi
@ -1295,6 +1296,23 @@ class Controller(object):
except exception.InvalidInput as error: except exception.InvalidInput as error:
raise webob.exc.HTTPBadRequest(explanation=error.msg) raise webob.exc.HTTPBadRequest(explanation=error.msg)
@staticmethod
def get_policy_checker(prefix):
@staticmethod
def policy_checker(req, action, resource=None):
ctxt = req.environ['cinder.context']
target = {
'project_id': ctxt.project_id,
'user_id': ctxt.user_id,
}
if resource:
target.update(resource)
_action = '%s:%s' % (prefix, action)
policy.enforce(ctxt, _action, target)
return ctxt
return policy_checker
class Fault(webob.exc.HTTPException): class Fault(webob.exc.HTTPException):
"""Wrap webob.exc.HTTPException to provide API friendly response.""" """Wrap webob.exc.HTTPException to provide API friendly response."""

132
cinder/api/v3/clusters.py Normal file
View File

@ -0,0 +1,132 @@
# Copyright (c) 2016 Red Hat Inc.
# 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 cinder.api.openstack import wsgi
from cinder.api.v3.views import clusters as clusters_view
from cinder import exception
from cinder.i18n import _
from cinder import objects
from cinder import utils
CLUSTER_MICRO_VERSION = '3.7'
class ClusterController(wsgi.Controller):
allowed_list_keys = {'name', 'binary', 'is_up', 'disabled', 'num_hosts',
'num_down_hosts', 'binary'}
policy_checker = wsgi.Controller.get_policy_checker('clusters')
@wsgi.Controller.api_version(CLUSTER_MICRO_VERSION)
def show(self, req, id, binary='cinder-volume'):
"""Return data for a given cluster name with optional binary."""
# Let the wsgi middleware convert NotAuthorized exceptions
context = self.policy_checker(req, 'get')
# Let the wsgi middleware convert NotFound exceptions
cluster = objects.Cluster.get_by_id(context, None, binary=binary,
name=id, services_summary=True)
return clusters_view.ViewBuilder.detail(cluster)
@wsgi.Controller.api_version(CLUSTER_MICRO_VERSION)
def index(self, req):
"""Return a non detailed list of all existing clusters.
Filter by is_up, disabled, num_hosts, and num_down_hosts.
"""
return self._get_clusters(req, detail=False)
@wsgi.Controller.api_version(CLUSTER_MICRO_VERSION)
def detail(self, req):
"""Return a detailed list of all existing clusters.
Filter by is_up, disabled, num_hosts, and num_down_hosts.
"""
return self._get_clusters(req, detail=True)
def _get_clusters(self, req, detail):
# Let the wsgi middleware convert NotAuthorized exceptions
context = self.policy_checker(req, 'get_all')
filters = dict(req.GET)
allowed = self.allowed_list_keys
# Check filters are valid
if not allowed.issuperset(filters):
invalid_keys = set(filters).difference(allowed)
msg = _('Invalid filter keys: %s') % ', '.join(invalid_keys)
raise exception.InvalidInput(reason=msg)
# Check boolean values
for bool_key in ('disabled', 'is_up'):
if bool_key in filters:
filters[bool_key] = utils.get_bool_param(bool_key, req.GET)
# For detailed view we need the services summary information
filters['services_summary'] = detail
clusters = objects.ClusterList.get_all(context, **filters)
return clusters_view.ViewBuilder.list(clusters, detail)
@wsgi.Controller.api_version(CLUSTER_MICRO_VERSION)
def update(self, req, id, body):
"""Enable/Disable scheduling for a cluster."""
# NOTE(geguileo): This method tries to be consistent with services
# update endpoint API.
# Let the wsgi middleware convert NotAuthorized exceptions
context = self.policy_checker(req, 'update')
if id not in ('enable', 'disable'):
raise exception.NotFound(message=_("Unknown action"))
disabled = id != 'enable'
disabled_reason = self._get_disabled_reason(body) if disabled else None
if not disabled and disabled_reason:
msg = _("Unexpected 'disabled_reason' found on enable request.")
raise exception.InvalidInput(reason=msg)
name = body.get('name')
if not name:
raise exception.MissingRequired(element='name')
binary = body.get('binary', 'cinder-volume')
# Let wsgi handle NotFound exception
cluster = objects.Cluster.get_by_id(context, None, binary=binary,
name=name)
cluster.disabled = disabled
cluster.disabled_reason = disabled_reason
cluster.save()
# We return summary data plus the disabled reason
ret_val = clusters_view.ViewBuilder.summary(cluster)
ret_val['cluster']['disabled_reason'] = disabled_reason
return ret_val
def _get_disabled_reason(self, body):
reason = body.get('disabled_reason')
if reason:
# Let wsgi handle InvalidInput exception
reason = reason.strip()
utils.check_string_length(reason, 'Disabled reason', min_length=1,
max_length=255)
return reason
def create_resource():
return wsgi.Resource(ClusterController())

View File

@ -26,6 +26,7 @@ from cinder.api.v2 import snapshot_metadata
from cinder.api.v2 import snapshots from cinder.api.v2 import snapshots
from cinder.api.v2 import types from cinder.api.v2 import types
from cinder.api.v2 import volume_metadata from cinder.api.v2 import volume_metadata
from cinder.api.v3 import clusters
from cinder.api.v3 import consistencygroups from cinder.api.v3 import consistencygroups
from cinder.api.v3 import messages from cinder.api.v3 import messages
from cinder.api.v3 import volumes from cinder.api.v3 import volumes
@ -55,6 +56,11 @@ class APIRouter(cinder.api.openstack.APIRouter):
controller=self.resources['messages'], controller=self.resources['messages'],
collection={'detail': 'GET'}) collection={'detail': 'GET'})
self.resources['clusters'] = clusters.create_resource()
mapper.resource('cluster', 'clusters',
controller=self.resources['clusters'],
collection={'detail': 'GET'})
self.resources['types'] = types.create_resource() self.resources['types'] = types.create_resource()
mapper.resource("type", "types", mapper.resource("type", "types",
controller=self.resources['types'], controller=self.resources['types'],

View File

@ -0,0 +1,63 @@
# Copyright (c) 2016 Red Hat Inc.
# 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_utils import timeutils
class ViewBuilder(object):
"""Map Cluster into dicts for API responses."""
_collection_name = 'clusters'
@staticmethod
def _normalize(date):
if date:
return timeutils.normalize_time(date)
return ''
@classmethod
def detail(cls, cluster, flat=False):
"""Detailed view of a cluster."""
result = cls.summary(cluster, flat=True)
result.update(
num_hosts=cluster.num_hosts,
num_down_hosts=cluster.num_down_hosts,
last_heartbeat=cls._normalize(cluster.last_heartbeat),
created_at=cls._normalize(cluster.created_at),
updated_at=cls._normalize(cluster.updated_at),
disabled_reason=cluster.disabled_reason
)
if flat:
return result
return {'cluster': result}
@staticmethod
def summary(cluster, flat=False):
"""Generic, non-detailed view of a cluster."""
result = {
'name': cluster.name,
'binary': cluster.binary,
'state': 'up' if cluster.is_up() else 'down',
'status': 'disabled' if cluster.disabled else 'enabled',
}
if flat:
return result
return {'cluster': result}
@classmethod
def list(cls, clusters, detail=False):
func = cls.detail if detail else cls.summary
return {'clusters': [func(n, flat=True) for n in clusters]}

View File

@ -265,6 +265,10 @@ class InvalidGlobalAPIVersion(Invalid):
"is %(min_ver)s and maximum is %(max_ver)s.") "is %(min_ver)s and maximum is %(max_ver)s.")
class MissingRequired(Invalid):
message = _("Missing required element '%(element)s' in request body.")
class APIException(CinderException): class APIException(CinderException):
message = _("Error while requesting %(service)s API.") message = _("Error while requesting %(service)s API.")

View File

@ -22,6 +22,7 @@ import webob.exc
from cinder.api.contrib import services from cinder.api.contrib import services
from cinder.api import extensions from cinder.api import extensions
from cinder.api.openstack import api_version_request as api_version
from cinder import context from cinder import context
from cinder import exception from cinder import exception
from cinder import test from cinder import test
@ -32,6 +33,7 @@ from cinder.tests.unit import fake_constants as fake
fake_services_list = [ fake_services_list = [
{'binary': 'cinder-scheduler', {'binary': 'cinder-scheduler',
'host': 'host1', 'host': 'host1',
'cluster_name': None,
'availability_zone': 'cinder', 'availability_zone': 'cinder',
'id': 1, 'id': 1,
'disabled': True, 'disabled': True,
@ -41,6 +43,7 @@ fake_services_list = [
'modified_at': ''}, 'modified_at': ''},
{'binary': 'cinder-volume', {'binary': 'cinder-volume',
'host': 'host1', 'host': 'host1',
'cluster_name': None,
'availability_zone': 'cinder', 'availability_zone': 'cinder',
'id': 2, 'id': 2,
'disabled': True, 'disabled': True,
@ -50,6 +53,7 @@ fake_services_list = [
'modified_at': ''}, 'modified_at': ''},
{'binary': 'cinder-scheduler', {'binary': 'cinder-scheduler',
'host': 'host2', 'host': 'host2',
'cluster_name': 'cluster1',
'availability_zone': 'cinder', 'availability_zone': 'cinder',
'id': 3, 'id': 3,
'disabled': False, 'disabled': False,
@ -59,6 +63,7 @@ fake_services_list = [
'modified_at': ''}, 'modified_at': ''},
{'binary': 'cinder-volume', {'binary': 'cinder-volume',
'host': 'host2', 'host': 'host2',
'cluster_name': 'cluster1',
'availability_zone': 'cinder', 'availability_zone': 'cinder',
'id': 4, 'id': 4,
'disabled': True, 'disabled': True,
@ -68,6 +73,7 @@ fake_services_list = [
'modified_at': ''}, 'modified_at': ''},
{'binary': 'cinder-volume', {'binary': 'cinder-volume',
'host': 'host2', 'host': 'host2',
'cluster_name': 'cluster2',
'availability_zone': 'cinder', 'availability_zone': 'cinder',
'id': 5, 'id': 5,
'disabled': True, 'disabled': True,
@ -77,6 +83,7 @@ fake_services_list = [
'modified_at': datetime.datetime(2012, 10, 29, 13, 42, 5)}, 'modified_at': datetime.datetime(2012, 10, 29, 13, 42, 5)},
{'binary': 'cinder-volume', {'binary': 'cinder-volume',
'host': 'host2', 'host': 'host2',
'cluster_name': 'cluster2',
'availability_zone': 'cinder', 'availability_zone': 'cinder',
'id': 6, 'id': 6,
'disabled': False, 'disabled': False,
@ -86,8 +93,9 @@ fake_services_list = [
'modified_at': datetime.datetime(2012, 9, 18, 8, 1, 38)}, 'modified_at': datetime.datetime(2012, 9, 18, 8, 1, 38)},
{'binary': 'cinder-scheduler', {'binary': 'cinder-scheduler',
'host': 'host2', 'host': 'host2',
'cluster_name': None,
'availability_zone': 'cinder', 'availability_zone': 'cinder',
'id': 6, 'id': 7,
'disabled': False, 'disabled': False,
'updated_at': None, 'updated_at': None,
'created_at': datetime.datetime(2012, 9, 18, 2, 46, 28), 'created_at': datetime.datetime(2012, 9, 18, 2, 46, 28),
@ -98,36 +106,45 @@ fake_services_list = [
class FakeRequest(object): class FakeRequest(object):
environ = {"cinder.context": context.get_admin_context()} environ = {"cinder.context": context.get_admin_context()}
GET = {}
def __init__(self, version='3.0', **kwargs):
self.GET = kwargs
self.headers = {'OpenStack-API-Version': 'volume ' + version}
self.api_version_request = api_version.APIVersionRequest(version)
# NOTE(uni): deprecating service request key, binary takes precedence # NOTE(uni): deprecating service request key, binary takes precedence
# Still keeping service key here for API compatibility sake. # Still keeping service key here for API compatibility sake.
class FakeRequestWithService(object): class FakeRequestWithService(FakeRequest):
environ = {"cinder.context": context.get_admin_context()} def __init__(self, **kwargs):
GET = {"service": "cinder-volume"} kwargs.setdefault('service', 'cinder-volume')
super(FakeRequestWithService, self).__init__(**kwargs)
class FakeRequestWithBinary(object): class FakeRequestWithBinary(FakeRequest):
environ = {"cinder.context": context.get_admin_context()} def __init__(self, **kwargs):
GET = {"binary": "cinder-volume"} kwargs.setdefault('binary', 'cinder-volume')
super(FakeRequestWithBinary, self).__init__(**kwargs)
class FakeRequestWithHost(object): class FakeRequestWithHost(FakeRequest):
environ = {"cinder.context": context.get_admin_context()} def __init__(self, **kwargs):
GET = {"host": "host1"} kwargs.setdefault('host', 'host1')
super(FakeRequestWithHost, self).__init__(**kwargs)
# NOTE(uni): deprecating service request key, binary takes precedence # NOTE(uni): deprecating service request key, binary takes precedence
# Still keeping service key here for API compatibility sake. # Still keeping service key here for API compatibility sake.
class FakeRequestWithHostService(object): class FakeRequestWithHostService(FakeRequestWithService):
environ = {"cinder.context": context.get_admin_context()} def __init__(self, **kwargs):
GET = {"host": "host1", "service": "cinder-volume"} kwargs.setdefault('host', 'host1')
super(FakeRequestWithHostService, self).__init__(**kwargs)
class FakeRequestWithHostBinary(object): class FakeRequestWithHostBinary(FakeRequestWithBinary):
environ = {"cinder.context": context.get_admin_context()} def __init__(self, **kwargs):
GET = {"host": "host1", "binary": "cinder-volume"} kwargs.setdefault('host', 'host1')
super(FakeRequestWithHostBinary, self).__init__(**kwargs)
def fake_service_get_all(context, **filters): def fake_service_get_all(context, **filters):
@ -236,6 +253,59 @@ class ServicesTest(test.TestCase):
]} ]}
self.assertEqual(response, res_dict) self.assertEqual(response, res_dict)
def test_services_list_with_cluster_name(self):
req = FakeRequest(version='3.7')
res_dict = self.controller.index(req)
response = {'services': [{'binary': 'cinder-scheduler',
'cluster': None,
'host': 'host1', 'zone': 'cinder',
'status': 'disabled', 'state': 'up',
'updated_at': datetime.datetime(
2012, 10, 29, 13, 42, 2)},
{'binary': 'cinder-volume',
'cluster': None,
'host': 'host1', 'zone': 'cinder',
'status': 'disabled', 'state': 'up',
'updated_at': datetime.datetime(
2012, 10, 29, 13, 42, 5)},
{'binary': 'cinder-scheduler',
'cluster': 'cluster1',
'host': 'host2',
'zone': 'cinder',
'status': 'enabled', 'state': 'down',
'updated_at': datetime.datetime(
2012, 9, 19, 6, 55, 34)},
{'binary': 'cinder-volume',
'cluster': 'cluster1',
'host': 'host2',
'zone': 'cinder',
'status': 'disabled', 'state': 'down',
'updated_at': datetime.datetime(
2012, 9, 18, 8, 3, 38)},
{'binary': 'cinder-volume',
'cluster': 'cluster2',
'host': 'host2',
'zone': 'cinder',
'status': 'disabled', 'state': 'down',
'updated_at': datetime.datetime(
2012, 10, 29, 13, 42, 5)},
{'binary': 'cinder-volume',
'cluster': 'cluster2',
'host': 'host2',
'zone': 'cinder',
'status': 'enabled', 'state': 'down',
'updated_at': datetime.datetime(
2012, 9, 18, 8, 3, 38)},
{'binary': 'cinder-scheduler',
'cluster': None,
'host': 'host2',
'zone': 'cinder',
'status': 'enabled', 'state': 'down',
'updated_at': None},
]}
self.assertEqual(response, res_dict)
def test_services_detail(self): def test_services_detail(self):
self.ext_mgr.extensions['os-extended-services'] = True self.ext_mgr.extensions['os-extended-services'] = True
self.controller = services.ServiceController(self.ext_mgr) self.controller = services.ServiceController(self.ext_mgr)

View File

@ -0,0 +1,251 @@
# Copyright (c) 2016 Red Hat, Inc.
# 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 datetime
import ddt
from iso8601 import iso8601
import mock
from cinder.api import extensions
from cinder.api.openstack import api_version_request as api_version
from cinder.api.v3 import clusters
from cinder import context
from cinder import exception
from cinder import test
from cinder.tests.unit import fake_cluster
CLUSTERS = [
fake_cluster.fake_db_cluster(
id=1,
last_heartbeat=datetime.datetime(2016, 6, 1, 2, 46, 28),
updated_at=datetime.datetime(2016, 6, 1, 2, 46, 28),
created_at=datetime.datetime(2016, 6, 1, 2, 46, 28)),
fake_cluster.fake_db_cluster(
id=2, name='cluster2', num_hosts=2, num_down_hosts=1, disabled=True,
updated_at=datetime.datetime(2016, 6, 1, 1, 46, 28),
created_at=datetime.datetime(2016, 6, 1, 1, 46, 28))
]
CLUSTERS_ORM = [fake_cluster.fake_cluster_orm(**kwargs) for kwargs in CLUSTERS]
EXPECTED = [{'created_at': datetime.datetime(2016, 6, 1, 2, 46, 28),
'disabled_reason': None,
'last_heartbeat': datetime.datetime(2016, 6, 1, 2, 46, 28),
'name': 'cluster_name',
'binary': 'cinder-volume',
'num_down_hosts': 0,
'num_hosts': 0,
'state': 'up',
'status': 'enabled',
'updated_at': datetime.datetime(2016, 6, 1, 2, 46, 28)},
{'created_at': datetime.datetime(2016, 6, 1, 1, 46, 28),
'updated_at': datetime.datetime(2016, 6, 1, 1, 46, 28),
'disabled_reason': None,
'last_heartbeat': '',
'name': 'cluster2',
'binary': 'cinder-volume',
'num_down_hosts': 1,
'num_hosts': 2,
'state': 'down',
'status': 'disabled',
'updated_at': datetime.datetime(2016, 6, 1, 1, 46, 28)}]
class FakeRequest(object):
def __init__(self, is_admin=True, version='3.7', **kwargs):
self.GET = kwargs
self.headers = {'OpenStack-API-Version': 'volume ' + version}
self.api_version_request = api_version.APIVersionRequest(version)
self.environ = {
'cinder.context': context.RequestContext(user_id=None,
project_id=None,
is_admin=is_admin,
read_deleted='no',
overwrite=False)
}
def fake_utcnow(with_timezone=False):
tzinfo = iso8601.Utc() if with_timezone else None
return datetime.datetime(2016, 6, 1, 2, 46, 30, tzinfo=tzinfo)
@ddt.ddt
@mock.patch('oslo_utils.timeutils.utcnow', fake_utcnow)
class ClustersTestCase(test.TestCase):
"""Test Case for Clusters."""
LIST_FILTERS = ({}, {'is_up': True}, {'disabled': False}, {'num_hosts': 2},
{'num_down_hosts': 1}, {'binary': 'cinder-volume'},
{'is_up': True, 'disabled': False, 'num_hosts': 2,
'num_down_hosts': 1, 'binary': 'cinder-volume'})
def setUp(self):
super(ClustersTestCase, self).setUp()
self.context = context.get_admin_context()
self.ext_mgr = extensions.ExtensionManager()
self.ext_mgr.extensions = {}
self.controller = clusters.ClusterController(self.ext_mgr)
@mock.patch('cinder.db.cluster_get_all', return_value=CLUSTERS_ORM)
def _test_list(self, get_all_mock, detailed, filters, expected=None):
req = FakeRequest(**filters)
method = getattr(self.controller, 'detail' if detailed else 'index')
clusters = method(req)
filters = filters.copy()
filters.setdefault('is_up', None)
filters.setdefault('read_deleted', 'no')
self.assertEqual(expected, clusters)
get_all_mock.assert_called_once_with(
req.environ['cinder.context'],
get_services=False,
services_summary=detailed,
**filters)
@ddt.data(*LIST_FILTERS)
def test_index_detail(self, filters):
"""Verify that we get all clusters with detailed data."""
expected = {'clusters': EXPECTED}
self._test_list(detailed=True, filters=filters, expected=expected)
@ddt.data(*LIST_FILTERS)
def test_index_summary(self, filters):
"""Verify that we get all clusters with summary data."""
expected = {'clusters': [{'name': 'cluster_name',
'binary': 'cinder-volume',
'state': 'up',
'status': 'enabled'},
{'name': 'cluster2',
'binary': 'cinder-volume',
'state': 'down',
'status': 'disabled'}]}
self._test_list(detailed=False, filters=filters, expected=expected)
@ddt.data(True, False)
def test_index_unauthorized(self, detailed):
"""Verify that unauthorized user can't list clusters."""
self.assertRaises(exception.PolicyNotAuthorized,
self._test_list, detailed=detailed,
filters={'is_admin': False})
@ddt.data(True, False)
def test_index_wrong_version(self, detailed):
"""Verify that unauthorized user can't list clusters."""
self.assertRaises(exception.VersionNotFoundForAPIMethod,
self._test_list, detailed=detailed,
filters={'version': '3.5'})
@mock.patch('cinder.db.sqlalchemy.api.cluster_get',
return_value=CLUSTERS_ORM[0])
def test_show(self, get_mock):
req = FakeRequest()
expected = {'cluster': EXPECTED[0]}
cluster = self.controller.show(req, mock.sentinel.name,
mock.sentinel.binary)
self.assertEqual(expected, cluster)
get_mock.assert_called_once_with(
req.environ['cinder.context'],
None,
services_summary=True,
name=mock.sentinel.name,
binary=mock.sentinel.binary)
def test_show_unauthorized(self):
req = FakeRequest(is_admin=False)
self.assertRaises(exception.PolicyNotAuthorized,
self.controller.show, req, 'name')
def test_show_wrong_version(self):
req = FakeRequest(version='3.5')
self.assertRaises(exception.VersionNotFoundForAPIMethod,
self.controller.show, req, 'name')
@mock.patch('cinder.db.sqlalchemy.api.cluster_update')
@mock.patch('cinder.db.sqlalchemy.api.cluster_get',
return_value=CLUSTERS_ORM[1])
def test_update_enable(self, get_mock, update_mock):
req = FakeRequest()
expected = {'cluster': {'name': u'cluster2',
'binary': 'cinder-volume',
'state': 'down',
'status': 'enabled',
'disabled_reason': None}}
res = self.controller.update(req, 'enable',
{'name': mock.sentinel.name,
'binary': mock.sentinel.binary})
self.assertEqual(expected, res)
ctxt = req.environ['cinder.context']
get_mock.assert_called_once_with(ctxt,
None, binary=mock.sentinel.binary,
name=mock.sentinel.name)
update_mock.assert_called_once_with(ctxt, get_mock.return_value.id,
{'disabled': False,
'disabled_reason': None})
@mock.patch('cinder.db.sqlalchemy.api.cluster_update')
@mock.patch('cinder.db.sqlalchemy.api.cluster_get',
return_value=CLUSTERS_ORM[0])
def test_update_disable(self, get_mock, update_mock):
req = FakeRequest()
disabled_reason = 'For testing'
expected = {'cluster': {'name': u'cluster_name',
'state': 'up',
'binary': 'cinder-volume',
'status': 'disabled',
'disabled_reason': disabled_reason}}
res = self.controller.update(req, 'disable',
{'name': mock.sentinel.name,
'binary': mock.sentinel.binary,
'disabled_reason': disabled_reason})
self.assertEqual(expected, res)
ctxt = req.environ['cinder.context']
get_mock.assert_called_once_with(ctxt,
None, binary=mock.sentinel.binary,
name=mock.sentinel.name)
update_mock.assert_called_once_with(
ctxt, get_mock.return_value.id,
{'disabled': True, 'disabled_reason': disabled_reason})
def test_update_wrong_action(self):
req = FakeRequest()
self.assertRaises(exception.NotFound, self.controller.update, req,
'action', {})
@ddt.data('enable', 'disable')
def test_update_missing_name(self, action):
req = FakeRequest()
self.assertRaises(exception.MissingRequired, self.controller.update,
req, action, {'binary': mock.sentinel.binary})
def test_update_wrong_disabled_reason(self):
req = FakeRequest()
self.assertRaises(exception.InvalidInput, self.controller.update, req,
'disable', {'name': mock.sentinel.name,
'disabled_reason': ' '})
@ddt.data('enable', 'disable')
def test_update_unauthorized(self, action):
req = FakeRequest(is_admin=False)
self.assertRaises(exception.PolicyNotAuthorized,
self.controller.update, req, action, {})
@ddt.data('enable', 'disable')
def test_update_wrong_version(self, action):
req = FakeRequest(version='3.5')
self.assertRaises(exception.VersionNotFoundForAPIMethod,
self.controller.update, req, action, {})

View File

@ -116,5 +116,9 @@
"message:delete": "rule:admin_or_owner", "message:delete": "rule:admin_or_owner",
"message:get": "rule:admin_or_owner", "message:get": "rule:admin_or_owner",
"message:get_all": "rule:admin_or_owner" "message:get_all": "rule:admin_or_owner",
"clusters:get": "rule:admin_api",
"clusters:get_all": "rule:admin_api",
"clusters:update": "rule:admin_api"
} }

View File

@ -111,5 +111,9 @@
"scheduler_extension:scheduler_stats:get_pools" : "rule:admin_api", "scheduler_extension:scheduler_stats:get_pools" : "rule:admin_api",
"message:delete": "rule:admin_or_owner", "message:delete": "rule:admin_or_owner",
"message:get": "rule:admin_or_owner", "message:get": "rule:admin_or_owner",
"message:get_all": "rule:admin_or_owner" "message:get_all": "rule:admin_or_owner",
"clusters:get": "rule:admin_api",
"clusters:get_all": "rule:admin_api",
"clusters:update": "rule:admin_api",
} }

View File

@ -14,3 +14,7 @@ features:
listings." listings."
- "HA A-A: Added cluster subcommand in manage command to list, remove, and - "HA A-A: Added cluster subcommand in manage command to list, remove, and
rename clusters." rename clusters."
- "HA A-A: Added clusters API endpoints for cluster related operations (index,
detail, show, enable/disable). Index and detail accept filtering by
`name`, `binary`, `disabled`, `num_hosts`, `num_down_hosts`, and up/down
status (`is_up`) as URL parameters. Also added their respective policies."