Add quota commands
* Add quota set and quota show commands; these work on both the compute and volume APIs * Add the --class variation on the above commands Note: this replaces the existing volume-only quota commands and eliminates quota list Blueprint: cinder-client Bug: 1172064 Change-Id: I766d40e410e48f05e36e17e567a4f01a9411b40e
This commit is contained in:
parent
aa81b8b8d5
commit
87104a28d7
192
openstackclient/common/quota.py
Normal file
192
openstackclient/common/quota.py
Normal file
@ -0,0 +1,192 @@
|
|||||||
|
# Copyright 2012 OpenStack Foundation
|
||||||
|
#
|
||||||
|
# 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.
|
||||||
|
#
|
||||||
|
|
||||||
|
"""Quota action implementations"""
|
||||||
|
|
||||||
|
import itertools
|
||||||
|
import logging
|
||||||
|
import six
|
||||||
|
import sys
|
||||||
|
|
||||||
|
from cliff import command
|
||||||
|
from cliff import show
|
||||||
|
|
||||||
|
|
||||||
|
# List the quota items, map the internal argument name to the option
|
||||||
|
# name that the user sees.
|
||||||
|
|
||||||
|
COMPUTE_QUOTAS = {
|
||||||
|
'cores': 'cores',
|
||||||
|
'fixed_ips': 'fixed-ips',
|
||||||
|
'floating_ips': 'floating-ips',
|
||||||
|
'injected_file_content_bytes': 'injected-file-size',
|
||||||
|
'injected_file_path_bytes': 'injected-path-size',
|
||||||
|
'injected_files': 'injected-files',
|
||||||
|
'instances': 'instances',
|
||||||
|
'key_pairs': 'key-pairs',
|
||||||
|
'metadata_items': 'properties',
|
||||||
|
'ram': 'ram',
|
||||||
|
'security_group_rules': 'secgroup-rules',
|
||||||
|
'security_groups': 'secgroups',
|
||||||
|
}
|
||||||
|
|
||||||
|
VOLUME_QUOTAS = {
|
||||||
|
'gigabytes': 'gigabytes',
|
||||||
|
'snapshots': 'snapshots',
|
||||||
|
'volumes': 'volumes',
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
class SetQuota(command.Command):
|
||||||
|
"""Set quotas for project or class"""
|
||||||
|
|
||||||
|
log = logging.getLogger(__name__ + '.SetQuota')
|
||||||
|
|
||||||
|
def get_parser(self, prog_name):
|
||||||
|
parser = super(SetQuota, self).get_parser(prog_name)
|
||||||
|
parser.add_argument(
|
||||||
|
'project',
|
||||||
|
metavar='<project/class>',
|
||||||
|
help='Set quotas for this project or class (name/ID)',
|
||||||
|
)
|
||||||
|
parser.add_argument(
|
||||||
|
'--class',
|
||||||
|
dest='quota_class',
|
||||||
|
action='store_true',
|
||||||
|
default=False,
|
||||||
|
help='Set quotas for <class>',
|
||||||
|
)
|
||||||
|
for k, v in itertools.chain(
|
||||||
|
COMPUTE_QUOTAS.items(), VOLUME_QUOTAS.items()):
|
||||||
|
parser.add_argument(
|
||||||
|
'--%s' % v,
|
||||||
|
metavar='<new-%s>' % v,
|
||||||
|
type=int,
|
||||||
|
help='New value for the %s quota' % v,
|
||||||
|
)
|
||||||
|
return parser
|
||||||
|
|
||||||
|
def take_action(self, parsed_args):
|
||||||
|
self.log.debug('take_action(%s)' % parsed_args)
|
||||||
|
|
||||||
|
compute_client = self.app.client_manager.compute
|
||||||
|
volume_client = self.app.client_manager.volume
|
||||||
|
|
||||||
|
compute_kwargs = {}
|
||||||
|
for k, v in COMPUTE_QUOTAS.items():
|
||||||
|
if v in parsed_args:
|
||||||
|
compute_kwargs[k] = getattr(parsed_args, v, None)
|
||||||
|
|
||||||
|
volume_kwargs = {}
|
||||||
|
for k, v in VOLUME_QUOTAS.items():
|
||||||
|
if v in parsed_args:
|
||||||
|
volume_kwargs[k] = getattr(parsed_args, v, None)
|
||||||
|
|
||||||
|
if compute_kwargs == {} and volume_kwargs == {}:
|
||||||
|
sys.stderr.write("No quotas updated")
|
||||||
|
return
|
||||||
|
|
||||||
|
if parsed_args.quota_class:
|
||||||
|
if compute_kwargs:
|
||||||
|
compute_client.quota_classes.update(
|
||||||
|
parsed_args.project,
|
||||||
|
**compute_kwargs)
|
||||||
|
if volume_kwargs:
|
||||||
|
volume_client.quota_classes.update(
|
||||||
|
parsed_args.project,
|
||||||
|
**volume_kwargs)
|
||||||
|
else:
|
||||||
|
if compute_kwargs:
|
||||||
|
compute_client.quotas.update(
|
||||||
|
parsed_args.project,
|
||||||
|
**compute_kwargs)
|
||||||
|
if volume_kwargs:
|
||||||
|
volume_client.quotas.update(
|
||||||
|
parsed_args.project,
|
||||||
|
**volume_kwargs)
|
||||||
|
|
||||||
|
|
||||||
|
class ShowQuota(show.ShowOne):
|
||||||
|
"""Show quotas for project or class"""
|
||||||
|
|
||||||
|
log = logging.getLogger(__name__ + '.ShowQuota')
|
||||||
|
|
||||||
|
def get_parser(self, prog_name):
|
||||||
|
parser = super(ShowQuota, self).get_parser(prog_name)
|
||||||
|
parser.add_argument(
|
||||||
|
'project',
|
||||||
|
metavar='<project/class>',
|
||||||
|
help='Show this project or class (name/ID)',
|
||||||
|
)
|
||||||
|
type_group = parser.add_mutually_exclusive_group()
|
||||||
|
type_group.add_argument(
|
||||||
|
'--class',
|
||||||
|
dest='quota_class',
|
||||||
|
action='store_true',
|
||||||
|
default=False,
|
||||||
|
help='Show quotas for <class>',
|
||||||
|
)
|
||||||
|
type_group.add_argument(
|
||||||
|
'--default',
|
||||||
|
dest='default',
|
||||||
|
action='store_true',
|
||||||
|
default=False,
|
||||||
|
help='Show default quotas for <project>'
|
||||||
|
)
|
||||||
|
return parser
|
||||||
|
|
||||||
|
def take_action(self, parsed_args):
|
||||||
|
self.log.debug('take_action(%s)' % parsed_args)
|
||||||
|
|
||||||
|
compute_client = self.app.client_manager.compute
|
||||||
|
volume_client = self.app.client_manager.volume
|
||||||
|
|
||||||
|
# NOTE(dtroyer): These quota API calls do not validate the project
|
||||||
|
# or class arguments and return what appears to be
|
||||||
|
# the default quota values if the project or class
|
||||||
|
# does not exist. If this is determined to be the
|
||||||
|
# intended behaviour of the API we will validate
|
||||||
|
# the argument with Identity ourselves later.
|
||||||
|
if parsed_args.quota_class:
|
||||||
|
compute_quota = compute_client.quota_classes.get(
|
||||||
|
parsed_args.project)
|
||||||
|
volume_quota = volume_client.quota_classes.get(
|
||||||
|
parsed_args.project)
|
||||||
|
elif parsed_args.default:
|
||||||
|
compute_quota = compute_client.quotas.defaults(
|
||||||
|
parsed_args.project)
|
||||||
|
volume_quota = volume_client.quotas.defaults(
|
||||||
|
parsed_args.project)
|
||||||
|
else:
|
||||||
|
compute_quota = compute_client.quotas.get(parsed_args.project)
|
||||||
|
volume_quota = volume_client.quotas.get(parsed_args.project)
|
||||||
|
|
||||||
|
info = {}
|
||||||
|
info.update(compute_quota._info)
|
||||||
|
info.update(volume_quota._info)
|
||||||
|
|
||||||
|
# Map the internal quota names to the external ones
|
||||||
|
for k, v in itertools.chain(
|
||||||
|
COMPUTE_QUOTAS.items(), VOLUME_QUOTAS.items()):
|
||||||
|
if not k == v and info[k]:
|
||||||
|
info[v] = info[k]
|
||||||
|
info.pop(k)
|
||||||
|
|
||||||
|
# Handle project ID special as it only appears in output
|
||||||
|
if info['id']:
|
||||||
|
info['project'] = info['id']
|
||||||
|
info.pop('id')
|
||||||
|
|
||||||
|
return zip(*sorted(six.iteritems(info)))
|
@ -1,114 +0,0 @@
|
|||||||
# Copyright 2012-2013 OpenStack, LLC.
|
|
||||||
#
|
|
||||||
# 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.
|
|
||||||
#
|
|
||||||
|
|
||||||
"""Volume v1 Quota action implementations"""
|
|
||||||
|
|
||||||
import logging
|
|
||||||
import sys
|
|
||||||
|
|
||||||
from cliff import command
|
|
||||||
from cliff import show
|
|
||||||
|
|
||||||
from openstackclient.common import utils
|
|
||||||
|
|
||||||
|
|
||||||
class ListQuota(show.ShowOne):
|
|
||||||
"""List quota command"""
|
|
||||||
|
|
||||||
api = 'volume'
|
|
||||||
log = logging.getLogger(__name__ + '.ListQuota')
|
|
||||||
|
|
||||||
def get_parser(self, prog_name):
|
|
||||||
parser = super(ListQuota, self).get_parser(prog_name)
|
|
||||||
parser.add_argument(
|
|
||||||
'tenant',
|
|
||||||
metavar='<tenant>',
|
|
||||||
help='ID of tenant to list the default quotas for')
|
|
||||||
return parser
|
|
||||||
|
|
||||||
def take_action(self, parsed_args):
|
|
||||||
self.log.debug('take_action(%s)' % parsed_args)
|
|
||||||
volume_client = self.app.client_manager.volume
|
|
||||||
defaults = volume_client.quotas.defaults(parsed_args.tenant)
|
|
||||||
|
|
||||||
return zip(*sorted(defaults._info.iteritems()))
|
|
||||||
|
|
||||||
|
|
||||||
class SetQuota(command.Command):
|
|
||||||
"""Set quota command"""
|
|
||||||
|
|
||||||
api = 'volume'
|
|
||||||
log = logging.getLogger(__name__ + '.SetQuota')
|
|
||||||
|
|
||||||
def get_parser(self, prog_name):
|
|
||||||
parser = super(SetQuota, self).get_parser(prog_name)
|
|
||||||
parser.add_argument(
|
|
||||||
'tenant',
|
|
||||||
metavar='<tenant>',
|
|
||||||
help='ID of tenant to set the quotas for')
|
|
||||||
parser.add_argument(
|
|
||||||
'--volumes',
|
|
||||||
metavar='<new-volumes>',
|
|
||||||
type=int,
|
|
||||||
help='New value for the volumes quota')
|
|
||||||
parser.add_argument(
|
|
||||||
'--gigabytes',
|
|
||||||
metavar='<new-gigabytes>',
|
|
||||||
type=int,
|
|
||||||
help='New value for the gigabytes quota')
|
|
||||||
return parser
|
|
||||||
|
|
||||||
def take_action(self, parsed_args):
|
|
||||||
self.log.debug('take_action(%s)' % parsed_args)
|
|
||||||
|
|
||||||
kwargs = {}
|
|
||||||
if parsed_args.volumes:
|
|
||||||
kwargs['volumes'] = parsed_args.volumes
|
|
||||||
if parsed_args.gigabytes:
|
|
||||||
kwargs['gigabytes'] = parsed_args.gigabytes
|
|
||||||
|
|
||||||
if kwargs == {}:
|
|
||||||
sys.stdout.write("Quota not updated, no arguments present")
|
|
||||||
return
|
|
||||||
|
|
||||||
volume_client = self.app.client_manager.volume
|
|
||||||
volume_client.quotas.update(parsed_args.tenant,
|
|
||||||
parsed_args.volumes,
|
|
||||||
parsed_args.gigabytes)
|
|
||||||
|
|
||||||
return
|
|
||||||
|
|
||||||
|
|
||||||
class ShowQuota(show.ShowOne):
|
|
||||||
"""Show quota command"""
|
|
||||||
|
|
||||||
api = 'volume'
|
|
||||||
log = logging.getLogger(__name__ + '.ShowQuota')
|
|
||||||
|
|
||||||
def get_parser(self, prog_name):
|
|
||||||
parser = super(ShowQuota, self).get_parser(prog_name)
|
|
||||||
parser.add_argument(
|
|
||||||
'tenant',
|
|
||||||
metavar='<tenant>',
|
|
||||||
help='ID of tenant to list the quotas for')
|
|
||||||
return parser
|
|
||||||
|
|
||||||
def take_action(self, parsed_args):
|
|
||||||
self.log.debug('take_action(%s)' % parsed_args)
|
|
||||||
volume_client = self.app.client_manager.volume
|
|
||||||
quota = utils.find_resource(volume_client.quotas,
|
|
||||||
parsed_args.tenant)
|
|
||||||
|
|
||||||
return zip(*sorted(quota._info.iteritems()))
|
|
@ -33,6 +33,8 @@ openstack.cli =
|
|||||||
|
|
||||||
openstack.common =
|
openstack.common =
|
||||||
limits_show = openstackclient.common.limits:ShowLimits
|
limits_show = openstackclient.common.limits:ShowLimits
|
||||||
|
quota_set = openstackclient.common.quota:SetQuota
|
||||||
|
quota_show = openstackclient.common.quota:ShowQuota
|
||||||
|
|
||||||
openstack.identity.v2_0 =
|
openstack.identity.v2_0 =
|
||||||
ec2_credentials_create = openstackclient.identity.v2_0.ec2creds:CreateEC2Creds
|
ec2_credentials_create = openstackclient.identity.v2_0.ec2creds:CreateEC2Creds
|
||||||
@ -203,10 +205,6 @@ openstack.compute.v2 =
|
|||||||
server_unpause = openstackclient.compute.v2.server:UnpauseServer
|
server_unpause = openstackclient.compute.v2.server:UnpauseServer
|
||||||
|
|
||||||
openstack.volume.v1 =
|
openstack.volume.v1 =
|
||||||
quota_list = openstackclient.volume.v1.quota:ListQuota
|
|
||||||
quota_set = openstackclient.volume.v1.quota:SetQuota
|
|
||||||
quota_show = openstackclient.volume.v1.quota:ShowQuota
|
|
||||||
|
|
||||||
snapshot_create = openstackclient.volume.v1.snapshot:CreateSnapshot
|
snapshot_create = openstackclient.volume.v1.snapshot:CreateSnapshot
|
||||||
snapshot_delete = openstackclient.volume.v1.snapshot:DeleteSnapshot
|
snapshot_delete = openstackclient.volume.v1.snapshot:DeleteSnapshot
|
||||||
snapshot_list = openstackclient.volume.v1.snapshot:ListSnapshot
|
snapshot_list = openstackclient.volume.v1.snapshot:ListSnapshot
|
||||||
|
Loading…
x
Reference in New Issue
Block a user