Add network support for "quota set"
The "quota set" command support compute and volume quotas previously. This patch add support network. Partially-implements: blueprint neutron-client-quota Closes-bug: 1489441 Change-Id: I9d297f52bc30614b3493f09ed15f8f1d3f8ff952
This commit is contained in:
parent
35ea7a9baa
commit
b92cf77fb5
@ -32,6 +32,20 @@ Set quotas for project
|
|||||||
[--volumes <new-volumes>]
|
[--volumes <new-volumes>]
|
||||||
[--volume-type <volume-type>]
|
[--volume-type <volume-type>]
|
||||||
|
|
||||||
|
# Network settings
|
||||||
|
[--floating-ips <num-floatingips>]
|
||||||
|
[--secgroup-rules <num-security-group-rules>]
|
||||||
|
[--secgroups <num-security-groups>]
|
||||||
|
[--networks <num-networks>]
|
||||||
|
[--subnets <num-subnets>]
|
||||||
|
[--ports <num-ports>]
|
||||||
|
[--routers <num-routers>]
|
||||||
|
[--rbac-policies <num-rbac-policies>]
|
||||||
|
[--vips <num-vips>]
|
||||||
|
[--subnetpools <num-subnetpools>]
|
||||||
|
[--members <num-members>]
|
||||||
|
[--health-monitors <num-health-monitors>]
|
||||||
|
|
||||||
<project>
|
<project>
|
||||||
|
|
||||||
Set quotas for class
|
Set quotas for class
|
||||||
@ -126,6 +140,42 @@ Set quotas for class
|
|||||||
|
|
||||||
Set quotas for a specific <volume-type>
|
Set quotas for a specific <volume-type>
|
||||||
|
|
||||||
|
.. option:: --networks <num-networks>
|
||||||
|
|
||||||
|
New value for the networks quota
|
||||||
|
|
||||||
|
.. option:: --subnets <num-subnets>
|
||||||
|
|
||||||
|
New value for the subnets quota
|
||||||
|
|
||||||
|
.. option:: --ports <num-ports>
|
||||||
|
|
||||||
|
New value for the ports quota
|
||||||
|
|
||||||
|
.. option:: --routers <num-routers>
|
||||||
|
|
||||||
|
New value for the routers quota
|
||||||
|
|
||||||
|
.. option:: --rbac-policies <num-rbac-policies>
|
||||||
|
|
||||||
|
New value for the rbac-policies quota
|
||||||
|
|
||||||
|
.. option:: --vips <num-vips>
|
||||||
|
|
||||||
|
New value for the vips quota
|
||||||
|
|
||||||
|
.. option:: --subnetpools <num-subnetpools>
|
||||||
|
|
||||||
|
New value for the subnetpools quota
|
||||||
|
|
||||||
|
.. option:: --members <num-members>
|
||||||
|
|
||||||
|
New value for the members quota
|
||||||
|
|
||||||
|
.. option:: --health-monitors <num-health-monitors>
|
||||||
|
|
||||||
|
New value for the health-monitors quota
|
||||||
|
|
||||||
quota show
|
quota show
|
||||||
----------
|
----------
|
||||||
|
|
||||||
|
@ -16,7 +16,7 @@ from functional.common import test
|
|||||||
class QuotaTests(test.TestCase):
|
class QuotaTests(test.TestCase):
|
||||||
"""Functional tests for quota. """
|
"""Functional tests for quota. """
|
||||||
# Test quota information for compute, network and volume.
|
# Test quota information for compute, network and volume.
|
||||||
EXPECTED_FIELDS = ['instances', 'network', 'volumes']
|
EXPECTED_FIELDS = ['instances', 'networks', 'volumes']
|
||||||
PROJECT_NAME = None
|
PROJECT_NAME = None
|
||||||
|
|
||||||
@classmethod
|
@classmethod
|
||||||
@ -25,12 +25,11 @@ class QuotaTests(test.TestCase):
|
|||||||
cls.get_openstack_configuration_value('auth.project_name')
|
cls.get_openstack_configuration_value('auth.project_name')
|
||||||
|
|
||||||
def test_quota_set(self):
|
def test_quota_set(self):
|
||||||
# TODO(rtheis): Add --network option once supported on set.
|
self.openstack('quota set --instances 11 --volumes 11 --networks 11 '
|
||||||
self.openstack('quota set --instances 11 --volumes 11 ' +
|
+ self.PROJECT_NAME)
|
||||||
self.PROJECT_NAME)
|
|
||||||
opts = self.get_show_opts(self.EXPECTED_FIELDS)
|
opts = self.get_show_opts(self.EXPECTED_FIELDS)
|
||||||
raw_output = self.openstack('quota show ' + self.PROJECT_NAME + opts)
|
raw_output = self.openstack('quota show ' + self.PROJECT_NAME + opts)
|
||||||
self.assertEqual("11\n10\n11\n", raw_output)
|
self.assertEqual("11\n11\n11\n", raw_output)
|
||||||
|
|
||||||
def test_quota_show(self):
|
def test_quota_show(self):
|
||||||
raw_output = self.openstack('quota show ' + self.PROJECT_NAME)
|
raw_output = self.openstack('quota show ' + self.PROJECT_NAME)
|
||||||
|
@ -29,7 +29,6 @@ from openstackclient.common import utils
|
|||||||
COMPUTE_QUOTAS = {
|
COMPUTE_QUOTAS = {
|
||||||
'cores': 'cores',
|
'cores': 'cores',
|
||||||
'fixed_ips': 'fixed-ips',
|
'fixed_ips': 'fixed-ips',
|
||||||
'floating_ips': 'floating-ips',
|
|
||||||
'injected_file_content_bytes': 'injected-file-size',
|
'injected_file_content_bytes': 'injected-file-size',
|
||||||
'injected_file_path_bytes': 'injected-path-size',
|
'injected_file_path_bytes': 'injected-path-size',
|
||||||
'injected_files': 'injected-files',
|
'injected_files': 'injected-files',
|
||||||
@ -37,8 +36,6 @@ COMPUTE_QUOTAS = {
|
|||||||
'key_pairs': 'key-pairs',
|
'key_pairs': 'key-pairs',
|
||||||
'metadata_items': 'properties',
|
'metadata_items': 'properties',
|
||||||
'ram': 'ram',
|
'ram': 'ram',
|
||||||
'security_group_rules': 'secgroup-rules',
|
|
||||||
'security_groups': 'secgroups',
|
|
||||||
}
|
}
|
||||||
|
|
||||||
VOLUME_QUOTAS = {
|
VOLUME_QUOTAS = {
|
||||||
@ -47,16 +44,41 @@ VOLUME_QUOTAS = {
|
|||||||
'volumes': 'volumes',
|
'volumes': 'volumes',
|
||||||
}
|
}
|
||||||
|
|
||||||
|
NOVA_NETWORK_QUOTAS = {
|
||||||
|
'floating_ips': 'floating-ips',
|
||||||
|
'security_group_rules': 'secgroup-rules',
|
||||||
|
'security_groups': 'secgroups',
|
||||||
|
}
|
||||||
|
|
||||||
NETWORK_QUOTAS = {
|
NETWORK_QUOTAS = {
|
||||||
'floatingip': 'floating-ips',
|
'floatingip': 'floating-ips',
|
||||||
'security_group_rule': 'secgroup-rules',
|
'security_group_rule': 'secgroup-rules',
|
||||||
'security_group': 'secgroups',
|
'security_group': 'secgroups',
|
||||||
|
'network': 'networks',
|
||||||
|
'subnet': 'subnets',
|
||||||
|
'port': 'ports',
|
||||||
|
'router': 'routers',
|
||||||
|
'rbac_policy': 'rbac-policies',
|
||||||
|
'vip': 'vips',
|
||||||
|
'subnetpool': 'subnetpools',
|
||||||
|
'member': 'members',
|
||||||
|
'health_monitor': 'health-monitors',
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
class SetQuota(command.Command):
|
class SetQuota(command.Command):
|
||||||
"""Set quotas for project or class"""
|
"""Set quotas for project or class"""
|
||||||
|
|
||||||
|
def _build_options_list(self):
|
||||||
|
if self.app.client_manager.is_network_endpoint_enabled():
|
||||||
|
return itertools.chain(COMPUTE_QUOTAS.items(),
|
||||||
|
VOLUME_QUOTAS.items(),
|
||||||
|
NETWORK_QUOTAS.items())
|
||||||
|
else:
|
||||||
|
return itertools.chain(COMPUTE_QUOTAS.items(),
|
||||||
|
VOLUME_QUOTAS.items(),
|
||||||
|
NOVA_NETWORK_QUOTAS.items())
|
||||||
|
|
||||||
def get_parser(self, prog_name):
|
def get_parser(self, prog_name):
|
||||||
parser = super(SetQuota, self).get_parser(prog_name)
|
parser = super(SetQuota, self).get_parser(prog_name)
|
||||||
parser.add_argument(
|
parser.add_argument(
|
||||||
@ -71,8 +93,7 @@ class SetQuota(command.Command):
|
|||||||
default=False,
|
default=False,
|
||||||
help='Set quotas for <class>',
|
help='Set quotas for <class>',
|
||||||
)
|
)
|
||||||
for k, v in itertools.chain(
|
for k, v in self._build_options_list():
|
||||||
COMPUTE_QUOTAS.items(), VOLUME_QUOTAS.items()):
|
|
||||||
parser.add_argument(
|
parser.add_argument(
|
||||||
'--%s' % v,
|
'--%s' % v,
|
||||||
metavar='<%s>' % v,
|
metavar='<%s>' % v,
|
||||||
@ -92,7 +113,7 @@ class SetQuota(command.Command):
|
|||||||
identity_client = self.app.client_manager.identity
|
identity_client = self.app.client_manager.identity
|
||||||
compute_client = self.app.client_manager.compute
|
compute_client = self.app.client_manager.compute
|
||||||
volume_client = self.app.client_manager.volume
|
volume_client = self.app.client_manager.volume
|
||||||
|
network_client = self.app.client_manager.network
|
||||||
compute_kwargs = {}
|
compute_kwargs = {}
|
||||||
for k, v in COMPUTE_QUOTAS.items():
|
for k, v in COMPUTE_QUOTAS.items():
|
||||||
value = getattr(parsed_args, k, None)
|
value = getattr(parsed_args, k, None)
|
||||||
@ -107,7 +128,20 @@ class SetQuota(command.Command):
|
|||||||
k = k + '_%s' % parsed_args.volume_type
|
k = k + '_%s' % parsed_args.volume_type
|
||||||
volume_kwargs[k] = value
|
volume_kwargs[k] = value
|
||||||
|
|
||||||
if compute_kwargs == {} and volume_kwargs == {}:
|
network_kwargs = {}
|
||||||
|
if self.app.client_manager.is_network_endpoint_enabled():
|
||||||
|
for k, v in NETWORK_QUOTAS.items():
|
||||||
|
value = getattr(parsed_args, k, None)
|
||||||
|
if value is not None:
|
||||||
|
network_kwargs[k] = value
|
||||||
|
else:
|
||||||
|
for k, v in NOVA_NETWORK_QUOTAS.items():
|
||||||
|
value = getattr(parsed_args, k, None)
|
||||||
|
if value is not None:
|
||||||
|
compute_kwargs[k] = value
|
||||||
|
|
||||||
|
if (compute_kwargs == {} and volume_kwargs == {}
|
||||||
|
and network_kwargs == {}):
|
||||||
sys.stderr.write("No quotas updated")
|
sys.stderr.write("No quotas updated")
|
||||||
return
|
return
|
||||||
|
|
||||||
@ -126,6 +160,9 @@ class SetQuota(command.Command):
|
|||||||
volume_client.quota_classes.update(
|
volume_client.quota_classes.update(
|
||||||
project.id,
|
project.id,
|
||||||
**volume_kwargs)
|
**volume_kwargs)
|
||||||
|
if network_kwargs:
|
||||||
|
sys.stderr.write("Network quotas are ignored since quota class"
|
||||||
|
"is not supported.")
|
||||||
else:
|
else:
|
||||||
if compute_kwargs:
|
if compute_kwargs:
|
||||||
compute_client.quotas.update(
|
compute_client.quotas.update(
|
||||||
@ -135,6 +172,10 @@ class SetQuota(command.Command):
|
|||||||
volume_client.quotas.update(
|
volume_client.quotas.update(
|
||||||
project.id,
|
project.id,
|
||||||
**volume_kwargs)
|
**volume_kwargs)
|
||||||
|
if network_kwargs:
|
||||||
|
network_client.update_quota(
|
||||||
|
project.id,
|
||||||
|
**network_kwargs)
|
||||||
|
|
||||||
|
|
||||||
class ShowQuota(command.ShowOne):
|
class ShowQuota(command.ShowOne):
|
||||||
@ -232,8 +273,8 @@ class ShowQuota(command.ShowOne):
|
|||||||
# neutron is enabled, quotas of these three resources
|
# neutron is enabled, quotas of these three resources
|
||||||
# in nova will be replaced by neutron's.
|
# in nova will be replaced by neutron's.
|
||||||
for k, v in itertools.chain(
|
for k, v in itertools.chain(
|
||||||
COMPUTE_QUOTAS.items(), VOLUME_QUOTAS.items(),
|
COMPUTE_QUOTAS.items(), NOVA_NETWORK_QUOTAS.items(),
|
||||||
NETWORK_QUOTAS.items()):
|
VOLUME_QUOTAS.items(), NETWORK_QUOTAS.items()):
|
||||||
if not k == v and info.get(k):
|
if not k == v and info.get(k):
|
||||||
info[v] = info[k]
|
info[v] = info[k]
|
||||||
info.pop(k)
|
info.pop(k)
|
||||||
|
@ -97,6 +97,9 @@ class TestQuotaSet(TestQuota):
|
|||||||
loaded=True,
|
loaded=True,
|
||||||
)
|
)
|
||||||
|
|
||||||
|
self.network_mock = self.app.client_manager.network
|
||||||
|
self.network_mock.update_quota = mock.Mock()
|
||||||
|
|
||||||
self.cmd = quota.SetQuota(self.app, None)
|
self.cmd = quota.SetQuota(self.app, None)
|
||||||
|
|
||||||
def test_quota_set(self):
|
def test_quota_set(self):
|
||||||
@ -132,6 +135,7 @@ class TestQuotaSet(TestQuota):
|
|||||||
('project', identity_fakes.project_name),
|
('project', identity_fakes.project_name),
|
||||||
]
|
]
|
||||||
|
|
||||||
|
self.app.client_manager.network_endpoint_enabled = False
|
||||||
parsed_args = self.check_parser(self.cmd, arglist, verifylist)
|
parsed_args = self.check_parser(self.cmd, arglist, verifylist)
|
||||||
|
|
||||||
self.cmd.take_action(parsed_args)
|
self.cmd.take_action(parsed_args)
|
||||||
@ -185,6 +189,61 @@ class TestQuotaSet(TestQuota):
|
|||||||
**kwargs
|
**kwargs
|
||||||
)
|
)
|
||||||
|
|
||||||
|
def test_quota_set_network(self):
|
||||||
|
arglist = [
|
||||||
|
'--subnets', str(network_fakes.QUOTA['subnet']),
|
||||||
|
'--networks', str(network_fakes.QUOTA['network']),
|
||||||
|
'--floating-ips', str(network_fakes.QUOTA['floatingip']),
|
||||||
|
'--subnetpools', str(network_fakes.QUOTA['subnetpool']),
|
||||||
|
'--secgroup-rules',
|
||||||
|
str(network_fakes.QUOTA['security_group_rule']),
|
||||||
|
'--secgroups', str(network_fakes.QUOTA['security_group']),
|
||||||
|
'--routers', str(network_fakes.QUOTA['router']),
|
||||||
|
'--rbac-policies', str(network_fakes.QUOTA['rbac_policy']),
|
||||||
|
'--ports', str(network_fakes.QUOTA['port']),
|
||||||
|
'--vips', str(network_fakes.QUOTA['vip']),
|
||||||
|
'--members', str(network_fakes.QUOTA['member']),
|
||||||
|
'--health-monitors', str(network_fakes.QUOTA['health_monitor']),
|
||||||
|
identity_fakes.project_name,
|
||||||
|
]
|
||||||
|
verifylist = [
|
||||||
|
('subnet', network_fakes.QUOTA['subnet']),
|
||||||
|
('network', network_fakes.QUOTA['network']),
|
||||||
|
('floatingip', network_fakes.QUOTA['floatingip']),
|
||||||
|
('subnetpool', network_fakes.QUOTA['subnetpool']),
|
||||||
|
('security_group_rule',
|
||||||
|
network_fakes.QUOTA['security_group_rule']),
|
||||||
|
('security_group', network_fakes.QUOTA['security_group']),
|
||||||
|
('router', network_fakes.QUOTA['router']),
|
||||||
|
('rbac_policy', network_fakes.QUOTA['rbac_policy']),
|
||||||
|
('port', network_fakes.QUOTA['port']),
|
||||||
|
('vip', network_fakes.QUOTA['vip']),
|
||||||
|
('member', network_fakes.QUOTA['member']),
|
||||||
|
('health_monitor', network_fakes.QUOTA['health_monitor']),
|
||||||
|
]
|
||||||
|
parsed_args = self.check_parser(self.cmd, arglist, verifylist)
|
||||||
|
|
||||||
|
self.cmd.take_action(parsed_args)
|
||||||
|
kwargs = {
|
||||||
|
'subnet': network_fakes.QUOTA['subnet'],
|
||||||
|
'network': network_fakes.QUOTA['network'],
|
||||||
|
'floatingip': network_fakes.QUOTA['floatingip'],
|
||||||
|
'subnetpool': network_fakes.QUOTA['subnetpool'],
|
||||||
|
'security_group_rule':
|
||||||
|
network_fakes.QUOTA['security_group_rule'],
|
||||||
|
'security_group': network_fakes.QUOTA['security_group'],
|
||||||
|
'router': network_fakes.QUOTA['router'],
|
||||||
|
'rbac_policy': network_fakes.QUOTA['rbac_policy'],
|
||||||
|
'port': network_fakes.QUOTA['port'],
|
||||||
|
'vip': network_fakes.QUOTA['vip'],
|
||||||
|
'member': network_fakes.QUOTA['member'],
|
||||||
|
'health_monitor': network_fakes.QUOTA['health_monitor'],
|
||||||
|
}
|
||||||
|
self.network_mock.update_quota.assert_called_with(
|
||||||
|
identity_fakes.project_id,
|
||||||
|
**kwargs
|
||||||
|
)
|
||||||
|
|
||||||
|
|
||||||
class TestQuotaShow(TestQuota):
|
class TestQuotaShow(TestQuota):
|
||||||
|
|
||||||
|
@ -36,6 +36,9 @@ QUOTA = {
|
|||||||
"router": 10,
|
"router": 10,
|
||||||
"rbac_policy": -1,
|
"rbac_policy": -1,
|
||||||
"port": 50,
|
"port": 50,
|
||||||
|
"vip": 10,
|
||||||
|
"member": 10,
|
||||||
|
"health_monitor": 10,
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
@ -0,0 +1,9 @@
|
|||||||
|
---
|
||||||
|
features:
|
||||||
|
- |
|
||||||
|
Add network support for ``quota set`` command. Options added includes
|
||||||
|
``--networks --subnets --subnetpools --ports --routers --rbac-policies``
|
||||||
|
``--vips --members --health-monitors``.
|
||||||
|
Options ``--floating-ips --secgroup-rules --secgroups`` now support
|
||||||
|
both network and compute API.
|
||||||
|
[Bug `1489441 <https://bugs.launchpad.net/bugs/1489441>`_]
|
Loading…
Reference in New Issue
Block a user