2015-12-14 13:29:43 -06:00
|
|
|
# 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.
|
|
|
|
#
|
|
|
|
|
|
|
|
import abc
|
2017-10-31 15:35:10 +01:00
|
|
|
import contextlib
|
2016-06-16 20:01:15 +08:00
|
|
|
import logging
|
2016-06-08 14:17:14 -05:00
|
|
|
|
2016-11-10 16:19:01 +01:00
|
|
|
import openstack.exceptions
|
2016-05-13 17:27:12 -05:00
|
|
|
from osc_lib.command import command
|
2016-06-08 14:17:14 -05:00
|
|
|
from osc_lib import exceptions
|
2015-12-14 13:29:43 -06:00
|
|
|
import six
|
|
|
|
|
2016-05-31 20:31:28 +08:00
|
|
|
from openstackclient.i18n import _
|
2015-12-14 13:29:43 -06:00
|
|
|
|
|
|
|
|
2016-06-16 20:01:15 +08:00
|
|
|
LOG = logging.getLogger(__name__)
|
|
|
|
|
2017-10-31 15:35:10 +01:00
|
|
|
_required_opt_extensions_map = {
|
|
|
|
'allowed_address_pairs': 'allowed-address-pairs',
|
|
|
|
'dns_domain': 'dns-integration',
|
|
|
|
'dns_name': 'dns-integration',
|
|
|
|
'extra_dhcp_opts': 'extra_dhcp_opt',
|
|
|
|
'qos_policy_id': 'qos',
|
|
|
|
'security_groups': 'security-groups',
|
|
|
|
}
|
|
|
|
|
2019-10-28 17:27:38 -05:00
|
|
|
_NET_TYPE_NEUTRON = 'neutron'
|
|
|
|
_NET_TYPE_COMPUTE = 'nova-network'
|
|
|
|
_QUALIFIER_FMT = "%s\n\n*%s*"
|
|
|
|
|
2017-10-31 15:35:10 +01:00
|
|
|
|
|
|
|
@contextlib.contextmanager
|
|
|
|
def check_missing_extension_if_error(client_manager, attrs):
|
|
|
|
# If specified option requires extension, then try to
|
|
|
|
# find out if it exists. If it does not exist,
|
|
|
|
# then an exception with the appropriate message
|
|
|
|
# will be thrown from within client.find_extension
|
|
|
|
try:
|
|
|
|
yield
|
|
|
|
except openstack.exceptions.HttpException:
|
|
|
|
for opt, ext in _required_opt_extensions_map.items():
|
|
|
|
if opt in attrs:
|
|
|
|
client_manager.find_extension(ext, ignore_missing=False)
|
|
|
|
raise
|
|
|
|
|
2016-06-16 20:01:15 +08:00
|
|
|
|
2015-12-14 13:29:43 -06:00
|
|
|
@six.add_metaclass(abc.ABCMeta)
|
2019-10-28 17:27:38 -05:00
|
|
|
class NetDetectionMixin(object):
|
|
|
|
"""Convenience methods for nova-network vs. neutron decisions.
|
2016-02-05 09:02:27 -06:00
|
|
|
|
2019-10-28 17:27:38 -05:00
|
|
|
A live environment detects which network type it is running and creates its
|
|
|
|
parser with only the options relevant to that network type.
|
2015-12-14 13:29:43 -06:00
|
|
|
|
2019-10-28 17:27:38 -05:00
|
|
|
But the command classes are used for docs builds as well, and docs must
|
|
|
|
present the options for both network types, often qualified accordingly.
|
|
|
|
"""
|
|
|
|
@property
|
|
|
|
def _network_type(self):
|
|
|
|
"""Discover whether the running cloud is using neutron or nova-network.
|
|
|
|
|
|
|
|
:return:
|
|
|
|
* ``NET_TYPE_NEUTRON`` if neutron is detected
|
|
|
|
* ``NET_TYPE_COMPUTE`` if running in a cloud but neutron is not
|
|
|
|
detected.
|
|
|
|
* ``None`` if not running in a cloud, which hopefully means we're
|
|
|
|
building docs.
|
|
|
|
"""
|
|
|
|
# Have we set it up yet for this command?
|
|
|
|
if not hasattr(self, '_net_type'):
|
|
|
|
# import pdb; pdb.set_trace()
|
|
|
|
try:
|
|
|
|
if self.app.client_manager.is_network_endpoint_enabled():
|
|
|
|
net_type = _NET_TYPE_NEUTRON
|
|
|
|
else:
|
|
|
|
net_type = _NET_TYPE_COMPUTE
|
|
|
|
except AttributeError:
|
|
|
|
LOG.warning(
|
|
|
|
"%s: Could not detect a network type. Assuming we are "
|
|
|
|
"building docs.", self.__class__.__name__)
|
|
|
|
net_type = None
|
|
|
|
self._net_type = net_type
|
|
|
|
return self._net_type
|
|
|
|
|
|
|
|
@property
|
|
|
|
def is_neutron(self):
|
|
|
|
return self._network_type is _NET_TYPE_NEUTRON
|
|
|
|
|
|
|
|
@property
|
|
|
|
def is_nova_network(self):
|
|
|
|
return self._network_type is _NET_TYPE_COMPUTE
|
|
|
|
|
|
|
|
@property
|
|
|
|
def is_docs_build(self):
|
|
|
|
return self._network_type is None
|
|
|
|
|
|
|
|
def enhance_help_neutron(self, _help):
|
|
|
|
if self.is_docs_build:
|
|
|
|
# Why can't we say 'neutron'?
|
|
|
|
return _QUALIFIER_FMT % (_help, _("Network version 2 only"))
|
|
|
|
return _help
|
|
|
|
|
|
|
|
def enhance_help_nova_network(self, _help):
|
|
|
|
if self.is_docs_build:
|
|
|
|
# Why can't we say 'nova-network'?
|
|
|
|
return _QUALIFIER_FMT % (_help, _("Compute version 2 only"))
|
|
|
|
return _help
|
|
|
|
|
|
|
|
@staticmethod
|
|
|
|
def split_help(network_help, compute_help):
|
|
|
|
return (
|
|
|
|
"*%(network_qualifier)s:*\n %(network_help)s\n\n"
|
|
|
|
"*%(compute_qualifier)s:*\n %(compute_help)s" % dict(
|
|
|
|
network_qualifier=_("Network version 2"),
|
|
|
|
network_help=network_help,
|
|
|
|
compute_qualifier=_("Compute version 2"),
|
|
|
|
compute_help=compute_help))
|
2015-12-14 13:29:43 -06:00
|
|
|
|
|
|
|
def get_parser(self, prog_name):
|
2016-06-16 20:01:15 +08:00
|
|
|
LOG.debug('get_parser(%s)', prog_name)
|
2019-10-28 17:27:38 -05:00
|
|
|
parser = super(NetDetectionMixin, self).get_parser(prog_name)
|
2015-12-14 13:29:43 -06:00
|
|
|
parser = self.update_parser_common(parser)
|
2016-06-16 20:01:15 +08:00
|
|
|
LOG.debug('common parser: %s', parser)
|
2019-10-28 17:27:38 -05:00
|
|
|
if self.is_neutron or self.is_docs_build:
|
|
|
|
parser = self.update_parser_network(parser)
|
|
|
|
if self.is_nova_network or self.is_docs_build:
|
|
|
|
# Add nova-net options if running nova-network or building docs
|
|
|
|
parser = self.update_parser_compute(parser)
|
|
|
|
return parser
|
2015-12-14 13:29:43 -06:00
|
|
|
|
|
|
|
def update_parser_common(self, parser):
|
|
|
|
"""Default is no updates to parser."""
|
|
|
|
return parser
|
|
|
|
|
|
|
|
def update_parser_network(self, parser):
|
|
|
|
"""Default is no updates to parser."""
|
|
|
|
return parser
|
|
|
|
|
|
|
|
def update_parser_compute(self, parser):
|
|
|
|
"""Default is no updates to parser."""
|
|
|
|
return parser
|
|
|
|
|
2019-10-28 17:27:38 -05:00
|
|
|
def take_action(self, parsed_args):
|
|
|
|
if self.is_neutron:
|
|
|
|
return self.take_action_network(self.app.client_manager.network,
|
|
|
|
parsed_args)
|
|
|
|
elif self.is_nova_network:
|
|
|
|
return self.take_action_compute(self.app.client_manager.compute,
|
|
|
|
parsed_args)
|
|
|
|
|
2015-12-14 13:29:43 -06:00
|
|
|
def take_action_network(self, client, parsed_args):
|
|
|
|
"""Override to do something useful."""
|
|
|
|
pass
|
|
|
|
|
|
|
|
def take_action_compute(self, client, parsed_args):
|
|
|
|
"""Override to do something useful."""
|
|
|
|
pass
|
2016-02-05 13:39:40 -06:00
|
|
|
|
|
|
|
|
2019-10-28 17:27:38 -05:00
|
|
|
@six.add_metaclass(abc.ABCMeta)
|
|
|
|
class NetworkAndComputeCommand(NetDetectionMixin, command.Command):
|
|
|
|
"""Network and Compute Command
|
|
|
|
|
|
|
|
Command class for commands that support implementation via
|
|
|
|
the network or compute endpoint. Such commands have different
|
|
|
|
implementations for take_action() and may even have different
|
|
|
|
arguments.
|
|
|
|
"""
|
|
|
|
pass
|
|
|
|
|
|
|
|
|
2016-03-14 14:03:04 +08:00
|
|
|
@six.add_metaclass(abc.ABCMeta)
|
|
|
|
class NetworkAndComputeDelete(NetworkAndComputeCommand):
|
|
|
|
"""Network and Compute Delete
|
|
|
|
|
|
|
|
Delete class for commands that support implementation via
|
|
|
|
the network or compute endpoint. Such commands have different
|
|
|
|
implementations for take_action() and may even have different
|
|
|
|
arguments. This class supports bulk deletion, and error handling
|
|
|
|
following the rules in doc/source/command-errors.rst.
|
|
|
|
"""
|
|
|
|
|
|
|
|
def take_action(self, parsed_args):
|
|
|
|
ret = 0
|
|
|
|
resources = getattr(parsed_args, self.resource, [])
|
|
|
|
|
|
|
|
for r in resources:
|
|
|
|
self.r = r
|
|
|
|
try:
|
|
|
|
if self.app.client_manager.is_network_endpoint_enabled():
|
|
|
|
self.take_action_network(self.app.client_manager.network,
|
|
|
|
parsed_args)
|
|
|
|
else:
|
|
|
|
self.take_action_compute(self.app.client_manager.compute,
|
|
|
|
parsed_args)
|
|
|
|
except Exception as e:
|
2016-05-31 20:31:28 +08:00
|
|
|
msg = _("Failed to delete %(resource)s with name or ID "
|
|
|
|
"'%(name_or_id)s': %(e)s") % {
|
|
|
|
"resource": self.resource,
|
|
|
|
"name_or_id": r,
|
|
|
|
"e": e,
|
|
|
|
}
|
2016-06-16 20:01:15 +08:00
|
|
|
LOG.error(msg)
|
2016-03-14 14:03:04 +08:00
|
|
|
ret += 1
|
|
|
|
|
|
|
|
if ret:
|
|
|
|
total = len(resources)
|
2016-07-14 18:34:19 +08:00
|
|
|
msg = _("%(num)s of %(total)s %(resource)ss failed to delete.") % {
|
2016-05-31 20:31:28 +08:00
|
|
|
"num": ret,
|
|
|
|
"total": total,
|
|
|
|
"resource": self.resource,
|
|
|
|
}
|
2016-03-14 14:03:04 +08:00
|
|
|
raise exceptions.CommandError(msg)
|
|
|
|
|
|
|
|
|
2016-02-05 13:39:40 -06:00
|
|
|
@six.add_metaclass(abc.ABCMeta)
|
2019-10-28 17:27:38 -05:00
|
|
|
class NetworkAndComputeLister(NetDetectionMixin, command.Lister):
|
2016-02-05 13:39:40 -06:00
|
|
|
"""Network and Compute Lister
|
|
|
|
|
|
|
|
Lister class for commands that support implementation via
|
|
|
|
the network or compute endpoint. Such commands have different
|
|
|
|
implementations for take_action() and may even have different
|
|
|
|
arguments.
|
|
|
|
"""
|
2019-10-28 17:27:38 -05:00
|
|
|
pass
|
2016-02-05 13:39:40 -06:00
|
|
|
|
|
|
|
|
|
|
|
@six.add_metaclass(abc.ABCMeta)
|
2019-10-28 17:27:38 -05:00
|
|
|
class NetworkAndComputeShowOne(NetDetectionMixin, command.ShowOne):
|
2016-02-05 13:39:40 -06:00
|
|
|
"""Network and Compute ShowOne
|
|
|
|
|
|
|
|
ShowOne class for commands that support implementation via
|
|
|
|
the network or compute endpoint. Such commands have different
|
|
|
|
implementations for take_action() and may even have different
|
|
|
|
arguments.
|
|
|
|
"""
|
|
|
|
|
|
|
|
def take_action(self, parsed_args):
|
2016-11-10 16:19:01 +01:00
|
|
|
try:
|
|
|
|
if self.app.client_manager.is_network_endpoint_enabled():
|
|
|
|
return self.take_action_network(
|
|
|
|
self.app.client_manager.network, parsed_args)
|
|
|
|
else:
|
|
|
|
return self.take_action_compute(
|
|
|
|
self.app.client_manager.compute, parsed_args)
|
|
|
|
except openstack.exceptions.HttpException as exc:
|
|
|
|
msg = _("Error while executing command: %s") % exc.message
|
2017-11-14 23:24:55 +00:00
|
|
|
if exc.details:
|
|
|
|
msg += ", " + six.text_type(exc.details)
|
2016-11-10 16:19:01 +01:00
|
|
|
raise exceptions.CommandError(msg)
|