Add "server group create" command
Support compute v2 "server group create" command in OSC. Implements: blueprint nova-server-group-support Partial-Bug: #1542171 Change-Id: I96ffb07764d3adb715e048943cfee3b879c280f6
This commit is contained in:
parent
4639148b1d
commit
a06bb28bcc
29
doc/source/command-objects/server-group.rst
Normal file
29
doc/source/command-objects/server-group.rst
Normal file
@ -0,0 +1,29 @@
|
|||||||
|
============
|
||||||
|
server group
|
||||||
|
============
|
||||||
|
|
||||||
|
Server group provide a mechanism to group servers according to certain policy.
|
||||||
|
|
||||||
|
Compute v2
|
||||||
|
|
||||||
|
server group create
|
||||||
|
-------------------
|
||||||
|
|
||||||
|
Create a new server group
|
||||||
|
|
||||||
|
.. program:: server group create
|
||||||
|
.. code-block:: bash
|
||||||
|
|
||||||
|
os server group create
|
||||||
|
--policy <policy> [--policy <policy>] ...
|
||||||
|
<name>
|
||||||
|
|
||||||
|
.. option:: --policy <policy>
|
||||||
|
|
||||||
|
Add a policy to :ref:`\<name\> <server_group_create-name>`
|
||||||
|
(repeat option to add multiple policies)
|
||||||
|
|
||||||
|
.. _server_group_create-name:
|
||||||
|
.. describe:: <name>
|
||||||
|
|
||||||
|
New server group name
|
@ -118,6 +118,7 @@ referring to both Compute and Volume quotas.
|
|||||||
* ``security group rule``: (**Compute**, **Network**) - the individual rules that define protocol/IP/port access
|
* ``security group rule``: (**Compute**, **Network**) - the individual rules that define protocol/IP/port access
|
||||||
* ``server``: (**Compute**) virtual machine instance
|
* ``server``: (**Compute**) virtual machine instance
|
||||||
* ``server dump``: (**Compute**) a dump file of a server created by features like kdump
|
* ``server dump``: (**Compute**) a dump file of a server created by features like kdump
|
||||||
|
* ``server group``: (**Compute**) a grouping of servers
|
||||||
* ``server image``: (**Compute**) saved server disk image
|
* ``server image``: (**Compute**) saved server disk image
|
||||||
* ``service``: (**Identity**) a cloud service
|
* ``service``: (**Identity**) a cloud service
|
||||||
* ``service provider``: (**Identity**) a resource that consumes assertions from an ``identity provider``
|
* ``service provider``: (**Identity**) a resource that consumes assertions from an ``identity provider``
|
||||||
|
68
openstackclient/compute/v2/server_group.py
Normal file
68
openstackclient/compute/v2/server_group.py
Normal file
@ -0,0 +1,68 @@
|
|||||||
|
# Copyright 2016 Huawei, 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.
|
||||||
|
#
|
||||||
|
|
||||||
|
"""Compute v2 Server Group action implementations"""
|
||||||
|
|
||||||
|
from openstackclient.common import command
|
||||||
|
from openstackclient.common import utils
|
||||||
|
|
||||||
|
|
||||||
|
_formatters = {
|
||||||
|
'policies': utils.format_list,
|
||||||
|
'members': utils.format_list,
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
def _get_columns(info):
|
||||||
|
columns = list(info.keys())
|
||||||
|
if 'metadata' in columns:
|
||||||
|
# NOTE(RuiChen): The metadata of server group is always empty since API
|
||||||
|
# compatible, so hide it in order to avoid confusion.
|
||||||
|
columns.remove('metadata')
|
||||||
|
return tuple(sorted(columns))
|
||||||
|
|
||||||
|
|
||||||
|
class CreateServerGroup(command.ShowOne):
|
||||||
|
"""Create a new server group."""
|
||||||
|
|
||||||
|
def get_parser(self, prog_name):
|
||||||
|
parser = super(CreateServerGroup, self).get_parser(prog_name)
|
||||||
|
parser.add_argument(
|
||||||
|
'name',
|
||||||
|
metavar='<name>',
|
||||||
|
help='New server group name',
|
||||||
|
)
|
||||||
|
parser.add_argument(
|
||||||
|
'--policy',
|
||||||
|
metavar='<policy>',
|
||||||
|
action='append',
|
||||||
|
required=True,
|
||||||
|
help='Add a policy to <name> '
|
||||||
|
'(repeat option to add multiple policies)',
|
||||||
|
)
|
||||||
|
return parser
|
||||||
|
|
||||||
|
def take_action(self, parsed_args):
|
||||||
|
compute_client = self.app.client_manager.compute
|
||||||
|
info = {}
|
||||||
|
server_group = compute_client.server_groups.create(
|
||||||
|
name=parsed_args.name,
|
||||||
|
policies=parsed_args.policy)
|
||||||
|
info.update(server_group._info)
|
||||||
|
|
||||||
|
columns = _get_columns(info)
|
||||||
|
data = utils.get_dict_properties(info, columns,
|
||||||
|
formatters=_formatters)
|
||||||
|
return columns, data
|
@ -177,6 +177,9 @@ class FakeComputev2Client(object):
|
|||||||
self.hosts = mock.Mock()
|
self.hosts = mock.Mock()
|
||||||
self.hosts.resource_class = fakes.FakeResource(None, {})
|
self.hosts.resource_class = fakes.FakeResource(None, {})
|
||||||
|
|
||||||
|
self.server_groups = mock.Mock()
|
||||||
|
self.server_groups.resource_class = fakes.FakeResource(None, {})
|
||||||
|
|
||||||
self.auth_token = kwargs['token']
|
self.auth_token = kwargs['token']
|
||||||
|
|
||||||
self.management_url = kwargs['endpoint']
|
self.management_url = kwargs['endpoint']
|
||||||
@ -899,3 +902,34 @@ class FakeHost(object):
|
|||||||
info=copy.deepcopy(host_info),
|
info=copy.deepcopy(host_info),
|
||||||
loaded=True)
|
loaded=True)
|
||||||
return host
|
return host
|
||||||
|
|
||||||
|
|
||||||
|
class FakeServerGroup(object):
|
||||||
|
"""Fake one server group"""
|
||||||
|
|
||||||
|
@staticmethod
|
||||||
|
def create_one_server_group(attrs=None):
|
||||||
|
"""Create a fake server group
|
||||||
|
|
||||||
|
:param Dictionary attrs:
|
||||||
|
A dictionary with all attributes
|
||||||
|
:return:
|
||||||
|
A FakeResource object, with id and other attributes
|
||||||
|
"""
|
||||||
|
if attrs is None:
|
||||||
|
attrs = {}
|
||||||
|
|
||||||
|
server_group_info = {
|
||||||
|
'id': 'server-group-id-' + uuid.uuid4().hex,
|
||||||
|
'members': [],
|
||||||
|
'metadata': {},
|
||||||
|
'name': 'server-group-name-' + uuid.uuid4().hex,
|
||||||
|
'policies': [],
|
||||||
|
'project_id': 'server-group-project-id-' + uuid.uuid4().hex,
|
||||||
|
'user_id': 'server-group-user-id-' + uuid.uuid4().hex,
|
||||||
|
}
|
||||||
|
server_group_info.update(attrs)
|
||||||
|
server_group = fakes.FakeResource(
|
||||||
|
info=copy.deepcopy(server_group_info),
|
||||||
|
loaded=True)
|
||||||
|
return server_group
|
||||||
|
108
openstackclient/tests/compute/v2/test_server_group.py
Normal file
108
openstackclient/tests/compute/v2/test_server_group.py
Normal file
@ -0,0 +1,108 @@
|
|||||||
|
# Copyright 2016 Huawei, 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 openstackclient.common import utils
|
||||||
|
from openstackclient.compute.v2 import server_group
|
||||||
|
from openstackclient.tests.compute.v2 import fakes as compute_fakes
|
||||||
|
from openstackclient.tests import utils as tests_utils
|
||||||
|
|
||||||
|
|
||||||
|
class TestServerGroup(compute_fakes.TestComputev2):
|
||||||
|
|
||||||
|
fake_server_group = compute_fakes.FakeServerGroup.create_one_server_group()
|
||||||
|
|
||||||
|
columns = (
|
||||||
|
'id',
|
||||||
|
'members',
|
||||||
|
'name',
|
||||||
|
'policies',
|
||||||
|
'project_id',
|
||||||
|
'user_id',
|
||||||
|
)
|
||||||
|
|
||||||
|
data = (
|
||||||
|
fake_server_group.id,
|
||||||
|
utils.format_list(fake_server_group.members),
|
||||||
|
fake_server_group.name,
|
||||||
|
utils.format_list(fake_server_group.policies),
|
||||||
|
fake_server_group.project_id,
|
||||||
|
fake_server_group.user_id,
|
||||||
|
)
|
||||||
|
|
||||||
|
def setUp(self):
|
||||||
|
super(TestServerGroup, self).setUp()
|
||||||
|
|
||||||
|
# Get a shortcut to the ServerGroupsManager Mock
|
||||||
|
self.server_groups_mock = self.app.client_manager.compute.server_groups
|
||||||
|
self.server_groups_mock.reset_mock()
|
||||||
|
|
||||||
|
|
||||||
|
class TestServerGroupCreate(TestServerGroup):
|
||||||
|
|
||||||
|
def setUp(self):
|
||||||
|
super(TestServerGroupCreate, self).setUp()
|
||||||
|
|
||||||
|
self.server_groups_mock.create.return_value = self.fake_server_group
|
||||||
|
self.cmd = server_group.CreateServerGroup(self.app, None)
|
||||||
|
|
||||||
|
def test_server_group_create(self):
|
||||||
|
arglist = [
|
||||||
|
'--policy', 'affinity',
|
||||||
|
'affinity_group',
|
||||||
|
]
|
||||||
|
verifylist = [
|
||||||
|
('policy', ['affinity']),
|
||||||
|
('name', 'affinity_group'),
|
||||||
|
]
|
||||||
|
parsed_args = self.check_parser(self.cmd, arglist, verifylist)
|
||||||
|
columns, data = self.cmd.take_action(parsed_args)
|
||||||
|
self.server_groups_mock.create.assert_called_once_with(
|
||||||
|
name=parsed_args.name,
|
||||||
|
policies=parsed_args.policy,
|
||||||
|
)
|
||||||
|
|
||||||
|
self.assertEqual(self.columns, columns)
|
||||||
|
self.assertEqual(self.data, data)
|
||||||
|
|
||||||
|
def test_server_group_create_with_multiple_policies(self):
|
||||||
|
arglist = [
|
||||||
|
'--policy', 'affinity',
|
||||||
|
'--policy', 'soft-affinity',
|
||||||
|
'affinity_group',
|
||||||
|
]
|
||||||
|
verifylist = [
|
||||||
|
('policy', ['affinity', 'soft-affinity']),
|
||||||
|
('name', 'affinity_group'),
|
||||||
|
]
|
||||||
|
parsed_args = self.check_parser(self.cmd, arglist, verifylist)
|
||||||
|
columns, data = self.cmd.take_action(parsed_args)
|
||||||
|
self.server_groups_mock.create.assert_called_once_with(
|
||||||
|
name=parsed_args.name,
|
||||||
|
policies=parsed_args.policy,
|
||||||
|
)
|
||||||
|
|
||||||
|
self.assertEqual(self.columns, columns)
|
||||||
|
self.assertEqual(self.data, data)
|
||||||
|
|
||||||
|
def test_server_group_create_no_policy(self):
|
||||||
|
arglist = [
|
||||||
|
'affinity_group',
|
||||||
|
]
|
||||||
|
verifylist = None
|
||||||
|
self.assertRaises(tests_utils.ParserException,
|
||||||
|
self.check_parser,
|
||||||
|
self.cmd,
|
||||||
|
arglist,
|
||||||
|
verifylist)
|
@ -131,6 +131,8 @@ openstack.compute.v2 =
|
|||||||
server_unset = openstackclient.compute.v2.server:UnsetServer
|
server_unset = openstackclient.compute.v2.server:UnsetServer
|
||||||
server_unshelve = openstackclient.compute.v2.server:UnshelveServer
|
server_unshelve = openstackclient.compute.v2.server:UnshelveServer
|
||||||
|
|
||||||
|
server_group_create = openstackclient.compute.v2.server_group:CreateServerGroup
|
||||||
|
|
||||||
usage_list = openstackclient.compute.v2.usage:ListUsage
|
usage_list = openstackclient.compute.v2.usage:ListUsage
|
||||||
usage_show = openstackclient.compute.v2.usage:ShowUsage
|
usage_show = openstackclient.compute.v2.usage:ShowUsage
|
||||||
|
|
||||||
|
Loading…
x
Reference in New Issue
Block a user