Merge "quota: Add 'quota delete' command"
This commit is contained in:
commit
eac38feef0
@ -91,7 +91,7 @@ qos-show,volume qos show,Shows a specified qos specs.
|
|||||||
quota-class-show,quota show --class,Lists quotas for a quota class.
|
quota-class-show,quota show --class,Lists quotas for a quota class.
|
||||||
quota-class-update,quota set --class,Updates quotas for a quota class.
|
quota-class-update,quota set --class,Updates quotas for a quota class.
|
||||||
quota-defaults,quota show --default,Lists default quotas for a tenant.
|
quota-defaults,quota show --default,Lists default quotas for a tenant.
|
||||||
quota-delete,,Delete the quotas for a tenant.
|
quota-delete,quota delete --volume,Delete the quotas for a tenant.
|
||||||
quota-show,quota show,Lists quotas for a tenant.
|
quota-show,quota show,Lists quotas for a tenant.
|
||||||
quota-update,quota set,Updates quotas for a tenant.
|
quota-update,quota set,Updates quotas for a tenant.
|
||||||
quota-usage,,Lists quota usage for a tenant.
|
quota-usage,,Lists quota usage for a tenant.
|
||||||
|
|
@ -188,7 +188,7 @@ qos-policy-list,network qos policy list,List QoS policies that belong to a given
|
|||||||
qos-policy-show,network qos policy show,Show information of a given qos policy.
|
qos-policy-show,network qos policy show,Show information of a given qos policy.
|
||||||
qos-policy-update,network qos policy set,Update a given qos policy.
|
qos-policy-update,network qos policy set,Update a given qos policy.
|
||||||
quota-default-show,quota show --default,Show default quotas for a given tenant.
|
quota-default-show,quota show --default,Show default quotas for a given tenant.
|
||||||
quota-delete,,Delete defined quotas of a given tenant.
|
quota-delete,quota delete --network,Delete defined quotas of a given tenant.
|
||||||
quota-list,quota list,List quotas of all tenants who have non-default quota values.
|
quota-list,quota list,List quotas of all tenants who have non-default quota values.
|
||||||
quota-show,quota show,Show quotas for a given tenant.
|
quota-show,quota show,Show quotas for a given tenant.
|
||||||
quota-update,quota set,Define tenant's quotas not to use defaults.
|
quota-update,quota set,Define tenant's quotas not to use defaults.
|
||||||
|
|
@ -70,7 +70,7 @@ pause,server pause,Pause a server.
|
|||||||
quota-class-show,quota show --class,List the quotas for a quota class.
|
quota-class-show,quota show --class,List the quotas for a quota class.
|
||||||
quota-class-update,quota set --class,Update the quotas for a quota class.
|
quota-class-update,quota set --class,Update the quotas for a quota class.
|
||||||
quota-defaults,quota list,List the default quotas for a tenant.
|
quota-defaults,quota list,List the default quotas for a tenant.
|
||||||
quota-delete,quota set,Delete quota for a tenant/user so their quota will Revert back to default.
|
quota-delete,quota delete --compute,Delete quota for a tenant/user so their quota will Revert back to default.
|
||||||
quota-show,quota show,List the quotas for a tenant/user.
|
quota-show,quota show,List the quotas for a tenant/user.
|
||||||
quota-update,quota set,Update the quotas for a tenant/user.
|
quota-update,quota set,Update the quotas for a tenant/user.
|
||||||
reboot,server reboot,Reboot a server.
|
reboot,server reboot,Reboot a server.
|
||||||
|
|
@ -697,3 +697,82 @@ class ShowQuota(command.ShowOne, BaseQuota):
|
|||||||
info['project_name'] = project_name
|
info['project_name'] = project_name
|
||||||
|
|
||||||
return zip(*sorted(info.items()))
|
return zip(*sorted(info.items()))
|
||||||
|
|
||||||
|
|
||||||
|
class DeleteQuota(command.Command):
|
||||||
|
_description = _(
|
||||||
|
"Delete configured quota for a project and revert to defaults."
|
||||||
|
)
|
||||||
|
|
||||||
|
def get_parser(self, prog_name):
|
||||||
|
parser = super().get_parser(prog_name)
|
||||||
|
parser.add_argument(
|
||||||
|
'project',
|
||||||
|
metavar='<project>',
|
||||||
|
help=_('Delete quotas for this project (name or ID)'),
|
||||||
|
)
|
||||||
|
option = parser.add_mutually_exclusive_group()
|
||||||
|
option.add_argument(
|
||||||
|
'--all',
|
||||||
|
action='store_const',
|
||||||
|
const='all',
|
||||||
|
dest='service',
|
||||||
|
default='all',
|
||||||
|
help=_('Delete project quotas for all services (default)'),
|
||||||
|
)
|
||||||
|
option.add_argument(
|
||||||
|
'--compute',
|
||||||
|
action='store_const',
|
||||||
|
const='compute',
|
||||||
|
dest='service',
|
||||||
|
default='all',
|
||||||
|
help=_(
|
||||||
|
'Delete compute quotas for the project '
|
||||||
|
'(including network quotas when using nova-network)'
|
||||||
|
),
|
||||||
|
)
|
||||||
|
option.add_argument(
|
||||||
|
'--volume',
|
||||||
|
action='store_const',
|
||||||
|
const='volume',
|
||||||
|
dest='service',
|
||||||
|
default='all',
|
||||||
|
help=_('Delete volume quotas for the project'),
|
||||||
|
)
|
||||||
|
option.add_argument(
|
||||||
|
'--network',
|
||||||
|
action='store_const',
|
||||||
|
const='network',
|
||||||
|
dest='service',
|
||||||
|
default='all',
|
||||||
|
help=_('Delete network quotas for the project'),
|
||||||
|
)
|
||||||
|
return parser
|
||||||
|
|
||||||
|
def take_action(self, parsed_args):
|
||||||
|
identity_client = self.app.client_manager.identity
|
||||||
|
project = utils.find_resource(
|
||||||
|
identity_client.projects,
|
||||||
|
parsed_args.project,
|
||||||
|
)
|
||||||
|
|
||||||
|
# compute quotas
|
||||||
|
if parsed_args.service in {'all', 'compute'}:
|
||||||
|
compute_client = self.app.client_manager.compute
|
||||||
|
compute_client.quotas.delete(project)
|
||||||
|
|
||||||
|
# volume quotas
|
||||||
|
if parsed_args.service in {'all', 'volume'}:
|
||||||
|
volume_client = self.app.client_manager.volume
|
||||||
|
volume_client.quotas.delete(project)
|
||||||
|
|
||||||
|
# network quotas (but only if we're not using nova-network, otherwise
|
||||||
|
# we already deleted the quotas in the compute step)
|
||||||
|
if (
|
||||||
|
parsed_args.service in {'all', 'network'}
|
||||||
|
and self.app.client_manager.is_network_endpoint_enabled()
|
||||||
|
):
|
||||||
|
network_client = self.app.client_manager.network
|
||||||
|
network_client.quotas.delete(project)
|
||||||
|
|
||||||
|
return None
|
||||||
|
@ -62,6 +62,9 @@ class TestQuota(compute_fakes.TestComputev2):
|
|||||||
self.app.client_manager.volume.quota_classes
|
self.app.client_manager.volume.quota_classes
|
||||||
self.volume_quotas_class_mock.reset_mock()
|
self.volume_quotas_class_mock.reset_mock()
|
||||||
|
|
||||||
|
self.app.client_manager.network.quotas = mock.Mock()
|
||||||
|
self.network_quotas_mock = self.app.client_manager.network.quotas
|
||||||
|
|
||||||
self.app.client_manager.auth_ref = mock.Mock()
|
self.app.client_manager.auth_ref = mock.Mock()
|
||||||
self.app.client_manager.auth_ref.service_catalog = mock.Mock()
|
self.app.client_manager.auth_ref.service_catalog = mock.Mock()
|
||||||
self.service_catalog_mock = \
|
self.service_catalog_mock = \
|
||||||
@ -1154,3 +1157,107 @@ class TestQuotaShow(TestQuota):
|
|||||||
self.assertEqual(len(network_fakes.QUOTA) - 1, len(result))
|
self.assertEqual(len(network_fakes.QUOTA) - 1, len(result))
|
||||||
# Go back to default mock
|
# Go back to default mock
|
||||||
self.network.get_quota = orig_get_quota
|
self.network.get_quota = orig_get_quota
|
||||||
|
|
||||||
|
|
||||||
|
class TestQuotaDelete(TestQuota):
|
||||||
|
"""Test cases for quota delete command"""
|
||||||
|
|
||||||
|
def setUp(self):
|
||||||
|
super().setUp()
|
||||||
|
|
||||||
|
self.cmd = quota.DeleteQuota(self.app, None)
|
||||||
|
|
||||||
|
def test_delete(self):
|
||||||
|
"""Delete all quotas"""
|
||||||
|
arglist = [
|
||||||
|
self.projects[0].id,
|
||||||
|
]
|
||||||
|
verifylist = [
|
||||||
|
('service', 'all'),
|
||||||
|
('project', self.projects[0].id),
|
||||||
|
]
|
||||||
|
|
||||||
|
parsed_args = self.check_parser(self.cmd, arglist, verifylist)
|
||||||
|
|
||||||
|
result = self.cmd.take_action(parsed_args)
|
||||||
|
|
||||||
|
self.assertIsNone(result)
|
||||||
|
self.projects_mock.get.assert_called_once_with(self.projects[0].id)
|
||||||
|
self.compute_quotas_mock.delete.assert_called_once_with(
|
||||||
|
self.projects[0],
|
||||||
|
)
|
||||||
|
self.volume_quotas_mock.delete.assert_called_once_with(
|
||||||
|
self.projects[0],
|
||||||
|
)
|
||||||
|
self.network_quotas_mock.delete.assert_called_once_with(
|
||||||
|
self.projects[0],
|
||||||
|
)
|
||||||
|
|
||||||
|
def test_delete__compute(self):
|
||||||
|
"""Delete compute quotas only"""
|
||||||
|
arglist = [
|
||||||
|
'--compute',
|
||||||
|
self.projects[0].id,
|
||||||
|
]
|
||||||
|
verifylist = [
|
||||||
|
('service', 'compute'),
|
||||||
|
('project', self.projects[0].id),
|
||||||
|
]
|
||||||
|
|
||||||
|
parsed_args = self.check_parser(self.cmd, arglist, verifylist)
|
||||||
|
|
||||||
|
result = self.cmd.take_action(parsed_args)
|
||||||
|
|
||||||
|
self.assertIsNone(result)
|
||||||
|
self.projects_mock.get.assert_called_once_with(self.projects[0].id)
|
||||||
|
self.compute_quotas_mock.delete.assert_called_once_with(
|
||||||
|
self.projects[0],
|
||||||
|
)
|
||||||
|
self.volume_quotas_mock.delete.assert_not_called()
|
||||||
|
self.network_quotas_mock.delete.assert_not_called()
|
||||||
|
|
||||||
|
def test_delete__volume(self):
|
||||||
|
"""Delete volume quotas only"""
|
||||||
|
arglist = [
|
||||||
|
'--volume',
|
||||||
|
self.projects[0].id,
|
||||||
|
]
|
||||||
|
verifylist = [
|
||||||
|
('service', 'volume'),
|
||||||
|
('project', self.projects[0].id),
|
||||||
|
]
|
||||||
|
|
||||||
|
parsed_args = self.check_parser(self.cmd, arglist, verifylist)
|
||||||
|
|
||||||
|
result = self.cmd.take_action(parsed_args)
|
||||||
|
|
||||||
|
self.assertIsNone(result)
|
||||||
|
self.projects_mock.get.assert_called_once_with(self.projects[0].id)
|
||||||
|
self.compute_quotas_mock.delete.assert_not_called()
|
||||||
|
self.volume_quotas_mock.delete.assert_called_once_with(
|
||||||
|
self.projects[0],
|
||||||
|
)
|
||||||
|
self.network_quotas_mock.delete.assert_not_called()
|
||||||
|
|
||||||
|
def test_delete__network(self):
|
||||||
|
"""Delete network quotas only"""
|
||||||
|
arglist = [
|
||||||
|
'--network',
|
||||||
|
self.projects[0].id,
|
||||||
|
]
|
||||||
|
verifylist = [
|
||||||
|
('service', 'network'),
|
||||||
|
('project', self.projects[0].id),
|
||||||
|
]
|
||||||
|
|
||||||
|
parsed_args = self.check_parser(self.cmd, arglist, verifylist)
|
||||||
|
|
||||||
|
result = self.cmd.take_action(parsed_args)
|
||||||
|
|
||||||
|
self.assertIsNone(result)
|
||||||
|
self.projects_mock.get.assert_called_once_with(self.projects[0].id)
|
||||||
|
self.compute_quotas_mock.delete.assert_not_called()
|
||||||
|
self.volume_quotas_mock.delete.assert_not_called()
|
||||||
|
self.network_quotas_mock.delete.assert_called_once_with(
|
||||||
|
self.projects[0],
|
||||||
|
)
|
||||||
|
5
releasenotes/notes/quota-delete-947df66ae5341cbf.yaml
Normal file
5
releasenotes/notes/quota-delete-947df66ae5341cbf.yaml
Normal file
@ -0,0 +1,5 @@
|
|||||||
|
---
|
||||||
|
features:
|
||||||
|
- |
|
||||||
|
Added a new command, ``quota delete``, that will allow admins delete quotas
|
||||||
|
set for projects. Supported by the compute, volume, and network services.
|
@ -49,6 +49,7 @@ openstack.common =
|
|||||||
quota_list = openstackclient.common.quota:ListQuota
|
quota_list = openstackclient.common.quota:ListQuota
|
||||||
quota_set = openstackclient.common.quota:SetQuota
|
quota_set = openstackclient.common.quota:SetQuota
|
||||||
quota_show = openstackclient.common.quota:ShowQuota
|
quota_show = openstackclient.common.quota:ShowQuota
|
||||||
|
quota_delete = openstackclient.common.quota:DeleteQuota
|
||||||
versions_show = openstackclient.common.versions:ShowVersions
|
versions_show = openstackclient.common.versions:ShowVersions
|
||||||
|
|
||||||
openstack.compute.v2 =
|
openstack.compute.v2 =
|
||||||
|
4
tox.ini
4
tox.ini
@ -134,7 +134,7 @@ show-source = True
|
|||||||
# H203: Use assertIs(Not)None to check for None
|
# H203: Use assertIs(Not)None to check for None
|
||||||
enable-extensions = H203
|
enable-extensions = H203
|
||||||
exclude = .venv,.git,.tox,dist,doc,*lib/python*,*egg,build,tools,releasenotes
|
exclude = .venv,.git,.tox,dist,doc,*lib/python*,*egg,build,tools,releasenotes
|
||||||
# W504 is disabled since you must choose between this or W503
|
# W503 and W504 are disabled since they're not very useful
|
||||||
ignore = W504
|
ignore = W503,W504
|
||||||
import-order-style = pep8
|
import-order-style = pep8
|
||||||
application_import_names = openstackclient
|
application_import_names = openstackclient
|
||||||
|
Loading…
x
Reference in New Issue
Block a user