Microversion 2.64 - Use new format policy in server group

Added support for microversion 2.64, which includes the
following changes:

- The ``--rule`` option is added to the
``nova server-group-create`` CLI that enables user to create
server group with specific policy rules.

- Remove ``metadata`` column in the output of
``nova server-group-create``, ``nova server-group-get``,
``nova server-group-list``.

- Remove ``policies`` column, , add ``policy`` and ``rules``
columns in the output of ``nova server-group-create``,
``nova server-group-get``, ``nova server-group-list``.

Depends-On: 3cd26f1e68b09ba7925e794ac8912566c239b6df
blueprint: complex-anti-affinity-policies

Change-Id: I903f4b5544806b9d3c8bac529448abbc9dd3cee9
This commit is contained in:
Yikun Jiang 2018-06-27 11:52:04 +08:00 committed by Matt Riedemann
parent 003ac57d9a
commit 6c398058a6
9 changed files with 243 additions and 15 deletions

View File

@ -2951,7 +2951,7 @@ nova server-group-create
.. code-block:: console .. code-block:: console
usage: nova server-group-create <name> <policy> usage: nova server-group-create [--rules <key=value>] <name> <policy>
Create a new server group with the specified details. Create a new server group with the specified details.
@ -2961,7 +2961,15 @@ Create a new server group with the specified details.
Server group name. Server group name.
``<policy>`` ``<policy>``
Policies for the server groups. Policy for the server groups.
``--rule``
Policy rules for the server groups. (Supported by API versions
'2.64' - '2.latest'. Currently, only the ``max_server_per_host`` rule
is supported for the ``anti-affinity`` policy. The ``max_server_per_host``
rule allows specifying how many members of the anti-affinity group can
reside on the same compute host. If not specified, only one member from
the same anti-affinity group can reside on a given host.
.. _nova_server-group-delete: .. _nova_server-group-delete:

View File

@ -25,4 +25,4 @@ API_MIN_VERSION = api_versions.APIVersion("2.1")
# when client supported the max version, and bumped sequentially, otherwise # when client supported the max version, and bumped sequentially, otherwise
# the client may break due to server side new version may include some # the client may break due to server side new version may include some
# backward incompatible change. # backward incompatible change.
API_MAX_VERSION = api_versions.APIVersion("2.63") API_MAX_VERSION = api_versions.APIVersion("2.64")

View File

@ -17,7 +17,9 @@ from novaclient.tests.functional.v2.legacy import test_server_groups
class TestServerGroupClientV213(test_server_groups.TestServerGroupClient): class TestServerGroupClientV213(test_server_groups.TestServerGroupClient):
"""Server groups v2.13 functional tests.""" """Server groups v2.13 functional tests."""
COMPUTE_API_VERSION = "2.latest" COMPUTE_API_VERSION = "2.13"
expected_metadata = True
expected_policy_rules = False
def test_create_server_group(self): def test_create_server_group(self):
sg_id = self._create_sg("affinity") sg_id = self._create_sg("affinity")
@ -29,6 +31,11 @@ class TestServerGroupClientV213(test_server_groups.TestServerGroupClient):
self._get_column_value_from_single_row_table( self._get_column_value_from_single_row_table(
sg, "Project Id") sg, "Project Id")
self.assertEqual(sg_id, result) self.assertEqual(sg_id, result)
self._get_column_value_from_single_row_table(sg, "Metadata")
self.assertIn(
'affinity',
self._get_column_value_from_single_row_table(sg, 'Policies'))
self.assertNotIn('Rules', sg)
def test_list_server_groups(self): def test_list_server_groups(self):
sg_id = self._create_sg("affinity") sg_id = self._create_sg("affinity")
@ -40,6 +47,22 @@ class TestServerGroupClientV213(test_server_groups.TestServerGroupClient):
self._get_column_value_from_single_row_table( self._get_column_value_from_single_row_table(
sg, "Project Id") sg, "Project Id")
self.assertEqual(sg_id, result) self.assertEqual(sg_id, result)
if self.expected_metadata:
self._get_column_value_from_single_row_table(sg, "Metadata")
else:
self.assertNotIn(sg, 'Metadata')
if self.expected_policy_rules:
self.assertEqual(
'affinity',
self._get_column_value_from_single_row_table(sg, "Policy"))
self.assertEqual(
'{}',
self._get_column_value_from_single_row_table(sg, "Rules"))
else:
self.assertIn(
'affinity',
self._get_column_value_from_single_row_table(sg, 'Policies'))
self.assertNotIn('Rules', sg)
def test_get_server_group(self): def test_get_server_group(self):
sg_id = self._create_sg("affinity") sg_id = self._create_sg("affinity")
@ -51,3 +74,47 @@ class TestServerGroupClientV213(test_server_groups.TestServerGroupClient):
self._get_column_value_from_single_row_table( self._get_column_value_from_single_row_table(
sg, "Project Id") sg, "Project Id")
self.assertEqual(sg_id, result) self.assertEqual(sg_id, result)
if self.expected_metadata:
self._get_column_value_from_single_row_table(sg, "Metadata")
else:
self.assertNotIn(sg, 'Metadata')
if self.expected_policy_rules:
self.assertEqual(
'affinity',
self._get_column_value_from_single_row_table(sg, "Policy"))
self.assertEqual(
'{}',
self._get_column_value_from_single_row_table(sg, "Rules"))
else:
self.assertIn(
'affinity',
self._get_column_value_from_single_row_table(sg, 'Policies'))
self.assertNotIn('Rules', sg)
class TestServerGroupClientV264(TestServerGroupClientV213):
"""Server groups v2.64 functional tests."""
COMPUTE_API_VERSION = "2.64"
expected_metadata = False
expected_policy_rules = True
def test_create_server_group(self):
output = self.nova('server-group-create complex-anti-affinity-group '
'anti-affinity --rule max_server_per_host=3')
sg_id = self._get_column_value_from_single_row_table(output, "Id")
self.addCleanup(self.nova, 'server-group-delete %s' % sg_id)
sg = self.nova('server-group-get %s' % sg_id)
result = self._get_column_value_from_single_row_table(sg, "Id")
self.assertEqual(sg_id, result)
self._get_column_value_from_single_row_table(
sg, "User Id")
self._get_column_value_from_single_row_table(
sg, "Project Id")
self.assertNotIn('Metadata', sg)
self.assertEqual(
'anti-affinity',
self._get_column_value_from_single_row_table(sg, "Policy"))
self.assertIn(
'max_server_per_host',
self._get_column_value_from_single_row_table(sg, "Rules"))

View File

@ -2236,8 +2236,13 @@ class FakeSessionClient(base_client.SessionClient):
return (200, {}, {"server_groups": server_groups}) return (200, {}, {"server_groups": server_groups})
def _return_server_group(self): def _return_server_group(self):
if self.api_version < api_versions.APIVersion("2.64"):
r = {'server_group': r = {'server_group':
self.get_os_server_groups()[2]['server_groups'][0]} self.get_os_server_groups()[2]['server_groups'][0]}
else:
r = {"members": [], "id": "2cbd51f4-fafe-4cdb-801b-cf913a6f288b",
'server_group': {'name': 'ig1', 'policy': 'anti-affinity',
'rules': {'max_server_per_host': 3}}}
return (200, {}, r) return (200, {}, r)
def post_os_server_groups(self, body, **kw): def post_os_server_groups(self, body, **kw):

View File

@ -13,6 +13,7 @@
# License for the specific language governing permissions and limitations # License for the specific language governing permissions and limitations
# under the License. # under the License.
from novaclient import api_versions
from novaclient import exceptions from novaclient import exceptions
from novaclient.tests.unit.fixture_data import client from novaclient.tests.unit.fixture_data import client
from novaclient.tests.unit.fixture_data import server_groups as data from novaclient.tests.unit.fixture_data import server_groups as data
@ -106,3 +107,36 @@ class ServerGroupsTest(utils.FixturedTestCase):
self.cs.server_groups.find, self.cs.server_groups.find,
**kwargs) **kwargs)
self.assert_called('GET', '/os-server-groups') self.assert_called('GET', '/os-server-groups')
class ServerGroupsTestV264(ServerGroupsTest):
def setUp(self):
super(ServerGroupsTestV264, self).setUp()
self.cs.api_version = api_versions.APIVersion("2.64")
def test_create_server_group(self):
name = 'ig1'
policy = 'anti-affinity'
server_group = self.cs.server_groups.create(name, policy)
self.assert_request_id(server_group, fakes.FAKE_REQUEST_ID_LIST)
body = {'server_group': {'name': name, 'policy': policy}}
self.assert_called('POST', '/os-server-groups', body)
self.assertIsInstance(server_group,
server_groups.ServerGroup)
def test_create_server_group_with_rules(self):
kwargs = {'name': 'ig1',
'policy': 'anti-affinity',
'rules': {'max_server_per_host': 3}}
server_group = self.cs.server_groups.create(**kwargs)
self.assert_request_id(server_group, fakes.FAKE_REQUEST_ID_LIST)
body = {
'server_group': {
'name': 'ig1',
'policy': 'anti-affinity',
'rules': {'max_server_per_host': 3}
}
}
self.assert_called('POST', '/os-server-groups', body)
self.assertIsInstance(server_group,
server_groups.ServerGroup)

View File

@ -3728,6 +3728,48 @@ class ShellTest(utils.TestCase):
{'server_group': {'name': 'wjsg', {'server_group': {'name': 'wjsg',
'policies': ['affinity']}}) 'policies': ['affinity']}})
def test_create_server_group_v2_64(self):
self.run_command('server-group-create sg1 affinity',
api_version='2.64')
self.assert_called('POST', '/os-server-groups',
{'server_group': {
'name': 'sg1',
'policy': 'affinity'
}})
def test_create_server_group_with_rules(self):
self.run_command('server-group-create sg1 anti-affinity '
'--rule max_server_per_host=3', api_version='2.64')
self.assert_called('POST', '/os-server-groups',
{'server_group': {
'name': 'sg1',
'policy': 'anti-affinity',
'rules': {'max_server_per_host': 3}
}})
def test_create_server_group_with_multi_rules(self):
self.run_command('server-group-create sg1 anti-affinity '
'--rule a=b --rule c=d', api_version='2.64')
self.assert_called('POST', '/os-server-groups',
{'server_group': {
'name': 'sg1',
'policy': 'anti-affinity',
'rules': {'a': 'b', 'c': 'd'}
}})
def test_create_server_group_with_invalid_value(self):
result = self.assertRaises(
exceptions.CommandError, self.run_command,
'server-group-create sg1 anti-affinity '
'--rule max_server_per_host=foo', api_version='2.64')
self.assertIn("Invalid 'max_server_per_host' value: foo",
six.text_type(result))
def test_create_server_group_with_rules_pre_264(self):
self.assertRaises(SystemExit, self.run_command,
'server-group-create sg1 anti-affinity '
'--rule max_server_per_host=3', api_version='2.63')
def test_create_server_group_with_multiple_policies(self): def test_create_server_group_with_multiple_policies(self):
self.assertRaises(SystemExit, self.run_command, self.assertRaises(SystemExit, self.run_command,
'server-group-create wjsg affinity anti-affinity') 'server-group-create wjsg affinity anti-affinity')
@ -3758,6 +3800,9 @@ class ShellTest(utils.TestCase):
7, # doesn't require any changes in novaclient 7, # doesn't require any changes in novaclient
9, # doesn't require any changes in novaclient 9, # doesn't require any changes in novaclient
12, # no longer supported 12, # no longer supported
13, # 13 adds information ``project_id`` and ``user_id`` to
# ``os-server-groups``, but is not explicitly tested
# via wraps and _SUBSTITUTIONS.
15, # doesn't require any changes in novaclient 15, # doesn't require any changes in novaclient
16, # doesn't require any changes in novaclient 16, # doesn't require any changes in novaclient
18, # NOTE(andreykurilin): this microversion requires changes in 18, # NOTE(andreykurilin): this microversion requires changes in

View File

@ -17,7 +17,10 @@
Server group interface. Server group interface.
""" """
from novaclient import api_versions
from novaclient import base from novaclient import base
from novaclient import exceptions
from novaclient.i18n import _
class ServerGroup(base.Resource): class ServerGroup(base.Resource):
@ -80,6 +83,7 @@ class ServerGroupsManager(base.ManagerWithFind):
""" """
return self._delete('/os-server-groups/%s' % id) return self._delete('/os-server-groups/%s' % id)
@api_versions.wraps("2.0", "2.63")
def create(self, name, policies): def create(self, name, policies):
"""Create (allocate) a server group. """Create (allocate) a server group.
@ -92,3 +96,29 @@ class ServerGroupsManager(base.ManagerWithFind):
body = {'server_group': {'name': name, body = {'server_group': {'name': name,
'policies': policies}} 'policies': policies}}
return self._create('/os-server-groups', body, 'server_group') return self._create('/os-server-groups', body, 'server_group')
@api_versions.wraps("2.64")
def create(self, name, policy, rules=None):
"""Create (allocate) a server group.
:param name: The name of the server group.
:param policy: Policy name to associate with the server group.
:param rules: The rules of policy which is a dict, can be applied to
the policy, now only ``max_server_per_host`` for ``anti-affinity``
policy would be supported (optional).
:rtype: list of :class:`ServerGroup`
"""
body = {'server_group': {
'name': name, 'policy': policy
}}
if rules:
key = 'max_server_per_host'
try:
if key in rules:
rules[key] = int(rules[key])
except ValueError:
msg = _("Invalid '%(key)s' value: %(value)s")
raise exceptions.CommandError(msg % {
'key': key, 'value': rules[key]})
body['server_group']['rules'] = rules
return self._create('/os-server-groups', body, 'server_group')

View File

@ -4525,16 +4525,15 @@ def do_availability_zone_list(cs, _args):
sortby_index=None) sortby_index=None)
@api_versions.wraps("2.0", "2.12")
def _print_server_group_details(cs, server_group): def _print_server_group_details(cs, server_group):
if cs.api_version < api_versions.APIVersion('2.13'):
columns = ['Id', 'Name', 'Policies', 'Members', 'Metadata'] columns = ['Id', 'Name', 'Policies', 'Members', 'Metadata']
utils.print_list(server_group, columns) elif cs.api_version < api_versions.APIVersion('2.64'):
@api_versions.wraps("2.13")
def _print_server_group_details(cs, server_group): # noqa
columns = ['Id', 'Name', 'Project Id', 'User Id', columns = ['Id', 'Name', 'Project Id', 'User Id',
'Policies', 'Members', 'Metadata'] 'Policies', 'Members', 'Metadata']
else:
columns = ['Id', 'Name', 'Project Id', 'User Id',
'Policy', 'Rules', 'Members']
utils.print_list(server_group, columns) utils.print_list(server_group, columns)
@ -4569,6 +4568,7 @@ def do_server_group_list(cs, args):
_print_server_group_details(cs, server_groups) _print_server_group_details(cs, server_groups)
@api_versions.wraps("2.0", "2.63")
@utils.arg('name', metavar='<name>', help=_('Server group name.')) @utils.arg('name', metavar='<name>', help=_('Server group name.'))
@utils.arg( @utils.arg(
'policy', 'policy',
@ -4581,6 +4581,30 @@ def do_server_group_create(cs, args):
_print_server_group_details(cs, [server_group]) _print_server_group_details(cs, [server_group])
@api_versions.wraps("2.64")
@utils.arg('name', metavar='<name>', help=_('Server group name.'))
@utils.arg(
'policy',
metavar='<policy>',
help=_('Policy for the server group.'))
@utils.arg(
'--rule',
metavar="<key=value>",
dest='rules',
action='append',
default=[],
help=_('A rule for the policy. Currently, only the '
'``max_server_per_host`` rule is supported for the '
'``anti-affinity`` policy.'))
def do_server_group_create(cs, args):
"""Create a new server group with the specified details."""
rules = _meta_parsing(args.rules)
server_group = cs.server_groups.create(name=args.name,
policy=args.policy,
rules=rules)
_print_server_group_details(cs, [server_group])
@utils.arg( @utils.arg(
'id', 'id',
metavar='<id>', metavar='<id>',

View File

@ -0,0 +1,15 @@
---
features:
- |
Added support for `microversion 2.64`_, which includes the following
changes:
* The ``--rule`` options is added to the ``nova server-group-create``
CLI that enables user to create server group with specific policy rules.
* Remove ``metadata`` column in the output of ``nova server-group-create``,
``nova server-group-get``, ``nova server-group-list``.
* Remove ``policies`` column, add ``policy`` and ``rules`` columns in
the output of ``nova server-group-create``, ``nova server-group-get``,
``nova server-group-list``.
.. _microversion 2.64: https://docs.openstack.org/nova/latest/api_microversion_history.html#id58