Add checks for aggregate affinity_zone
* Do not allow empty affinity zone * A node can not be in different affinity zone Partially Implements: bp server-group-api-extension Change-Id: I0b96ddd16bd51975d1fadb93760854bb8c486be7
This commit is contained in:
parent
de9e8777d1
commit
b95ab6b9f9
@ -98,6 +98,21 @@ class AggregateCollection(base.APIBase):
|
|||||||
class AggregateNodeController(rest.RestController):
|
class AggregateNodeController(rest.RestController):
|
||||||
"""REST controller for aggregate nodes."""
|
"""REST controller for aggregate nodes."""
|
||||||
|
|
||||||
|
def _check_aggregates_conflict(self, node, node_aggregates, key, value):
|
||||||
|
"""Check aggregates conflict with the given key"""
|
||||||
|
aggregates = objects.AggregateList.get_by_metadata_key(
|
||||||
|
pecan.request.context, key)
|
||||||
|
conflicts = [
|
||||||
|
agg.metadata[key] for agg in aggregates
|
||||||
|
if agg.uuid in node_aggregates and
|
||||||
|
agg.metadata[key] != value]
|
||||||
|
if conflicts:
|
||||||
|
msg = _("Node %(node)s is already in %(key)s(s) "
|
||||||
|
"%(conflicts)s") % {"node": node, "key": key,
|
||||||
|
"conflicts": conflicts}
|
||||||
|
raise wsme.exc.ClientSideError(
|
||||||
|
msg, status_code=http_client.BAD_REQUEST)
|
||||||
|
|
||||||
@policy.authorize_wsgi("mogan:aggregate_node", "get_all")
|
@policy.authorize_wsgi("mogan:aggregate_node", "get_all")
|
||||||
@expose.expose(wtypes.text, types.uuid)
|
@expose.expose(wtypes.text, types.uuid)
|
||||||
def get_all(self, aggregate_uuid):
|
def get_all(self, aggregate_uuid):
|
||||||
@ -118,24 +133,20 @@ class AggregateNodeController(rest.RestController):
|
|||||||
validation.check_schema(node, agg_schema.add_aggregate_node)
|
validation.check_schema(node, agg_schema.add_aggregate_node)
|
||||||
|
|
||||||
node = node['node']
|
node = node['node']
|
||||||
# check whether the node is already in another az
|
|
||||||
db_aggregate = objects.Aggregate.get(pecan.request.context,
|
db_aggregate = objects.Aggregate.get(pecan.request.context,
|
||||||
aggregate_uuid)
|
aggregate_uuid)
|
||||||
if 'availability_zone' in db_aggregate['metadata']:
|
node_aggregates = pecan.request.engine_api.list_node_aggregates(
|
||||||
node_aggs = pecan.request.engine_api.list_node_aggregates(
|
|
||||||
pecan.request.context, node)
|
pecan.request.context, node)
|
||||||
aggregates = objects.AggregateList.get_by_metadata_key(
|
# check whether the node is already in another az
|
||||||
pecan.request.context, 'availability_zone')
|
if 'availability_zone' in db_aggregate['metadata']:
|
||||||
agg_az = db_aggregate.metadata['availability_zone']
|
self._check_aggregates_conflict(
|
||||||
conflict_azs = [
|
node, node_aggregates['aggregates'], 'availability_zone',
|
||||||
agg.metadata['availability_zone'] for agg in aggregates
|
db_aggregate.metadata['availability_zone'])
|
||||||
if agg.uuid in node_aggs['aggregates'] and
|
# check whether the node is already in another affinity zone
|
||||||
agg.metadata['availability_zone'] != agg_az]
|
if 'affinity_zone' in db_aggregate['metadata']:
|
||||||
if conflict_azs:
|
self._check_aggregates_conflict(
|
||||||
msg = _("Node %(node)s is already in availability zone(s) "
|
node, node_aggregates['aggregates'], 'affinity_zone',
|
||||||
"%(az)s") % {"node": node, "az": conflict_azs}
|
db_aggregate.metadata['affinity_zone'])
|
||||||
raise wsme.exc.ClientSideError(
|
|
||||||
msg, status_code=http_client.BAD_REQUEST)
|
|
||||||
|
|
||||||
pecan.request.engine_api.add_aggregate_node(
|
pecan.request.engine_api.add_aggregate_node(
|
||||||
pecan.request.context, aggregate_uuid, node)
|
pecan.request.context, aggregate_uuid, node)
|
||||||
@ -157,6 +168,52 @@ class AggregateController(rest.RestController):
|
|||||||
|
|
||||||
nodes = AggregateNodeController()
|
nodes = AggregateNodeController()
|
||||||
|
|
||||||
|
def _check_metadata_conflicts(self, aggregate_uuid, key, value):
|
||||||
|
"""Check if metadata conflict with the given key"""
|
||||||
|
|
||||||
|
nodes = pecan.request.engine_api.list_aggregate_nodes(
|
||||||
|
pecan.request.context, aggregate_uuid)
|
||||||
|
aggregates = objects.AggregateList.get_by_metadata_key(
|
||||||
|
pecan.request.context, key)
|
||||||
|
filtered_aggs = []
|
||||||
|
for agg in aggregates:
|
||||||
|
agg_nodes = \
|
||||||
|
pecan.request.engine_api.list_aggregate_nodes(
|
||||||
|
pecan.request.context, agg.uuid)
|
||||||
|
for node in agg_nodes['nodes']:
|
||||||
|
if node in nodes['nodes']:
|
||||||
|
filtered_aggs.append(agg)
|
||||||
|
break
|
||||||
|
|
||||||
|
conflicts = [agg.metadata[key] for agg in filtered_aggs
|
||||||
|
if agg.metadata[key] != value and
|
||||||
|
agg.uuid != aggregate_uuid]
|
||||||
|
if conflicts:
|
||||||
|
msg = _("One or more nodes already in different "
|
||||||
|
"%(key)s(s) %(conflicts)s") % {"key": key,
|
||||||
|
"conflicts": conflicts}
|
||||||
|
raise wsme.exc.ClientSideError(
|
||||||
|
msg, status_code=http_client.BAD_REQUEST)
|
||||||
|
|
||||||
|
def _is_safe_to_update_metadata(self, patch, aggregate_uuid):
|
||||||
|
"""Check if it's safe to update aggregate metadata"""
|
||||||
|
|
||||||
|
keys = ['availability_zone', 'affinity_zone']
|
||||||
|
# Check if this tries to change *keys* to empty.
|
||||||
|
for patch_dict in patch:
|
||||||
|
for key in keys:
|
||||||
|
if patch_dict['path'] == '/metadata/' + key \
|
||||||
|
and patch_dict['op'] != 'remove':
|
||||||
|
if not patch_dict['value']:
|
||||||
|
msg = _("Aggregate %(uuid)s does not support empty "
|
||||||
|
"named %(key)s") % {"uuid": aggregate_uuid,
|
||||||
|
"key": key}
|
||||||
|
raise wsme.exc.ClientSideError(
|
||||||
|
msg, status_code=http_client.BAD_REQUEST)
|
||||||
|
else:
|
||||||
|
self._check_metadata_conflicts(
|
||||||
|
aggregate_uuid, key, patch_dict['value'])
|
||||||
|
|
||||||
@policy.authorize_wsgi("mogan:aggregate", "get_all")
|
@policy.authorize_wsgi("mogan:aggregate", "get_all")
|
||||||
@expose.expose(AggregateCollection)
|
@expose.expose(AggregateCollection)
|
||||||
def get_all(self):
|
def get_all(self):
|
||||||
@ -186,10 +243,12 @@ class AggregateController(rest.RestController):
|
|||||||
"""
|
"""
|
||||||
validation.check_schema(aggregate, agg_schema.create_aggregate)
|
validation.check_schema(aggregate, agg_schema.create_aggregate)
|
||||||
metadata = aggregate.get('metadata')
|
metadata = aggregate.get('metadata')
|
||||||
if metadata and 'availability_zone' in metadata:
|
if metadata:
|
||||||
if not metadata['availability_zone']:
|
for key in ['availability_zone', 'affinity_zone']:
|
||||||
msg = _("Aggregate %s does not support empty named "
|
if key in metadata and not metadata[key]:
|
||||||
"availability zone") % aggregate['name']
|
msg = _("Aggregate %(name)s does not support empty named "
|
||||||
|
"%(key)s") % {"name": aggregate['name'],
|
||||||
|
"key": key}
|
||||||
raise wsme.exc.ClientSideError(
|
raise wsme.exc.ClientSideError(
|
||||||
msg, status_code=http_client.BAD_REQUEST)
|
msg, status_code=http_client.BAD_REQUEST)
|
||||||
|
|
||||||
@ -220,13 +279,8 @@ class AggregateController(rest.RestController):
|
|||||||
except api_utils.JSONPATCH_EXCEPTIONS as e:
|
except api_utils.JSONPATCH_EXCEPTIONS as e:
|
||||||
raise exception.PatchError(patch=patch, reason=e)
|
raise exception.PatchError(patch=patch, reason=e)
|
||||||
|
|
||||||
# Check if this tries to change availability zone
|
# Check whether it is safe to update metadata
|
||||||
az_changed = False
|
self._is_safe_to_update_metadata(patch, db_aggregate['uuid'])
|
||||||
for patch_dict in patch:
|
|
||||||
if patch_dict['path'] == '/metadata/availability_zone' \
|
|
||||||
and patch_dict['op'] != 'remove':
|
|
||||||
az_changed = True
|
|
||||||
break
|
|
||||||
|
|
||||||
# Update only the fields that have changed
|
# Update only the fields that have changed
|
||||||
for field in objects.Aggregate.fields:
|
for field in objects.Aggregate.fields:
|
||||||
@ -240,41 +294,6 @@ class AggregateController(rest.RestController):
|
|||||||
if db_aggregate[field] != patch_val:
|
if db_aggregate[field] != patch_val:
|
||||||
db_aggregate[field] = patch_val
|
db_aggregate[field] = patch_val
|
||||||
|
|
||||||
if field == 'metadata' and az_changed:
|
|
||||||
if not patch_val['availability_zone']:
|
|
||||||
msg = _("Aggregate %s does not support empty named "
|
|
||||||
"availability zone") % aggregate.name
|
|
||||||
raise wsme.exc.ClientSideError(
|
|
||||||
msg, status_code=http_client.BAD_REQUEST)
|
|
||||||
|
|
||||||
# ensure it's safe to update availability_zone
|
|
||||||
aggregates = objects.AggregateList.get_by_metadata_key(
|
|
||||||
pecan.request.context, 'availability_zone')
|
|
||||||
nodes = pecan.request.engine_api.list_aggregate_nodes(
|
|
||||||
pecan.request.context, db_aggregate['uuid'])
|
|
||||||
filtered_aggs = []
|
|
||||||
for agg in aggregates:
|
|
||||||
agg_nodes = \
|
|
||||||
pecan.request.engine_api.list_aggregate_nodes(
|
|
||||||
pecan.request.context, agg.uuid)
|
|
||||||
for node in agg_nodes['nodes']:
|
|
||||||
if node in nodes['nodes']:
|
|
||||||
filtered_aggs.append(agg)
|
|
||||||
break
|
|
||||||
|
|
||||||
new_az = patch_val['availability_zone']
|
|
||||||
conflict_azs = [
|
|
||||||
agg.metadata['availability_zone']
|
|
||||||
for agg in filtered_aggs
|
|
||||||
if agg.metadata['availability_zone'] != new_az and
|
|
||||||
agg.uuid != db_aggregate['uuid']
|
|
||||||
]
|
|
||||||
if conflict_azs:
|
|
||||||
msg = _("One or more nodes already in different "
|
|
||||||
"availability zone(s) %s") % conflict_azs
|
|
||||||
raise wsme.exc.ClientSideError(
|
|
||||||
msg, status_code=http_client.BAD_REQUEST)
|
|
||||||
|
|
||||||
db_aggregate.save()
|
db_aggregate.save()
|
||||||
|
|
||||||
return Aggregate.convert_with_links(db_aggregate)
|
return Aggregate.convert_with_links(db_aggregate)
|
||||||
|
@ -59,6 +59,15 @@ class TestAggregate(v1_test.APITestV1):
|
|||||||
self.assertEqual('application/json', response.content_type)
|
self.assertEqual('application/json', response.content_type)
|
||||||
self.assertTrue(response.json['error_message'])
|
self.assertTrue(response.json['error_message'])
|
||||||
|
|
||||||
|
def test_aggregate_post_with_empty_affinity_zone(self):
|
||||||
|
body = {"name": "test_empty",
|
||||||
|
"metadata": {"affinity_zone": ""}}
|
||||||
|
response = self.post_json(
|
||||||
|
'/aggregates', body, headers=self.headers, expect_errors=True)
|
||||||
|
self.assertEqual(http_client.BAD_REQUEST, response.status_code)
|
||||||
|
self.assertEqual('application/json', response.content_type)
|
||||||
|
self.assertTrue(response.json['error_message'])
|
||||||
|
|
||||||
def test_aggregate_get_all(self):
|
def test_aggregate_get_all(self):
|
||||||
self._prepare_aggregates()
|
self._prepare_aggregates()
|
||||||
resp = self.get_json('/aggregates', headers=self.headers)
|
resp = self.get_json('/aggregates', headers=self.headers)
|
||||||
@ -108,3 +117,14 @@ class TestAggregate(v1_test.APITestV1):
|
|||||||
self.assertEqual(http_client.BAD_REQUEST, response.status_code)
|
self.assertEqual(http_client.BAD_REQUEST, response.status_code)
|
||||||
self.assertEqual('application/json', response.content_type)
|
self.assertEqual('application/json', response.content_type)
|
||||||
self.assertTrue(response.json['error_message'])
|
self.assertTrue(response.json['error_message'])
|
||||||
|
|
||||||
|
def test_aggregate_update_with_empty_affinity_zone(self):
|
||||||
|
self._prepare_aggregates()
|
||||||
|
response = self.patch_json('/aggregates/' + self.AGGREGATE_UUIDS[0],
|
||||||
|
[{'path': '/metadata/affinity_zone',
|
||||||
|
'value': '', 'op': 'add'}],
|
||||||
|
headers=self.headers,
|
||||||
|
expect_errors=True)
|
||||||
|
self.assertEqual(http_client.BAD_REQUEST, response.status_code)
|
||||||
|
self.assertEqual('application/json', response.content_type)
|
||||||
|
self.assertTrue(response.json['error_message'])
|
||||||
|
Loading…
x
Reference in New Issue
Block a user