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:
parent
003ac57d9a
commit
6c398058a6
@ -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:
|
||||||
|
|
||||||
|
@ -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")
|
||||||
|
@ -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"))
|
||||||
|
@ -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):
|
||||||
r = {'server_group':
|
if self.api_version < api_versions.APIVersion("2.64"):
|
||||||
self.get_os_server_groups()[2]['server_groups'][0]}
|
r = {'server_group':
|
||||||
|
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):
|
||||||
|
@ -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)
|
||||||
|
@ -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
|
||||||
|
@ -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')
|
||||||
|
@ -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):
|
||||||
columns = ['Id', 'Name', 'Policies', 'Members', 'Metadata']
|
if cs.api_version < api_versions.APIVersion('2.13'):
|
||||||
utils.print_list(server_group, columns)
|
columns = ['Id', 'Name', 'Policies', 'Members', 'Metadata']
|
||||||
|
elif cs.api_version < api_versions.APIVersion('2.64'):
|
||||||
|
columns = ['Id', 'Name', 'Project Id', 'User Id',
|
||||||
@api_versions.wraps("2.13")
|
'Policies', 'Members', 'Metadata']
|
||||||
def _print_server_group_details(cs, server_group): # noqa
|
else:
|
||||||
columns = ['Id', 'Name', 'Project Id', 'User Id',
|
columns = ['Id', 'Name', 'Project Id', 'User Id',
|
||||||
'Policies', 'Members', 'Metadata']
|
'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>',
|
||||||
|
@ -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
|
Loading…
Reference in New Issue
Block a user