666 lines
		
	
	
		
			23 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
			
		
		
	
	
			666 lines
		
	
	
		
			23 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
| #   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 sys
 | |
| 
 | |
| from osc_lib.command import command
 | |
| from osc_lib import utils
 | |
| 
 | |
| from openstackclient.i18n import _
 | |
| from openstackclient.network import common
 | |
| 
 | |
| 
 | |
| LOG = logging.getLogger(__name__)
 | |
| 
 | |
| # 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',
 | |
|     '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',
 | |
|     'server_groups': 'server-groups',
 | |
|     'server_group_members': 'server-group-members',
 | |
| }
 | |
| 
 | |
| VOLUME_QUOTAS = {
 | |
|     'backups': 'backups',
 | |
|     'backup_gigabytes': 'backup-gigabytes',
 | |
|     'gigabytes': 'gigabytes',
 | |
|     'per_volume_gigabytes': 'per-volume-gigabytes',
 | |
|     'snapshots': 'snapshots',
 | |
|     'volumes': 'volumes',
 | |
| }
 | |
| 
 | |
| IMPACT_VOLUME_TYPE_QUOTAS = [
 | |
|     'gigabytes',
 | |
|     'snapshots',
 | |
|     'volumes',
 | |
| ]
 | |
| 
 | |
| NOVA_NETWORK_QUOTAS = {
 | |
|     'floating_ips': 'floating-ips',
 | |
|     'security_group_rules': 'secgroup-rules',
 | |
|     'security_groups': 'secgroups',
 | |
| }
 | |
| 
 | |
| NETWORK_QUOTAS = {
 | |
|     'floatingip': 'floating-ips',
 | |
|     'security_group_rule': 'secgroup-rules',
 | |
|     'security_group': 'secgroups',
 | |
|     'network': 'networks',
 | |
|     'subnet': 'subnets',
 | |
|     'port': 'ports',
 | |
|     'router': 'routers',
 | |
|     'rbac_policy': 'rbac-policies',
 | |
|     'subnetpool': 'subnetpools',
 | |
| }
 | |
| 
 | |
| NETWORK_KEYS = ['floating_ips', 'networks', 'rbac_policies', 'routers',
 | |
|                 'ports', 'security_group_rules', 'security_groups',
 | |
|                 'subnet_pools', 'subnets']
 | |
| 
 | |
| 
 | |
| def _xform_get_quota(data, value, keys):
 | |
|     res = []
 | |
|     res_info = {}
 | |
|     for key in keys:
 | |
|         res_info[key] = getattr(data, key, '')
 | |
| 
 | |
|     res_info['id'] = value
 | |
|     res.append(res_info)
 | |
|     return res
 | |
| 
 | |
| 
 | |
| class BaseQuota(object):
 | |
|     def _get_project(self, parsed_args):
 | |
|         if parsed_args.project is not None:
 | |
|             identity_client = self.app.client_manager.identity
 | |
|             project = utils.find_resource(
 | |
|                 identity_client.projects,
 | |
|                 parsed_args.project,
 | |
|             )
 | |
|             project_id = project.id
 | |
|             project_name = project.name
 | |
|         elif self.app.client_manager.auth_ref:
 | |
|             # Get the project from the current auth
 | |
|             project = self.app.client_manager.auth_ref
 | |
|             project_id = project.project_id
 | |
|             project_name = project.project_name
 | |
|         else:
 | |
|             project = None
 | |
|             project_id = None
 | |
|             project_name = None
 | |
|         project_info = {}
 | |
|         project_info['id'] = project_id
 | |
|         project_info['name'] = project_name
 | |
|         return project_info
 | |
| 
 | |
|     def get_compute_quota(self, client, parsed_args):
 | |
|         quota_class = (
 | |
|             parsed_args.quota_class if 'quota_class' in parsed_args else False)
 | |
|         detail = parsed_args.detail if 'detail' in parsed_args else False
 | |
|         default = parsed_args.default if 'default' in parsed_args else False
 | |
|         try:
 | |
|             if quota_class:
 | |
|                 quota = client.quota_classes.get(parsed_args.project)
 | |
|             else:
 | |
|                 project_info = self._get_project(parsed_args)
 | |
|                 project = project_info['id']
 | |
|                 if default:
 | |
|                     quota = client.quotas.defaults(project)
 | |
|                 else:
 | |
|                     quota = client.quotas.get(project, detail=detail)
 | |
|         except Exception as e:
 | |
|             if type(e).__name__ == 'EndpointNotFound':
 | |
|                 return {}
 | |
|             else:
 | |
|                 raise
 | |
|         return quota._info
 | |
| 
 | |
|     def get_volume_quota(self, client, parsed_args):
 | |
|         quota_class = (
 | |
|             parsed_args.quota_class if 'quota_class' in parsed_args else False)
 | |
|         default = parsed_args.default if 'default' in parsed_args else False
 | |
|         try:
 | |
|             if quota_class:
 | |
|                 quota = client.quota_classes.get(parsed_args.project)
 | |
|             else:
 | |
|                 project_info = self._get_project(parsed_args)
 | |
|                 project = project_info['id']
 | |
|                 if default:
 | |
|                     quota = client.quotas.defaults(project)
 | |
|                 else:
 | |
|                     quota = client.quotas.get(project)
 | |
|         except Exception as e:
 | |
|             if type(e).__name__ == 'EndpointNotFound':
 | |
|                 return {}
 | |
|             else:
 | |
|                 raise
 | |
|         return quota._info
 | |
| 
 | |
|     def get_network_quota(self, parsed_args):
 | |
|         quota_class = (
 | |
|             parsed_args.quota_class if 'quota_class' in parsed_args else False)
 | |
|         detail = parsed_args.detail if 'detail' in parsed_args else False
 | |
|         default = parsed_args.default if 'default' in parsed_args else False
 | |
|         if quota_class:
 | |
|             return {}
 | |
|         if self.app.client_manager.is_network_endpoint_enabled():
 | |
|             project_info = self._get_project(parsed_args)
 | |
|             project = project_info['id']
 | |
|             client = self.app.client_manager.network
 | |
|             if default:
 | |
|                 network_quota = client.get_quota_default(project)
 | |
|                 if type(network_quota) is not dict:
 | |
|                     network_quota = network_quota.to_dict()
 | |
|             else:
 | |
|                 network_quota = client.get_quota(project,
 | |
|                                                  details=detail)
 | |
|                 if type(network_quota) is not dict:
 | |
|                     network_quota = network_quota.to_dict()
 | |
|                 if detail:
 | |
|                     # NOTE(slaweq): Neutron returns values with key "used" but
 | |
|                     # Nova for example returns same data with key "in_use"
 | |
|                     # instead.
 | |
|                     # Because of that we need to convert Neutron key to
 | |
|                     # the same as is returned from Nova to make result
 | |
|                     # more consistent
 | |
|                     for key, values in network_quota.items():
 | |
|                         if type(values) is dict and "used" in values:
 | |
|                             values[u'in_use'] = values.pop("used")
 | |
|                         network_quota[key] = values
 | |
|             return network_quota
 | |
|         else:
 | |
|             return {}
 | |
| 
 | |
| 
 | |
| class ListQuota(command.Lister, BaseQuota):
 | |
|     _description = _(
 | |
|         "List quotas for all projects with non-default quota values or "
 | |
|         "list detailed quota informations for requested project")
 | |
| 
 | |
|     def _get_detailed_quotas(self, parsed_args):
 | |
|         columns = (
 | |
|             'resource',
 | |
|             'in_use',
 | |
|             'reserved',
 | |
|             'limit'
 | |
|         )
 | |
|         column_headers = (
 | |
|             'Resource',
 | |
|             'In Use',
 | |
|             'Reserved',
 | |
|             'Limit'
 | |
|         )
 | |
|         quotas = {}
 | |
|         if parsed_args.compute:
 | |
|             quotas.update(self.get_compute_quota(
 | |
|                 self.app.client_manager.compute, parsed_args))
 | |
|         if parsed_args.network:
 | |
|             quotas.update(self.get_network_quota(parsed_args))
 | |
| 
 | |
|         result = []
 | |
|         for resource, values in quotas.items():
 | |
|             # NOTE(slaweq): there is no detailed quotas info for some resources
 | |
|             # and it should't be displayed here
 | |
|             if type(values) is dict:
 | |
|                 result.append({
 | |
|                     'resource': resource,
 | |
|                     'in_use': values.get('in_use'),
 | |
|                     'reserved': values.get('reserved'),
 | |
|                     'limit': values.get('limit')
 | |
|                 })
 | |
|         return (column_headers,
 | |
|                 (utils.get_dict_properties(
 | |
|                     s, columns,
 | |
|                 ) for s in result))
 | |
| 
 | |
|     def get_parser(self, prog_name):
 | |
|         parser = super(ListQuota, self).get_parser(prog_name)
 | |
|         parser.add_argument(
 | |
|             '--project',
 | |
|             metavar='<project>',
 | |
|             help=_('List quotas for this project <project> (name or ID)'),
 | |
|         )
 | |
|         parser.add_argument(
 | |
|             '--detail',
 | |
|             dest='detail',
 | |
|             action='store_true',
 | |
|             default=False,
 | |
|             help=_('Show details about quotas usage')
 | |
|         )
 | |
|         option = parser.add_mutually_exclusive_group(required=True)
 | |
|         option.add_argument(
 | |
|             '--compute',
 | |
|             action='store_true',
 | |
|             default=False,
 | |
|             help=_('List compute quota'),
 | |
|         )
 | |
|         option.add_argument(
 | |
|             '--volume',
 | |
|             action='store_true',
 | |
|             default=False,
 | |
|             help=_('List volume quota'),
 | |
|         )
 | |
|         option.add_argument(
 | |
|             '--network',
 | |
|             action='store_true',
 | |
|             default=False,
 | |
|             help=_('List network quota'),
 | |
|         )
 | |
|         return parser
 | |
| 
 | |
|     def take_action(self, parsed_args):
 | |
|         projects = self.app.client_manager.identity.projects.list()
 | |
|         result = []
 | |
|         project_ids = [getattr(p, 'id', '') for p in projects]
 | |
| 
 | |
|         if parsed_args.compute:
 | |
|             if parsed_args.detail:
 | |
|                 return self._get_detailed_quotas(parsed_args)
 | |
|             compute_client = self.app.client_manager.compute
 | |
|             for p in project_ids:
 | |
|                 try:
 | |
|                     data = compute_client.quotas.get(p)
 | |
|                 except Exception as ex:
 | |
|                     if (
 | |
|                         type(ex).__name__ == 'NotFound' or
 | |
|                         ex.http_status >= 400 and ex.http_status <= 499
 | |
|                     ):
 | |
|                         # Project not found, move on to next one
 | |
|                         LOG.warning("Project %s not found: %s" % (p, ex))
 | |
|                         continue
 | |
|                     else:
 | |
|                         raise
 | |
| 
 | |
|                 result_data = _xform_get_quota(
 | |
|                     data,
 | |
|                     p,
 | |
|                     COMPUTE_QUOTAS.keys(),
 | |
|                 )
 | |
|                 default_data = compute_client.quotas.defaults(p)
 | |
|                 result_default = _xform_get_quota(
 | |
|                     default_data,
 | |
|                     p,
 | |
|                     COMPUTE_QUOTAS.keys(),
 | |
|                 )
 | |
|                 if result_default != result_data:
 | |
|                     result += result_data
 | |
| 
 | |
|             columns = (
 | |
|                 'id',
 | |
|                 'cores',
 | |
|                 'fixed_ips',
 | |
|                 'injected_files',
 | |
|                 'injected_file_content_bytes',
 | |
|                 'injected_file_path_bytes',
 | |
|                 'instances',
 | |
|                 'key_pairs',
 | |
|                 'metadata_items',
 | |
|                 'ram',
 | |
|                 'server_groups',
 | |
|                 'server_group_members',
 | |
|             )
 | |
|             column_headers = (
 | |
|                 'Project ID',
 | |
|                 'Cores',
 | |
|                 'Fixed IPs',
 | |
|                 'Injected Files',
 | |
|                 'Injected File Content Bytes',
 | |
|                 'Injected File Path Bytes',
 | |
|                 'Instances',
 | |
|                 'Key Pairs',
 | |
|                 'Metadata Items',
 | |
|                 'Ram',
 | |
|                 'Server Groups',
 | |
|                 'Server Group Members',
 | |
|             )
 | |
|             return (column_headers,
 | |
|                     (utils.get_dict_properties(
 | |
|                         s, columns,
 | |
|                     ) for s in result))
 | |
| 
 | |
|         if parsed_args.volume:
 | |
|             if parsed_args.detail:
 | |
|                 LOG.warning("Volume service doesn't provide detailed quota"
 | |
|                             " information")
 | |
|             volume_client = self.app.client_manager.volume
 | |
|             for p in project_ids:
 | |
|                 try:
 | |
|                     data = volume_client.quotas.get(p)
 | |
|                 except Exception as ex:
 | |
|                     if type(ex).__name__ == 'NotFound':
 | |
|                         # Project not found, move on to next one
 | |
|                         LOG.warning("Project %s not found: %s" % (p, ex))
 | |
|                         continue
 | |
|                     else:
 | |
|                         raise
 | |
| 
 | |
|                 result_data = _xform_get_quota(
 | |
|                     data,
 | |
|                     p,
 | |
|                     VOLUME_QUOTAS.keys(),
 | |
|                 )
 | |
|                 default_data = volume_client.quotas.defaults(p)
 | |
|                 result_default = _xform_get_quota(
 | |
|                     default_data,
 | |
|                     p,
 | |
|                     VOLUME_QUOTAS.keys(),
 | |
|                 )
 | |
|                 if result_default != result_data:
 | |
|                     result += result_data
 | |
| 
 | |
|             columns = (
 | |
|                 'id',
 | |
|                 'backups',
 | |
|                 'backup_gigabytes',
 | |
|                 'gigabytes',
 | |
|                 'per_volume_gigabytes',
 | |
|                 'snapshots',
 | |
|                 'volumes',
 | |
|             )
 | |
|             column_headers = (
 | |
|                 'Project ID',
 | |
|                 'Backups',
 | |
|                 'Backup Gigabytes',
 | |
|                 'Gigabytes',
 | |
|                 'Per Volume Gigabytes',
 | |
|                 'Snapshots',
 | |
|                 'Volumes',
 | |
|             )
 | |
|             return (column_headers,
 | |
|                     (utils.get_dict_properties(
 | |
|                         s, columns,
 | |
|                     ) for s in result))
 | |
| 
 | |
|         if parsed_args.network:
 | |
|             if parsed_args.detail:
 | |
|                 return self._get_detailed_quotas(parsed_args)
 | |
|             client = self.app.client_manager.network
 | |
|             for p in project_ids:
 | |
|                 try:
 | |
|                     data = client.get_quota(p)
 | |
|                 except Exception as ex:
 | |
|                     if type(ex).__name__ == 'NotFound':
 | |
|                         # Project not found, move on to next one
 | |
|                         LOG.warning("Project %s not found: %s" % (p, ex))
 | |
|                         continue
 | |
|                     else:
 | |
|                         raise
 | |
| 
 | |
|                 result_data = _xform_get_quota(
 | |
|                     data,
 | |
|                     p,
 | |
|                     NETWORK_KEYS,
 | |
|                 )
 | |
|                 default_data = client.get_quota_default(p)
 | |
|                 result_default = _xform_get_quota(
 | |
|                     default_data,
 | |
|                     p,
 | |
|                     NETWORK_KEYS,
 | |
|                 )
 | |
|                 if result_default != result_data:
 | |
|                     result += result_data
 | |
| 
 | |
|             columns = (
 | |
|                 'id',
 | |
|                 'floating_ips',
 | |
|                 'networks',
 | |
|                 'ports',
 | |
|                 'rbac_policies',
 | |
|                 'routers',
 | |
|                 'security_groups',
 | |
|                 'security_group_rules',
 | |
|                 'subnets',
 | |
|                 'subnet_pools',
 | |
|             )
 | |
|             column_headers = (
 | |
|                 'Project ID',
 | |
|                 'Floating IPs',
 | |
|                 'Networks',
 | |
|                 'Ports',
 | |
|                 'RBAC Policies',
 | |
|                 'Routers',
 | |
|                 'Security Groups',
 | |
|                 'Security Group Rules',
 | |
|                 'Subnets',
 | |
|                 'Subnet Pools'
 | |
|             )
 | |
|             return (column_headers,
 | |
|                     (utils.get_dict_properties(
 | |
|                         s, columns,
 | |
|                     ) for s in result))
 | |
| 
 | |
|         return ((), ())
 | |
| 
 | |
| 
 | |
| class SetQuota(common.NetDetectionMixin, command.Command):
 | |
|     _description = _("Set quotas for project or class")
 | |
| 
 | |
|     def _build_options_list(self):
 | |
|         help_fmt = _('New value for the %s quota')
 | |
|         # Compute and volume quota options are always the same
 | |
|         rets = [(k, v, help_fmt % v) for k, v in itertools.chain(
 | |
|             COMPUTE_QUOTAS.items(),
 | |
|             VOLUME_QUOTAS.items(),
 | |
|         )]
 | |
|         # For docs build, we want to produce helps for both neutron and
 | |
|         # nova-network options. They overlap, so we have to figure out which
 | |
|         # need to be tagged as specific to one network type or the other.
 | |
|         if self.is_docs_build:
 | |
|             # NOTE(efried): This takes advantage of the fact that we know the
 | |
|             # nova-net options are a subset of the neutron options. If that
 | |
|             # ever changes, this algorithm will need to be adjusted accordingly
 | |
|             inv_compute = set(NOVA_NETWORK_QUOTAS.values())
 | |
|             for k, v in NETWORK_QUOTAS.items():
 | |
|                 _help = help_fmt % v
 | |
|                 if v not in inv_compute:
 | |
|                     # This one is unique to neutron
 | |
|                     _help = self.enhance_help_neutron(_help)
 | |
|                 rets.append((k, v, _help))
 | |
|         elif self.is_neutron:
 | |
|             rets.extend(
 | |
|                 [(k, v, help_fmt % v) for k, v in NETWORK_QUOTAS.items()])
 | |
|         elif self.is_nova_network:
 | |
|             rets.extend(
 | |
|                 [(k, v, help_fmt % v) for k, v in NOVA_NETWORK_QUOTAS.items()])
 | |
|         return rets
 | |
| 
 | |
|     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, h in self._build_options_list():
 | |
|             parser.add_argument(
 | |
|                 '--%s' % v,
 | |
|                 metavar='<%s>' % v,
 | |
|                 dest=k,
 | |
|                 type=int,
 | |
|                 help=h,
 | |
|             )
 | |
|         parser.add_argument(
 | |
|             '--volume-type',
 | |
|             metavar='<volume-type>',
 | |
|             help=_('Set quotas for a specific <volume-type>'),
 | |
|         )
 | |
|         return parser
 | |
| 
 | |
|     def take_action(self, parsed_args):
 | |
| 
 | |
|         identity_client = self.app.client_manager.identity
 | |
|         compute_client = self.app.client_manager.compute
 | |
|         volume_client = self.app.client_manager.volume
 | |
|         compute_kwargs = {}
 | |
|         for k, v in COMPUTE_QUOTAS.items():
 | |
|             value = getattr(parsed_args, k, None)
 | |
|             if value is not None:
 | |
|                 compute_kwargs[k] = value
 | |
| 
 | |
|         volume_kwargs = {}
 | |
|         for k, v in VOLUME_QUOTAS.items():
 | |
|             value = getattr(parsed_args, k, None)
 | |
|             if value is not None:
 | |
|                 if (parsed_args.volume_type and
 | |
|                         k in IMPACT_VOLUME_TYPE_QUOTAS):
 | |
|                     k = k + '_%s' % parsed_args.volume_type
 | |
|                 volume_kwargs[k] = value
 | |
| 
 | |
|         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 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)
 | |
|             if network_kwargs:
 | |
|                 sys.stderr.write("Network quotas are ignored since quota class"
 | |
|                                  " is not supported.")
 | |
|         else:
 | |
|             project = utils.find_resource(
 | |
|                 identity_client.projects,
 | |
|                 parsed_args.project,
 | |
|             ).id
 | |
|             if compute_kwargs:
 | |
|                 compute_client.quotas.update(
 | |
|                     project,
 | |
|                     **compute_kwargs)
 | |
|             if volume_kwargs:
 | |
|                 volume_client.quotas.update(
 | |
|                     project,
 | |
|                     **volume_kwargs)
 | |
|             if (
 | |
|                     network_kwargs and
 | |
|                     self.app.client_manager.is_network_endpoint_enabled()
 | |
|             ):
 | |
|                 network_client = self.app.client_manager.network
 | |
|                 network_client.update_quota(
 | |
|                     project,
 | |
|                     **network_kwargs)
 | |
| 
 | |
| 
 | |
| class ShowQuota(command.ShowOne, BaseQuota):
 | |
|     _description = _(
 | |
|         "Show quotas for project or class. Specify "
 | |
|         "``--os-compute-api-version 2.50`` or higher to see ``server-groups`` "
 | |
|         "and ``server-group-members`` output for a given quota class.")
 | |
| 
 | |
|     def get_parser(self, prog_name):
 | |
|         parser = super(ShowQuota, self).get_parser(prog_name)
 | |
|         parser.add_argument(
 | |
|             'project',
 | |
|             metavar='<project/class>',
 | |
|             nargs='?',
 | |
|             help=_('Show quotas for this project or class (name or 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):
 | |
| 
 | |
|         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.
 | |
|         compute_quota_info = self.get_compute_quota(compute_client,
 | |
|                                                     parsed_args)
 | |
|         volume_quota_info = self.get_volume_quota(volume_client,
 | |
|                                                   parsed_args)
 | |
|         network_quota_info = self.get_network_quota(parsed_args)
 | |
|         # NOTE(reedip): Remove the below check once requirement for
 | |
|         #               Openstack SDK is fixed to version 0.9.12 and above
 | |
|         if type(network_quota_info) is not dict:
 | |
|             network_quota_info = network_quota_info.to_dict()
 | |
| 
 | |
|         info = {}
 | |
|         info.update(compute_quota_info)
 | |
|         info.update(volume_quota_info)
 | |
|         info.update(network_quota_info)
 | |
| 
 | |
|         # Map the internal quota names to the external ones
 | |
|         # COMPUTE_QUOTAS and NETWORK_QUOTAS share floating-ips,
 | |
|         # secgroup-rules and secgroups as dict value, so when
 | |
|         # neutron is enabled, quotas of these three resources
 | |
|         # in nova will be replaced by neutron's.
 | |
|         for k, v in itertools.chain(
 | |
|                 COMPUTE_QUOTAS.items(), NOVA_NETWORK_QUOTAS.items(),
 | |
|                 VOLUME_QUOTAS.items(), NETWORK_QUOTAS.items()):
 | |
|             if not k == v and info.get(k) is not None:
 | |
|                 info[v] = info[k]
 | |
|                 info.pop(k)
 | |
| 
 | |
|         # Handle project ID special as it only appears in output
 | |
|         if 'id' in info:
 | |
|             info['project'] = info.pop('id')
 | |
|             if 'project_id' in info:
 | |
|                 del info['project_id']
 | |
|             project_info = self._get_project(parsed_args)
 | |
|             project_name = project_info['name']
 | |
|             info['project_name'] = project_name
 | |
| 
 | |
|         return zip(*sorted(info.items()))
 | 
