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:
Fang Zhen 2016-03-14 19:27:36 +08:00
parent 35ea7a9baa
commit b92cf77fb5
6 changed files with 175 additions and 14 deletions

View File

@ -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
---------- ----------

View File

@ -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)

View File

@ -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)

View File

@ -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):

View File

@ -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,
} }

View File

@ -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>`_]