python-openstackclient/openstackclient/compute/v2/server.py
Stephen Finucane 209f8e9e17 network: Replace use of in-tree API client
None of these are actually supported by openstacksdk (intentionally so)
so we add our own manual implementations.

Change-Id: Ifd24f04ae4d1e56e0ce5ba0afe63828403bb7a6f
Signed-off-by: Stephen Finucane <stephenfin@redhat.com>
2024-07-09 18:19:36 +01:00

5242 lines
183 KiB
Python

# Copyright 2012-2013 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.
#
"""Compute v2 Server action implementations"""
import argparse
import base64
import getpass
import json
import logging
import os
import typing as ty
from cliff import columns as cliff_columns
import iso8601
from openstack import exceptions as sdk_exceptions
from openstack import utils as sdk_utils
from osc_lib.cli import format_columns
from osc_lib.cli import parseractions
from osc_lib.command import command
from osc_lib import exceptions
from osc_lib import utils
from openstackclient.api import compute_v2
from openstackclient.common import pagination
from openstackclient.i18n import _
from openstackclient.identity import common as identity_common
from openstackclient.network import common as network_common
LOG = logging.getLogger(__name__)
IMAGE_STRING_FOR_BFV = 'N/A (booted from volume)'
class PowerStateColumn(cliff_columns.FormattableColumn):
"""Generate a formatted string of a server's power state."""
power_states = [
'NOSTATE', # 0x00
'Running', # 0x01
'', # 0x02
'Paused', # 0x03
'Shutdown', # 0x04
'', # 0x05
'Crashed', # 0x06
'Suspended', # 0x07
]
def human_readable(self):
try:
return self.power_states[self._value]
except Exception:
return 'N/A'
class AddressesColumn(cliff_columns.FormattableColumn):
"""Generate a formatted string of a server's addresses."""
def human_readable(self):
try:
return utils.format_dict_of_list(
{
k: [i['addr'] for i in v if 'addr' in i]
for k, v in self._value.items()
}
)
except Exception:
return 'N/A'
def machine_readable(self):
return {
k: [i['addr'] for i in v if 'addr' in i]
for k, v in (self._value.items() if self._value else [])
}
class HostColumn(cliff_columns.FormattableColumn):
"""Generate a formatted string of a hostname."""
def human_readable(self):
if self._value is None:
return ''
return self._value
def _get_ip_address(addresses, address_type, ip_address_family):
# Old style addresses
if address_type in addresses:
for addy in addresses[address_type]:
if int(addy['version']) in ip_address_family:
return addy['addr']
# New style addresses
new_address_type = address_type
if address_type == 'public':
new_address_type = 'floating'
if address_type == 'private':
new_address_type = 'fixed'
for network in addresses:
for addy in addresses[network]:
# Case where it is list of strings
if isinstance(addy, str):
if new_address_type == 'fixed':
return addresses[network][0]
else:
return addresses[network][-1]
# Case where it is a dict
if 'OS-EXT-IPS:type' not in addy:
continue
if addy['OS-EXT-IPS:type'] == new_address_type:
if int(addy['version']) in ip_address_family:
return addy['addr']
msg = _("ERROR: No %(type)s IP version %(family)s address found")
raise exceptions.CommandError(
msg % {"type": address_type, "family": ip_address_family}
)
def _prep_server_detail(compute_client, image_client, server, *, refresh=True):
"""Prepare the detailed server dict for printing
:param compute_client: a compute client instance
:param image_client: an image client instance
:param server: a Server resource
:param refresh: Flag indicating if ``server`` is already the latest version
or if it needs to be refreshed, for example when showing the latest
details of a server after creating it.
:rtype: a dict of server details
"""
info = server.to_dict()
if refresh:
server = compute_client.get_server(info['id'])
# we only update if the field is not empty, to avoid overwriting
# existing values
info.update(
**{x: y for x, y in server.to_dict().items() if x not in info or y}
)
# Some commands using this routine were originally implemented with the
# nova python wrappers, and were later migrated to use the SDK. Map the
# SDK's property names to the original property names to maintain backward
# compatibility for existing users.
column_map = {
'access_ipv4': 'accessIPv4',
'access_ipv6': 'accessIPv6',
'admin_password': 'adminPass',
'attached_volumes': 'volumes_attached',
'availability_zone': 'OS-EXT-AZ:availability_zone',
'compute_host': 'OS-EXT-SRV-ATTR:host',
'created_at': 'created',
'disk_config': 'OS-DCF:diskConfig',
'has_config_drive': 'config_drive',
'host_id': 'hostId',
'fault': 'fault',
'hostname': 'OS-EXT-SRV-ATTR:hostname',
'hypervisor_hostname': 'OS-EXT-SRV-ATTR:hypervisor_hostname',
'instance_name': 'OS-EXT-SRV-ATTR:instance_name',
'is_locked': 'locked',
'kernel_id': 'OS-EXT-SRV-ATTR:kernel_id',
'launch_index': 'OS-EXT-SRV-ATTR:launch_index',
'launched_at': 'OS-SRV-USG:launched_at',
'power_state': 'OS-EXT-STS:power_state',
'project_id': 'tenant_id',
'ramdisk_id': 'OS-EXT-SRV-ATTR:ramdisk_id',
'reservation_id': 'OS-EXT-SRV-ATTR:reservation_id',
'root_device_name': 'OS-EXT-SRV-ATTR:root_device_name',
'task_state': 'OS-EXT-STS:task_state',
'terminated_at': 'OS-SRV-USG:terminated_at',
'updated_at': 'updated',
'user_data': 'OS-EXT-SRV-ATTR:user_data',
'vm_state': 'OS-EXT-STS:vm_state',
'pinned_availability_zone': 'pinned_availability_zone',
}
# Some columns returned by openstacksdk should not be shown because they're
# either irrelevant or duplicates
ignored_columns = {
# computed columns
'interface_ip',
'location',
'private_v4',
'private_v6',
'public_v4',
'public_v6',
# create-only columns
'block_device_mapping',
'flavor_id',
'host',
'image_id',
'max_count',
'min_count',
'networks',
'personality',
'scheduler_hints',
# aliases
'volumes',
# unnecessary
'links',
}
# Some columns are only present in certain responses and should not be
# shown otherwise.
optional_columns = {
# only in create responses if '[api] enable_instance_password' is set
'admin_password',
# only present in errored servers
'fault',
# only present in create, detail responses
'security_groups',
}
data = {}
for key, value in info.items():
if key in ignored_columns:
continue
if key in optional_columns:
if info[key] is None:
continue
alias = column_map.get(key)
data[alias or key] = value
info = data
# Convert the image blob to a name
image_info = info.get('image', {})
if image_info and any(image_info.values()):
image_id = image_info.get('id', '')
try:
image = image_client.get_image(image_id)
info['image'] = f"{image.name} ({image_id})"
except Exception:
info['image'] = image_id
else:
# NOTE(melwitt): An server booted from a volume will have no image
# associated with it. We fill in the image with "N/A (booted from
# volume)" to help users who want to be able to grep for
# boot-from-volume servers when using the CLI.
info['image'] = IMAGE_STRING_FOR_BFV
# Convert the flavor blob to a name
flavor_info = info.get('flavor', {})
# Microversion 2.47 puts the embedded flavor into the server response
# body. The presence of the 'original_name' attribute indicates this.
if flavor_info.get('original_name') is None: # microversion < 2.47
flavor_id = flavor_info.get('id', '')
try:
flavor = compute_client.find_flavor(
flavor_id, ignore_missing=False
)
info['flavor'] = f"{flavor.name} ({flavor_id})"
except Exception:
info['flavor'] = flavor_id
else: # microversion >= 2.47
info['flavor'] = format_columns.DictColumn(flavor_info)
# there's a lot of redundant information in BDMs - strip it
if 'volumes_attached' in info:
info.update(
{
'volumes_attached': format_columns.ListDictColumn(
[
{
k: v
for k, v in volume.items()
if v is not None and k != 'location'
}
for volume in info.pop('volumes_attached') or []
]
)
}
)
if 'security_groups' in info:
info.update(
{
'security_groups': format_columns.ListDictColumn(
info.pop('security_groups'),
)
}
)
if 'tags' in info:
info.update(
{'tags': format_columns.ListColumn(info.pop('tags') or [])}
)
# Map 'networks' to 'addresses', if present. Note that the 'networks' key
# is used for create responses, otherwise it's 'addresses'. We know it'll
# be set because this is one of our optional columns.
if 'networks' in info:
info['addresses'] = format_columns.DictListColumn(
info.pop('networks', {}),
)
else:
info['addresses'] = AddressesColumn(info.get('addresses', {}))
# Map 'metadata' field to 'properties' and format
info['properties'] = format_columns.DictColumn(info.pop('metadata'))
# Migrate tenant_id to project_id naming
if 'tenant_id' in info:
info['project_id'] = info.pop('tenant_id')
# Map power state num to meaningful string
if 'OS-EXT-STS:power_state' in info:
info['OS-EXT-STS:power_state'] = PowerStateColumn(
info['OS-EXT-STS:power_state']
)
return info
def bool_from_str(value, strict=False):
true_strings = ('1', 't', 'true', 'on', 'y', 'yes')
false_strings = ('0', 'f', 'false', 'off', 'n', 'no')
if isinstance(value, bool):
return value
lowered = value.strip().lower()
if lowered in true_strings:
return True
elif lowered in false_strings or not strict:
return False
msg = _(
"Unrecognized value '%(value)s'; acceptable values are: %(valid)s"
) % {
'value': value,
'valid': ', '.join(
f"'{s}'" for s in sorted(true_strings + false_strings)
),
}
raise ValueError(msg)
def boolenv(*vars, default=False):
"""Search for the first defined of possibly many bool-like env vars.
Returns the first environment variable defined in vars, or returns the
default.
:param vars: Arbitrary strings to search for. Case sensitive.
:param default: The default to return if no value found.
:returns: A boolean corresponding to the value found, else the default if
no value found.
"""
for v in vars:
value = os.environ.get(v, None)
if value:
return bool_from_str(value)
return default
class AddFixedIP(command.ShowOne):
_description = _("Add fixed IP address to server")
def get_parser(self, prog_name):
parser = super().get_parser(prog_name)
parser.add_argument(
"server",
metavar="<server>",
help=_("Server to receive the fixed IP address (name or ID)"),
)
parser.add_argument(
"network",
metavar="<network>",
help=_(
"Network to allocate the fixed IP address from (name or ID)"
),
)
parser.add_argument(
"--fixed-ip-address",
metavar="<ip-address>",
help=_("Requested fixed IP address"),
)
parser.add_argument(
'--tag',
metavar='<tag>',
help=_(
'Tag for the attached interface. '
'(supported by --os-compute-api-version 2.49 or above)'
),
)
return parser
def take_action(self, parsed_args):
compute_client = self.app.client_manager.sdk_connection.compute
server = compute_client.find_server(
parsed_args.server, ignore_missing=False
)
if parsed_args.tag:
if not sdk_utils.supports_microversion(compute_client, '2.49'):
msg = _(
'--os-compute-api-version 2.49 or greater is required to '
'support the --tag option'
)
raise exceptions.CommandError(msg)
if self.app.client_manager.is_network_endpoint_enabled():
network_client = self.app.client_manager.network
net_id = network_client.find_network(
parsed_args.network, ignore_missing=False
).id
else:
net_id = parsed_args.network
kwargs = {'net_id': net_id}
if parsed_args.fixed_ip_address:
kwargs['fixed_ips'] = [
{"ip_address": parsed_args.fixed_ip_address}
]
if parsed_args.tag:
kwargs['tag'] = parsed_args.tag
interface = compute_client.create_server_interface(server.id, **kwargs)
columns = (
'port_id',
'server_id',
'net_id',
'mac_addr',
'port_state',
'fixed_ips',
)
column_headers = (
'Port ID',
'Server ID',
'Network ID',
'MAC Address',
'Port State',
'Fixed IPs',
)
if parsed_args.tag:
if sdk_utils.supports_microversion(compute_client, '2.49'):
columns += ('tag',)
column_headers += ('Tag',)
return (
column_headers,
utils.get_item_properties(
interface,
columns,
formatters={
'fixed_ips': format_columns.ListDictColumn,
},
),
)
class AddFloatingIP(network_common.NetworkAndComputeCommand):
_description = _("Add floating IP address to server")
def update_parser_common(self, parser):
parser.add_argument(
"server",
metavar="<server>",
help=_("Server to receive the floating IP address (name or ID)"),
)
parser.add_argument(
"ip_address",
metavar="<ip-address>",
help=_(
"Floating IP address to assign to the first available "
"server port (IP only)"
),
)
parser.add_argument(
"--fixed-ip-address",
metavar="<ip-address>",
help=_(
"Fixed IP address to associate with this floating IP address. "
"The first server port containing the fixed IP address will "
"be used"
),
)
return parser
def take_action_network(self, client, parsed_args):
compute_client = self.app.client_manager.sdk_connection.compute
attrs = {}
obj = client.find_ip(
parsed_args.ip_address,
ignore_missing=False,
)
server = compute_client.find_server(
parsed_args.server, ignore_missing=False
)
ports = list(client.ports(device_id=server.id))
if not ports:
msg = _('No attached ports found to associate floating IP with')
raise exceptions.CommandError(msg)
# If the fixed IP address was specified, we need to find the
# corresponding port.
if parsed_args.fixed_ip_address:
fip_address = parsed_args.fixed_ip_address
attrs['fixed_ip_address'] = fip_address
for port in ports:
for ip in port.fixed_ips:
if ip['ip_address'] == fip_address:
attrs['port_id'] = port.id
break
else:
continue
break
if 'port_id' not in attrs:
msg = _('No port found for fixed IP address %s')
raise exceptions.CommandError(msg % fip_address)
client.update_ip(obj, **attrs)
else:
# It's possible that one or more ports are not connected to a
# router and thus could fail association with a floating IP.
# Try each port until one succeeds. If none succeed, re-raise the
# last exception.
error = None
for port in ports:
attrs['port_id'] = port.id
try:
client.update_ip(obj, **attrs)
except sdk_exceptions.NotFoundException as exp:
# 404 ExternalGatewayForFloatingIPNotFound from neutron
LOG.info(
'Skipped port %s because it is not attached to '
'an external gateway',
port.id,
)
error = exp
continue
else:
error = None
break
if error:
raise error
def take_action_compute(self, client, parsed_args):
server = client.find_server(parsed_args.server, ignore_missing=False)
client.add_floating_ip_to_server(
server,
parsed_args.ip_address,
fixed_address=parsed_args.fixed_ip_address,
)
class AddPort(command.Command):
_description = _("Add port to server")
def get_parser(self, prog_name):
parser = super().get_parser(prog_name)
parser.add_argument(
"server",
metavar="<server>",
help=_("Server to add the port to (name or ID)"),
)
parser.add_argument(
"port",
metavar="<port>",
help=_("Port to add to the server (name or ID)"),
)
parser.add_argument(
'--tag',
metavar='<tag>',
help=_(
'Tag for the attached interface '
'(supported by --os-compute-api-version 2.49 or later)'
),
)
return parser
def take_action(self, parsed_args):
compute_client = self.app.client_manager.sdk_connection.compute
server = compute_client.find_server(
parsed_args.server, ignore_missing=False
)
if self.app.client_manager.is_network_endpoint_enabled():
network_client = self.app.client_manager.network
port_id = network_client.find_port(
parsed_args.port, ignore_missing=False
).id
else:
port_id = parsed_args.port
kwargs = {'port_id': port_id}
if parsed_args.tag:
if not sdk_utils.supports_microversion(compute_client, '2.49'):
msg = _(
'--os-compute-api-version 2.49 or greater is required to '
'support the --tag option'
)
raise exceptions.CommandError(msg)
kwargs['tag'] = parsed_args.tag
compute_client.create_server_interface(server, **kwargs)
class AddNetwork(command.Command):
_description = _("Add network to server")
def get_parser(self, prog_name):
parser = super().get_parser(prog_name)
parser.add_argument(
"server",
metavar="<server>",
help=_("Server to add the network to (name or ID)"),
)
parser.add_argument(
"network",
metavar="<network>",
help=_("Network to add to the server (name or ID)"),
)
parser.add_argument(
'--tag',
metavar='<tag>',
help=_(
'Tag for the attached interface. '
'(supported by --os-compute-api-version 2.49 or above)'
),
)
return parser
def take_action(self, parsed_args):
compute_client = self.app.client_manager.sdk_connection.compute
server = compute_client.find_server(
parsed_args.server, ignore_missing=False
)
if self.app.client_manager.is_network_endpoint_enabled():
network_client = self.app.client_manager.network
net_id = network_client.find_network(
parsed_args.network, ignore_missing=False
).id
else:
net_id = parsed_args.network
kwargs = {'net_id': net_id}
if parsed_args.tag:
if not sdk_utils.supports_microversion(compute_client, '2.49'):
msg = _(
'--os-compute-api-version 2.49 or greater is required to '
'support the --tag option'
)
raise exceptions.CommandError(msg)
kwargs['tag'] = parsed_args.tag
compute_client.create_server_interface(server, **kwargs)
class AddServerSecurityGroup(command.Command):
_description = _("Add security group to server")
def get_parser(self, prog_name):
parser = super().get_parser(prog_name)
parser.add_argument(
'server',
metavar='<server>',
help=_('Server (name or ID)'),
)
parser.add_argument(
'group',
metavar='<group>',
help=_('Security group to add (name or ID)'),
)
return parser
def take_action(self, parsed_args):
compute_client = self.app.client_manager.sdk_connection.compute
server = compute_client.find_server(
parsed_args.server, ignore_missing=False
)
if self.app.client_manager.is_network_endpoint_enabled():
# the server handles both names and IDs for neutron SGs, so just
# pass things through
security_group = parsed_args.group
else:
# however, if using nova-network then it needs a name, not an ID
security_group = compute_v2.find_security_group(
compute_client, parsed_args.group
)['name']
compute_client.add_security_group_to_server(server, security_group)
class AddServerVolume(command.ShowOne):
_description = _(
"""Add volume to server.
Specify ``--os-compute-api-version 2.20`` or higher to add a volume to a server
with status ``SHELVED`` or ``SHELVED_OFFLOADED``."""
)
def get_parser(self, prog_name):
parser = super().get_parser(prog_name)
parser.add_argument(
'server',
metavar='<server>',
help=_('Server (name or ID)'),
)
parser.add_argument(
'volume',
metavar='<volume>',
help=_('Volume to add (name or ID)'),
)
parser.add_argument(
'--device',
metavar='<device>',
help=_('Server internal device name for volume'),
)
parser.add_argument(
'--tag',
metavar='<tag>',
help=_(
'Tag for the attached volume '
'(supported by --os-compute-api-version 2.49 or above)'
),
)
# TODO(stephenfin): These should be called 'delete-on-termination' and
# 'preserve-on-termination'
termination_group = parser.add_mutually_exclusive_group()
termination_group.add_argument(
'--enable-delete-on-termination',
action='store_true',
help=_(
'Delete the volume when the server is destroyed '
'(supported by --os-compute-api-version 2.79 or above)'
),
)
termination_group.add_argument(
'--disable-delete-on-termination',
action='store_true',
help=_(
'Do not delete the volume when the server is destroyed '
'(supported by --os-compute-api-version 2.79 or above)'
),
)
return parser
def take_action(self, parsed_args):
compute_client = self.app.client_manager.sdk_connection.compute
volume_client = self.app.client_manager.sdk_connection.volume
server = compute_client.find_server(
parsed_args.server,
ignore_missing=False,
)
volume = volume_client.find_volume(
parsed_args.volume,
ignore_missing=False,
)
kwargs = {"volumeId": volume.id, "device": parsed_args.device}
if parsed_args.tag:
if not sdk_utils.supports_microversion(compute_client, '2.49'):
msg = _(
'--os-compute-api-version 2.49 or greater is required to '
'support the --tag option'
)
raise exceptions.CommandError(msg)
kwargs['tag'] = parsed_args.tag
if parsed_args.enable_delete_on_termination:
if not sdk_utils.supports_microversion(compute_client, '2.79'):
msg = _(
'--os-compute-api-version 2.79 or greater is required to '
'support the --enable-delete-on-termination option.'
)
raise exceptions.CommandError(msg)
kwargs['delete_on_termination'] = True
if parsed_args.disable_delete_on_termination:
if not sdk_utils.supports_microversion(compute_client, '2.79'):
msg = _(
'--os-compute-api-version 2.79 or greater is required to '
'support the --disable-delete-on-termination option.'
)
raise exceptions.CommandError(msg)
kwargs['delete_on_termination'] = False
volume_attachment = compute_client.create_volume_attachment(
server,
**kwargs,
)
columns = ('id', 'server id', 'volume id', 'device')
column_headers = ('ID', 'Server ID', 'Volume ID', 'Device')
if sdk_utils.supports_microversion(compute_client, '2.49'):
columns += ('tag',)
column_headers += ('Tag',)
if sdk_utils.supports_microversion(compute_client, '2.79'):
columns += ('delete_on_termination',)
column_headers += ('Delete On Termination',)
return (
column_headers,
utils.get_item_properties(
volume_attachment,
columns,
),
)
class NoneNICAction(argparse.Action):
def __init__(self, option_strings, dest, help=None):
super().__init__(
option_strings=option_strings,
dest=dest,
nargs=0,
default=[],
required=False,
help=help,
)
def __call__(self, parser, namespace, values, option_string=None):
# Make sure we have an empty dict rather than None
if getattr(namespace, self.dest, None) is None:
setattr(namespace, self.dest, [])
getattr(namespace, self.dest).append('none')
class AutoNICAction(argparse.Action):
def __init__(self, option_strings, dest, help=None):
super().__init__(
option_strings=option_strings,
dest=dest,
nargs=0,
default=[],
required=False,
help=help,
)
def __call__(self, parser, namespace, values, option_string=None):
# Make sure we have an empty dict rather than None
if getattr(namespace, self.dest, None) is None:
setattr(namespace, self.dest, [])
getattr(namespace, self.dest).append('auto')
class NICAction(argparse.Action):
def __init__(
self,
option_strings,
dest,
help=None,
metavar=None,
key=None,
):
self.key = key
super().__init__(
option_strings=option_strings,
dest=dest,
nargs=None,
const=None,
default=[],
type=None,
choices=None,
required=False,
help=help,
metavar=metavar,
)
def __call__(self, parser, namespace, values, option_string=None):
# Make sure we have an empty dict rather than None
if getattr(namespace, self.dest, None) is None:
setattr(namespace, self.dest, [])
if self.key:
if ',' in values or '=' in values:
msg = _(
"Invalid argument %s; characters ',' and '=' are not "
"allowed"
)
raise argparse.ArgumentError(self, msg % values)
values = '='.join([self.key, values])
else:
# Handle the special auto/none cases but only when a key isn't set
# (otherwise those could be valid values for the key)
if values in ('auto', 'none'):
getattr(namespace, self.dest).append(values)
return
# We don't include 'tag' here by default since that requires a
# particular microversion
info = {
'net-id': '',
'port-id': '',
'v4-fixed-ip': '',
'v6-fixed-ip': '',
}
for kv_str in values.split(','):
k, sep, v = kv_str.partition('=')
if k not in list(info) + ['tag'] or not v:
msg = _(
"Invalid argument %s; argument must be of form "
"'net-id=net-uuid,port-id=port-uuid,v4-fixed-ip=ip-addr,"
"v6-fixed-ip=ip-addr,tag=tag'"
)
raise argparse.ArgumentError(self, msg % values)
info[k] = v
if info['net-id'] and info['port-id']:
msg = _(
"Invalid argument %s; either 'network' or 'port' should be "
"specified but not both"
)
raise argparse.ArgumentError(self, msg % values)
if info['v4-fixed-ip'] and info['v6-fixed-ip']:
msg = _(
"Invalid argument %s; either 'v4-fixed-ip' or 'v6-fixed-ip' "
"should be specified but not both"
)
raise argparse.ArgumentError(self, msg % values)
getattr(namespace, self.dest).append(info)
class BDMLegacyAction(argparse.Action):
def __call__(self, parser, namespace, values, option_string=None):
# Make sure we have an empty list rather than None
if getattr(namespace, self.dest, None) is None:
setattr(namespace, self.dest, [])
dev_name, sep, dev_map = values.partition('=')
dev_map = dev_map.split(':') if dev_map else dev_map
if not dev_name or not dev_map or len(dev_map) > 4:
msg = _(
"Invalid argument %s; argument must be of form "
"'dev-name=id[:type[:size[:delete-on-terminate]]]'"
)
raise argparse.ArgumentError(self, msg % values)
mapping = {
'device_name': dev_name,
# store target; this may be a name and will need verification later
'uuid': dev_map[0],
'source_type': 'volume',
'destination_type': 'volume',
}
# decide source and destination type
if len(dev_map) > 1 and dev_map[1]:
if dev_map[1] not in ('volume', 'snapshot', 'image'):
msg = _(
"Invalid argument %s; 'type' must be one of: volume, "
"snapshot, image"
)
raise argparse.ArgumentError(self, msg % values)
mapping['source_type'] = dev_map[1]
# 3. append size and delete_on_termination, if present
if len(dev_map) > 2 and dev_map[2]:
mapping['volume_size'] = dev_map[2]
if len(dev_map) > 3 and dev_map[3]:
mapping['delete_on_termination'] = dev_map[3]
getattr(namespace, self.dest).append(mapping)
class BDMAction(parseractions.MultiKeyValueAction):
def __init__(self, option_strings, dest, **kwargs):
required_keys = []
optional_keys = [
'uuid',
'source_type',
'destination_type',
'disk_bus',
'device_type',
'device_name',
'volume_size',
'guest_format',
'boot_index',
'delete_on_termination',
'tag',
'volume_type',
]
super().__init__(
option_strings,
dest,
required_keys=required_keys,
optional_keys=optional_keys,
**kwargs,
)
# TODO(stephenfin): Remove once I549d0897ef3704b7f47000f867d6731ad15d3f2b
# or similar lands in a release
def validate_keys(self, keys):
"""Validate the provided keys.
:param keys: A list of keys to validate.
"""
valid_keys = self.required_keys | self.optional_keys
invalid_keys = [k for k in keys if k not in valid_keys]
if invalid_keys:
msg = _(
"Invalid keys %(invalid_keys)s specified.\n"
"Valid keys are: %(valid_keys)s"
)
raise argparse.ArgumentError(
self,
msg
% {
'invalid_keys': ', '.join(invalid_keys),
'valid_keys': ', '.join(valid_keys),
},
)
missing_keys = [k for k in self.required_keys if k not in keys]
if missing_keys:
msg = _(
"Missing required keys %(missing_keys)s.\n"
"Required keys are: %(required_keys)s"
)
raise argparse.ArgumentError(
self,
msg
% {
'missing_keys': ', '.join(missing_keys),
'required_keys': ', '.join(self.required_keys),
},
)
def __call__(self, parser, namespace, values, option_string=None):
if getattr(namespace, self.dest, None) is None:
setattr(namespace, self.dest, [])
if os.path.exists(values):
with open(values) as fh:
data = json.load(fh)
# Validate the keys - other validation is left to later
self.validate_keys(list(data))
getattr(namespace, self.dest, []).append(data)
else:
super().__call__(parser, namespace, values, option_string)
class CreateServer(command.ShowOne):
_description = _("Create a new server")
def get_parser(self, prog_name):
parser = super().get_parser(prog_name)
parser.add_argument(
'server_name',
metavar='<server-name>',
help=_('New server name'),
)
parser.add_argument(
'--flavor',
metavar='<flavor>',
required=True,
help=_('Create server with this flavor (name or ID)'),
)
disk_group = parser.add_mutually_exclusive_group()
disk_group.add_argument(
'--image',
metavar='<image>',
help=_('Create server boot disk from this image (name or ID)'),
)
# TODO(stephenfin): Is this actually useful? Looks like a straight port
# from 'nova boot --image-with'. Perhaps we should deprecate this.
disk_group.add_argument(
'--image-property',
metavar='<key=value>',
action=parseractions.KeyValueAction,
dest='image_properties',
help=_(
"Create server using the image that matches the specified "
"property. Property must match exactly one property."
),
)
disk_group.add_argument(
'--volume',
metavar='<volume>',
help=_(
'Create server using this volume as the boot disk (name or ID)'
'\n'
'This option automatically creates a block device mapping '
'with a boot index of 0. On many hypervisors (libvirt/kvm '
'for example) this will be device vda. Do not create a '
'duplicate mapping using --block-device-mapping for this '
'volume.'
),
)
disk_group.add_argument(
'--snapshot',
metavar='<snapshot>',
help=_(
'Create server using this snapshot as the boot disk (name or '
'ID)\n'
'This option automatically creates a block device mapping '
'with a boot index of 0. On many hypervisors (libvirt/kvm '
'for example) this will be device vda. Do not create a '
'duplicate mapping using --block-device-mapping for this '
'volume.'
),
)
parser.add_argument(
'--boot-from-volume',
metavar='<volume-size>',
type=int,
help=_(
'When used in conjunction with the ``--image`` or '
'``--image-property`` option, this option automatically '
'creates a block device mapping with a boot index of 0 '
'and tells the compute service to create a volume of the '
'given size (in GB) from the specified image and use it '
'as the root disk of the server. The root volume will not '
'be deleted when the server is deleted. This option is '
'mutually exclusive with the ``--volume`` and ``--snapshot`` '
'options.'
),
)
# TODO(stephenfin): Remove this in the v7.0
parser.add_argument(
'--block-device-mapping',
metavar='<dev-name=mapping>',
action=BDMLegacyAction,
default=[],
# NOTE(RuiChen): Add '\n' to the end of line to improve formatting;
# see cliff's _SmartHelpFormatter for more details.
help=_(
'**Deprecated** Create a block device on the server.\n'
'Block device mapping in the format\n'
'<dev-name>=<id>:<type>:<size(GB)>:<delete-on-terminate>\n'
'<dev-name>: block device name, like: vdb, xvdc '
'(required)\n'
'<id>: Name or ID of the volume, volume snapshot or image '
'(required)\n'
'<type>: volume, snapshot or image; default: volume '
'(optional)\n'
'<size(GB)>: volume size if create from image or snapshot '
'(optional)\n'
'<delete-on-terminate>: true or false; default: false '
'(optional)\n'
'Replaced by --block-device'
),
)
parser.add_argument(
'--block-device',
metavar='<block-device>',
action=BDMAction,
dest='block_devices',
default=[],
help=_(
'Create a block device on the server.\n'
'Either a path to a JSON file or a CSV-serialized string '
'describing the block device mapping.\n'
'The following keys are accepted for both:\n'
'uuid=<uuid>: UUID of the volume, snapshot or ID '
'(required if using source image, snapshot or volume),\n'
'source_type=<source_type>: source type '
'(one of: image, snapshot, volume, blank),\n'
'destination_type=<destination_type>: destination type '
'(one of: volume, local) (optional),\n'
'disk_bus=<disk_bus>: device bus '
'(one of: uml, lxc, virtio, ...) (optional),\n'
'device_type=<device_type>: device type '
'(one of: disk, cdrom, etc. (optional),\n'
'device_name=<device_name>: name of the device (optional),\n'
'volume_size=<volume_size>: size of the block device in MiB '
'(for swap) or GiB (for everything else) (optional),\n'
'guest_format=<guest_format>: format of device (optional),\n'
'boot_index=<boot_index>: index of disk used to order boot '
'disk '
'(required for volume-backed instances),\n'
'delete_on_termination=<true|false>: whether to delete the '
'volume upon deletion of server (optional),\n'
'tag=<tag>: device metadata tag (optional),\n'
'volume_type=<volume_type>: type of volume to create (name or '
'ID) when source if blank, image or snapshot and dest is '
'volume (optional)'
),
)
parser.add_argument(
'--swap',
metavar='<swap>',
type=int,
help=(
"Create and attach a local swap block device of <swap_size> "
"MiB."
),
)
parser.add_argument(
'--ephemeral',
metavar='<size=size[,format=format]>',
action=parseractions.MultiKeyValueAction,
dest='ephemerals',
default=[],
required_keys=['size'],
optional_keys=['format'],
help=(
"Create and attach a local ephemeral block device of <size> "
"GiB and format it to <format>."
),
)
parser.add_argument(
'--network',
metavar='<network>',
dest='nics',
action=NICAction,
key='net-id',
help=_(
"Create a NIC on the server and connect it to network. "
"Specify option multiple times to create multiple NICs. "
"This is a wrapper for the '--nic net-id=<network>' "
"parameter that provides simple syntax for the standard "
"use case of connecting a new server to a given network. "
"For more advanced use cases, refer to the '--nic' parameter."
),
)
parser.add_argument(
'--port',
metavar='<port>',
dest='nics',
action=NICAction,
key='port-id',
help=_(
"Create a NIC on the server and connect it to port. "
"Specify option multiple times to create multiple NICs. "
"This is a wrapper for the '--nic port-id=<port>' "
"parameter that provides simple syntax for the standard "
"use case of connecting a new server to a given port. For "
"more advanced use cases, refer to the '--nic' parameter."
),
)
parser.add_argument(
'--no-network',
dest='nics',
action=NoneNICAction,
help=_(
"Do not attach a network to the server. "
"This is a wrapper for the '--nic none' option that provides "
"a simple syntax for disabling network connectivity for a new "
"server. "
"For more advanced use cases, refer to the '--nic' parameter. "
"(supported by --os-compute-api-version 2.37 or above)"
),
)
parser.add_argument(
'--auto-network',
dest='nics',
action=AutoNICAction,
help=_(
"Automatically allocate a network to the server. "
"This is the default network allocation policy. "
"This is a wrapper for the '--nic auto' option that provides "
"a simple syntax for enabling automatic configuration of "
"network connectivity for a new server. "
"For more advanced use cases, refer to the '--nic' parameter. "
"(supported by --os-compute-api-version 2.37 or above)"
),
)
parser.add_argument(
'--nic',
metavar="<net-id=net-uuid,port-id=port-uuid,v4-fixed-ip=ip-addr,"
"v6-fixed-ip=ip-addr,tag=tag,auto,none>",
dest='nics',
action=NICAction,
# NOTE(RuiChen): Add '\n' to the end of line to improve formatting;
# see cliff's _SmartHelpFormatter for more details.
help=_(
"Create a NIC on the server.\n"
"NIC in the format:\n"
"net-id=<net-uuid>: attach NIC to network with this UUID,\n"
"port-id=<port-uuid>: attach NIC to port with this UUID,\n"
"v4-fixed-ip=<ip-addr>: IPv4 fixed address for NIC (optional),"
"\n"
"v6-fixed-ip=<ip-addr>: IPv6 fixed address for NIC (optional),"
"\n"
"tag: interface metadata tag (optional) "
"(supported by --os-compute-api-version 2.43 or above),\n"
"none: (v2.37+) no network is attached,\n"
"auto: (v2.37+) the compute service will automatically "
"allocate a network.\n"
"\n"
"Specify option multiple times to create multiple NICs.\n"
"Specifying a --nic of auto or none cannot be used with any "
"other --nic value.\n"
"Either net-id or port-id must be provided, but not both."
),
)
parser.add_argument(
'--password',
metavar='<password>',
help=_(
'Set the password to this server. '
'This option requires cloud support.'
),
)
parser.add_argument(
'--security-group',
metavar='<security-group>',
action='append',
default=[],
help=_(
'Security group to assign to this server (name or ID) '
'(repeat option to set multiple groups)'
),
)
parser.add_argument(
'--key-name',
metavar='<key-name>',
help=_('Keypair to inject into this server'),
)
parser.add_argument(
'--property',
metavar='<key=value>',
action=parseractions.KeyValueAction,
dest='properties',
help=_(
'Set a property on this server '
'(repeat option to set multiple values)'
),
)
parser.add_argument(
'--file',
metavar='<dest-filename=source-filename>',
action='append',
default=[],
help=_(
'File(s) to inject into image before boot '
'(repeat option to set multiple files)'
'(supported by --os-compute-api-version 2.57 or below)'
),
)
parser.add_argument(
'--user-data',
metavar='<user-data>',
help=_('User data file to serve from the metadata server'),
)
parser.add_argument(
'--description',
metavar='<description>',
help=_(
'Set description for the server '
'(supported by --os-compute-api-version 2.19 or above)'
),
)
parser.add_argument(
'--availability-zone',
metavar='<zone-name>',
help=_(
'Select an availability zone for the server. '
'Host and node are optional parameters. '
'Availability zone in the format '
'<zone-name>:<host-name>:<node-name>, '
'<zone-name>::<node-name>, <zone-name>:<host-name> '
'or <zone-name>'
),
)
parser.add_argument(
'--host',
metavar='<host>',
help=_(
'Requested host to create servers. '
'(admin only) '
'(supported by --os-compute-api-version 2.74 or above)'
),
)
parser.add_argument(
'--hypervisor-hostname',
metavar='<hypervisor-hostname>',
help=_(
'Requested hypervisor hostname to create servers. '
'(admin only) '
'(supported by --os-compute-api-version 2.74 or above)'
),
)
parser.add_argument(
'--server-group',
metavar='<server-group>',
help=_(
"Server group to create the server within "
"(this is an alias for '--hint group=<server-group-id>')"
),
)
parser.add_argument(
'--hint',
metavar='<key=value>',
action=parseractions.KeyValueAppendAction,
dest='hints',
default={},
help=_('Hints for the scheduler'),
)
config_drive_group = parser.add_mutually_exclusive_group()
config_drive_group.add_argument(
'--use-config-drive',
action='store_true',
dest='config_drive',
help=_("Enable config drive."),
)
config_drive_group.add_argument(
'--no-config-drive',
action='store_false',
dest='config_drive',
help=_("Disable config drive."),
)
# TODO(stephenfin): Drop support in the next major version bump after
# Victoria
config_drive_group.add_argument(
'--config-drive',
metavar='<config-drive-volume>|True',
default=False,
help=_(
"**Deprecated** Use specified volume as the config drive, "
"or 'True' to use an ephemeral drive. Replaced by "
"'--use-config-drive'."
),
)
parser.add_argument(
'--min',
metavar='<count>',
type=int,
default=1,
help=_('Minimum number of servers to launch (default=1)'),
)
parser.add_argument(
'--max',
metavar='<count>',
type=int,
default=1,
help=_('Maximum number of servers to launch (default=1)'),
)
parser.add_argument(
'--tag',
metavar='<tag>',
action='append',
default=[],
dest='tags',
help=_(
'Tags for the server. '
'Specify multiple times to add multiple tags. '
'(supported by --os-compute-api-version 2.52 or above)'
),
)
parser.add_argument(
'--hostname',
metavar='<hostname>',
help=_(
'Hostname configured for the server in the metadata service. '
'If unset, a hostname will be automatically generated from '
'the server name. '
'A utility such as cloud-init is required to propagate the '
'hostname in the metadata service to the guest OS itself. '
'(supported by --os-compute-api-version 2.90 or above)'
),
)
parser.add_argument(
'--wait',
action='store_true',
help=_('Wait for build to complete'),
)
parser.add_argument(
'--trusted-image-cert',
metavar='<trusted-cert-id>',
action='append',
dest='trusted_image_certs',
help=_(
'Trusted image certificate IDs used to validate certificates '
'during the image signature verification process. '
'May be specified multiple times to pass multiple trusted '
'image certificate IDs. '
'(supported by --os-compute-api-version 2.63 or above)'
),
)
return parser
def take_action(self, parsed_args):
def _show_progress(progress):
if progress:
self.app.stdout.write('\rProgress: %s' % progress)
self.app.stdout.flush()
compute_client = self.app.client_manager.sdk_connection.compute
volume_client = self.app.client_manager.volume
image_client = self.app.client_manager.image
# Lookup parsed_args.image
image = None
if parsed_args.image:
image = image_client.find_image(
parsed_args.image, ignore_missing=False
)
if not image and parsed_args.image_properties:
def emit_duplicated_warning(img):
img_uuid_list = [str(image.id) for image in img]
LOG.warning(
'Multiple matching images: %(img_uuid_list)s\n'
'Using image: %(chosen_one)s',
{
'img_uuid_list': img_uuid_list,
'chosen_one': img_uuid_list[0],
},
)
def _match_image(image_api, wanted_properties):
image_list = image_api.images()
images_matched = []
for img in image_list:
img_dict = {}
# exclude any unhashable entries
img_dict_items = list(img.items())
if img.properties:
img_dict_items.extend(list(img.properties.items()))
for key, value in img_dict_items:
try:
{key, value}
except TypeError:
if key != 'properties':
LOG.debug(
'Skipped the \'%s\' attribute. '
'That cannot be compared. '
'(image: %s, value: %s)',
key,
img.id,
value,
)
pass
else:
img_dict[key] = value
if all(
k in img_dict and img_dict[k] == v
for k, v in wanted_properties.items()
):
images_matched.append(img)
return images_matched
images = _match_image(image_client, parsed_args.image_properties)
if len(images) > 1:
emit_duplicated_warning(images, parsed_args.image_properties)
if images:
image = images[0]
else:
msg = _(
'No images match the property expected by '
'--image-property'
)
raise exceptions.CommandError(msg)
volume = None
if parsed_args.volume:
# --volume and --boot-from-volume are mutually exclusive.
if parsed_args.boot_from_volume:
msg = _('--volume is not allowed with --boot-from-volume')
raise exceptions.CommandError(msg)
volume = utils.find_resource(
volume_client.volumes,
parsed_args.volume,
).id
snapshot = None
if parsed_args.snapshot:
# --snapshot and --boot-from-volume are mutually exclusive.
if parsed_args.boot_from_volume:
msg = _('--snapshot is not allowed with --boot-from-volume')
raise exceptions.CommandError(msg)
snapshot = utils.find_resource(
volume_client.volume_snapshots,
parsed_args.snapshot,
).id
flavor = compute_client.find_flavor(
parsed_args.flavor, ignore_missing=False
)
if parsed_args.file:
if sdk_utils.supports_microversion(compute_client, '2.57'):
msg = _(
'Personality files are deprecated and are not supported '
'for --os-compute-api-version greater than 2.56; use '
'user data instead'
)
raise exceptions.CommandError(msg)
files = {}
for f in parsed_args.file:
dst, src = f.split('=', 1)
try:
files[dst] = open(src, 'rb')
except OSError as e:
msg = _("Can't open '%(source)s': %(exception)s")
raise exceptions.CommandError(
msg % {'source': src, 'exception': e}
)
if parsed_args.min > parsed_args.max:
msg = _("min instances should be <= max instances")
raise exceptions.CommandError(msg)
if parsed_args.min < 1:
msg = _("min instances should be > 0")
raise exceptions.CommandError(msg)
if parsed_args.max < 1:
msg = _("max instances should be > 0")
raise exceptions.CommandError(msg)
user_data = None
if parsed_args.user_data:
try:
with open(parsed_args.user_data, 'rb') as fh:
# TODO(stephenfin): SDK should do this for us
user_data = base64.b64encode(fh.read()).decode('utf-8')
except OSError as e:
msg = _("Can't open '%(data)s': %(exception)s")
raise exceptions.CommandError(
msg % {'data': parsed_args.user_data, 'exception': e}
)
if parsed_args.description:
if not sdk_utils.supports_microversion(compute_client, '2.19'):
msg = _(
'--os-compute-api-version 2.19 or greater is '
'required to support the --description option'
)
raise exceptions.CommandError(msg)
block_device_mapping_v2 = []
if parsed_args.boot_from_volume:
# Tell nova to create a root volume from the image provided.
if not image:
msg = _(
"An image (--image or --image-property) is required "
"to support --boot-from-volume option"
)
raise exceptions.CommandError(msg)
block_device_mapping_v2 = [
{
'uuid': image.id,
'boot_index': 0,
'source_type': 'image',
'destination_type': 'volume',
'volume_size': parsed_args.boot_from_volume,
}
]
# If booting from volume we do not pass an image to compute.
image = None
elif image:
block_device_mapping_v2 = [
{
'uuid': image.id,
'boot_index': 0,
'source_type': 'image',
'destination_type': 'local',
'delete_on_termination': True,
}
]
elif volume:
block_device_mapping_v2 = [
{
'uuid': volume,
'boot_index': 0,
'source_type': 'volume',
'destination_type': 'volume',
}
]
elif snapshot:
block_device_mapping_v2 = [
{
'uuid': snapshot,
'boot_index': 0,
'source_type': 'snapshot',
'destination_type': 'volume',
'delete_on_termination': False,
}
]
if parsed_args.swap:
block_device_mapping_v2.append(
{
'boot_index': -1,
'source_type': 'blank',
'destination_type': 'local',
'guest_format': 'swap',
'volume_size': parsed_args.swap,
'delete_on_termination': True,
}
)
for mapping in parsed_args.ephemerals:
block_device_mapping_dict = {
'boot_index': -1,
'source_type': 'blank',
'destination_type': 'local',
'delete_on_termination': True,
'volume_size': mapping['size'],
}
if 'format' in mapping:
block_device_mapping_dict['guest_format'] = mapping['format']
block_device_mapping_v2.append(block_device_mapping_dict)
# Handle block device by device name order, like: vdb -> vdc -> vdd
for mapping in parsed_args.block_device_mapping:
# The 'uuid' field isn't necessarily a UUID yet; let's validate it
# just in case
if mapping['source_type'] == 'volume':
volume_id = utils.find_resource(
volume_client.volumes,
mapping['uuid'],
).id
mapping['uuid'] = volume_id
elif mapping['source_type'] == 'snapshot':
snapshot_id = utils.find_resource(
volume_client.volume_snapshots,
mapping['uuid'],
).id
mapping['uuid'] = snapshot_id
elif mapping['source_type'] == 'image':
# NOTE(mriedem): In case --image is specified with the same
# image, that becomes the root disk for the server. If the
# block device is specified with a root device name, e.g.
# vda, then the compute API will likely fail complaining
# that there is a conflict. So if using the same image ID,
# which doesn't really make sense but it's allowed, the
# device name would need to be a non-root device, e.g. vdb.
# Otherwise if the block device image is different from the
# one specified by --image, then the compute service will
# create a volume from the image and attach it to the
# server as a non-root volume.
image_id = image_client.find_image(
mapping['uuid'],
ignore_missing=False,
).id
mapping['uuid'] = image_id
block_device_mapping_v2.append(mapping)
for mapping in parsed_args.block_devices:
if 'boot_index' in mapping:
try:
mapping['boot_index'] = int(mapping['boot_index'])
except ValueError:
msg = _(
'The boot_index key of --block-device should be an '
'integer'
)
raise exceptions.CommandError(msg)
if 'tag' in mapping and (
not sdk_utils.supports_microversion(compute_client, '2.42')
):
msg = _(
'--os-compute-api-version 2.42 or greater is '
'required to support the tag key of --block-device'
)
raise exceptions.CommandError(msg)
if 'volume_type' in mapping and (
not sdk_utils.supports_microversion(compute_client, '2.67')
):
msg = _(
'--os-compute-api-version 2.67 or greater is '
'required to support the volume_type key of --block-device'
)
raise exceptions.CommandError(msg)
if 'source_type' in mapping:
if mapping['source_type'] not in (
'volume',
'image',
'snapshot',
'blank',
):
msg = _(
'The source_type key of --block-device should be one '
'of: volume, image, snapshot, blank'
)
raise exceptions.CommandError(msg)
else:
mapping['source_type'] = 'blank'
if 'destination_type' in mapping:
if mapping['destination_type'] not in ('local', 'volume'):
msg = _(
'The destination_type key of --block-device should be '
'one of: local, volume'
)
raise exceptions.CommandError(msg)
else:
if mapping['source_type'] in ('blank',):
mapping['destination_type'] = 'local'
else: # volume, image, snapshot
mapping['destination_type'] = 'volume'
if 'delete_on_termination' in mapping:
try:
value = bool_from_str(
mapping['delete_on_termination'],
strict=True,
)
except ValueError:
msg = _(
'The delete_on_termination key of --block-device '
'should be a boolean-like value'
)
raise exceptions.CommandError(msg)
mapping['delete_on_termination'] = value
else:
if mapping['destination_type'] == 'local':
mapping['delete_on_termination'] = True
block_device_mapping_v2.append(mapping)
if not any(
[bdm.get('boot_index') == 0 for bdm in block_device_mapping_v2]
):
msg = _(
'An image (--image, --image-property) or bootable volume '
'(--volume, --snapshot, --block-device) is required'
)
raise exceptions.CommandError(msg)
# Default to empty list if nothing was specified and let nova
# decide the default behavior.
networks: ty.Union[str, ty.List[ty.Dict[str, str]], None] = []
if 'auto' in parsed_args.nics or 'none' in parsed_args.nics:
if len(parsed_args.nics) > 1:
msg = _(
'Specifying a --nic of auto or none cannot '
'be used with any other --nic, --network '
'or --port value.'
)
raise exceptions.CommandError(msg)
if not sdk_utils.supports_microversion(compute_client, '2.37'):
msg = _(
'--os-compute-api-version 2.37 or greater is '
'required to support explicit auto-allocation of a '
'network or to disable network allocation'
)
raise exceptions.CommandError(msg)
networks = parsed_args.nics[0]
else:
for nic in parsed_args.nics:
if 'tag' in nic:
if not sdk_utils.supports_microversion(
compute_client, '2.43'
):
msg = _(
'--os-compute-api-version 2.43 or greater is '
'required to support the --nic tag field'
)
raise exceptions.CommandError(msg)
if self.app.client_manager.is_network_endpoint_enabled():
network_client = self.app.client_manager.network
if nic['net-id']:
net = network_client.find_network(
nic['net-id'],
ignore_missing=False,
)
nic['net-id'] = net.id
if nic['port-id']:
port = network_client.find_port(
nic['port-id'],
ignore_missing=False,
)
nic['port-id'] = port.id
else:
if nic['net-id']:
net = compute_v2.find_network(
compute_client,
nic['net-id'],
)
nic['net-id'] = net['id']
if nic['port-id']:
msg = _(
"Can't create server with port specified "
"since network endpoint not enabled"
)
raise exceptions.CommandError(msg)
# convert from the novaclient-derived "NIC" view to the actual
# "network" view
network = {}
if nic['net-id']:
network['uuid'] = nic['net-id']
if nic['port-id']:
network['port'] = nic['port-id']
if nic['v4-fixed-ip']:
network['fixed'] = nic['v4-fixed-ip']
elif nic['v6-fixed-ip']:
network['fixed'] = nic['v6-fixed-ip']
if nic.get('tag'): # tags are optional
network['tag'] = nic['tag']
networks.append(network)
if not parsed_args.nics and sdk_utils.supports_microversion(
compute_client, '2.37'
):
# Compute API version >= 2.37 requires a value, so default to
# 'auto' to maintain legacy behavior if a nic wasn't specified.
networks = 'auto'
# Check security group exist and convert ID to name
security_groups = []
if self.app.client_manager.is_network_endpoint_enabled():
network_client = self.app.client_manager.network
for each_sg in parsed_args.security_group:
sg = network_client.find_security_group(
each_sg, ignore_missing=False
)
# Use security group ID to avoid multiple security group have
# same name in neutron networking backend
security_groups.append({'name': sg.id})
else:
# Handle nova-network case
for each_sg in parsed_args.security_group:
sg = compute_v2.find_security_group(compute_client, each_sg)
security_groups.append({'name': sg['name']})
hints = {}
for key, values in parsed_args.hints.items():
# only items with multiple values will result in a list
if len(values) == 1:
hints[key] = values[0]
else:
hints[key] = values
if parsed_args.server_group:
server_group_obj = compute_client.find_server_group(
parsed_args.server_group, ignore_missing=False
)
hints['group'] = server_group_obj.id
if isinstance(parsed_args.config_drive, bool):
# NOTE(stephenfin): The API doesn't accept False as a value :'(
config_drive = parsed_args.config_drive or None
else:
# TODO(stephenfin): Remove when we drop support for
# '--config-drive'
if str(parsed_args.config_drive).lower() in ("true", "1"):
config_drive = True
elif str(parsed_args.config_drive).lower() in (
"false",
"0",
"",
"none",
):
config_drive = None
else:
config_drive = parsed_args.config_drive
kwargs = {
'name': parsed_args.server_name,
'image_id': image.id if image else '',
'flavor_id': flavor.id,
'min_count': parsed_args.min,
'max_count': parsed_args.max,
}
if parsed_args.description:
kwargs['description'] = parsed_args.description
if parsed_args.availability_zone:
kwargs['availability_zone'] = parsed_args.availability_zone
if parsed_args.password:
kwargs['admin_password'] = parsed_args.password
if parsed_args.properties:
kwargs['metadata'] = parsed_args.properties
if parsed_args.key_name:
kwargs['key_name'] = parsed_args.key_name
if user_data:
kwargs['user_data'] = user_data
if files:
kwargs['personality'] = files
if security_groups:
kwargs['security_groups'] = security_groups
if block_device_mapping_v2:
kwargs['block_device_mapping'] = block_device_mapping_v2
if hints:
kwargs['scheduler_hints'] = hints
if networks is not None:
kwargs['networks'] = networks
if config_drive is not None:
kwargs['config_drive'] = config_drive
if parsed_args.tags:
if not sdk_utils.supports_microversion(compute_client, '2.52'):
msg = _(
'--os-compute-api-version 2.52 or greater is required to '
'support the --tag option'
)
raise exceptions.CommandError(msg)
kwargs['tags'] = parsed_args.tags
if parsed_args.host:
if not sdk_utils.supports_microversion(compute_client, '2.74'):
msg = _(
'--os-compute-api-version 2.74 or greater is required to '
'support the --host option'
)
raise exceptions.CommandError(msg)
kwargs['host'] = parsed_args.host
if parsed_args.hypervisor_hostname:
if not sdk_utils.supports_microversion(compute_client, '2.74'):
msg = _(
'--os-compute-api-version 2.74 or greater is required to '
'support the --hypervisor-hostname option'
)
raise exceptions.CommandError(msg)
kwargs['hypervisor_hostname'] = parsed_args.hypervisor_hostname
if parsed_args.hostname:
if not sdk_utils.supports_microversion(compute_client, '2.90'):
msg = _(
'--os-compute-api-version 2.90 or greater is required to '
'support the --hostname option'
)
raise exceptions.CommandError(msg)
kwargs['hostname'] = parsed_args.hostname
# TODO(stephenfin): Handle OS_TRUSTED_IMAGE_CERTIFICATE_IDS
if parsed_args.trusted_image_certs:
if not (image and not parsed_args.boot_from_volume):
msg = _(
'--trusted-image-cert option is only supported for '
'servers booted directly from images'
)
raise exceptions.CommandError(msg)
if not sdk_utils.supports_microversion(compute_client, '2.63'):
msg = _(
'--os-compute-api-version 2.63 or greater is required to '
'support the --trusted-image-cert option'
)
raise exceptions.CommandError(msg)
certs = parsed_args.trusted_image_certs
kwargs['trusted_image_certificates'] = certs
LOG.debug('boot_kwargs: %s', kwargs)
# Wrap the call to catch exceptions in order to close files
try:
server = compute_client.create_server(**kwargs)
finally:
# Clean up open files - make sure they are not strings
for f in files:
if hasattr(f, 'close'):
f.close()
if parsed_args.wait:
if utils.wait_for_status(
compute_client.get_server,
server.id,
callback=_show_progress,
):
self.app.stdout.write('\n')
else:
msg = _('Error creating server: %s') % parsed_args.server_name
raise exceptions.CommandError(msg)
data = _prep_server_detail(compute_client, image_client, server)
return zip(*sorted(data.items()))
class CreateServerDump(command.Command):
"""Create a dump file in server(s)
Trigger crash dump in server(s) with features like kdump in Linux.
It will create a dump file in the server(s) dumping the server(s)'
memory, and also crash the server(s). This is contingent on guest operating
system support, and the location of the dump file inside the guest will
depend on the exact guest operating system.
This command requires ``--os-compute-api-version`` 2.17 or greater.
"""
def get_parser(self, prog_name):
parser = super().get_parser(prog_name)
parser.add_argument(
'server',
metavar='<server>',
nargs='+',
help=_('Server(s) to create dump file (name or ID)'),
)
return parser
def take_action(self, parsed_args):
compute_client = self.app.client_manager.sdk_connection.compute
for name_or_id in parsed_args.server:
server = compute_client.find_server(name_or_id)
server.trigger_crash_dump(compute_client)
class DeleteServer(command.Command):
_description = _("Delete server(s)")
def get_parser(self, prog_name):
parser = super().get_parser(prog_name)
parser.add_argument(
'server',
metavar='<server>',
nargs="+",
help=_('Server(s) to delete (name or ID)'),
)
parser.add_argument(
'--force',
action='store_true',
help=_('Force delete server(s)'),
)
parser.add_argument(
'--all-projects',
action='store_true',
default=boolenv('ALL_PROJECTS'),
help=_(
'Delete server(s) in another project by name (admin only)'
'(can be specified using the ALL_PROJECTS envvar)'
),
)
parser.add_argument(
'--wait',
action='store_true',
help=_('Wait for delete to complete'),
)
return parser
def take_action(self, parsed_args):
def _show_progress(progress):
if progress:
self.app.stdout.write('\rProgress: %s' % progress)
self.app.stdout.flush()
compute_client = self.app.client_manager.sdk_connection.compute
for server in parsed_args.server:
server_obj = compute_client.find_server(
server,
ignore_missing=False,
all_projects=parsed_args.all_projects,
)
compute_client.delete_server(server_obj, force=parsed_args.force)
if parsed_args.wait:
try:
compute_client.wait_for_delete(
server_obj, callback=_show_progress
)
except sdk_exceptions.ResourceTimeout:
msg = _('Error deleting server: %s') % server_obj.id
raise exceptions.CommandError(msg)
class PercentAction(argparse.Action):
def __init__(
self,
option_strings,
dest,
nargs=None,
const=None,
default=None,
type=None,
choices=None,
required=False,
help=None,
metavar=None,
):
if nargs == 0:
raise ValueError(
'nargs for store actions must be != 0; if you '
'have nothing to store, actions such as store '
'true or store const may be more appropriate'
)
if const is not None:
raise ValueError('const does not make sense for PercentAction')
super().__init__(
option_strings=option_strings,
dest=dest,
nargs=nargs,
const=const,
default=default,
type=type,
choices=choices,
required=required,
help=help,
metavar=metavar,
)
def __call__(self, parser, namespace, values, option_string=None):
x = int(values)
if not 0 < x <= 100:
raise argparse.ArgumentError(self, "Must be between 0 and 100")
setattr(namespace, self.dest, x)
class ListServer(command.Lister):
_description = _("List servers")
def get_parser(self, prog_name):
parser = super().get_parser(prog_name)
parser.add_argument(
'--reservation-id',
metavar='<reservation-id>',
help=_('Only return instances that match the reservation'),
)
parser.add_argument(
'--ip',
metavar='<ip-address-regex>',
help=_('Regular expression to match IP addresses'),
)
parser.add_argument(
'--ip6',
metavar='<ip-address-regex>',
help=_(
'Regular expression to match IPv6 addresses. Note '
'that this option only applies for non-admin users '
'when using ``--os-compute-api-version`` 2.5 or greater.'
),
)
parser.add_argument(
'--name',
metavar='<name-regex>',
help=_('Regular expression to match names'),
)
parser.add_argument(
'--instance-name',
metavar='<server-name>',
help=_('Regular expression to match instance name (admin only)'),
)
# taken from 'task_and_vm_state_from_status' function in nova
# the API sadly reports these in upper case and while it would be
# wonderful to plaster over this ugliness client-side, there are
# already users in the wild doing this in upper case that we need to
# support
parser.add_argument(
'--status',
metavar='<status>',
choices=(
'ACTIVE',
'BUILD',
'DELETED',
'ERROR',
'HARD_REBOOT',
'MIGRATING',
'PASSWORD',
'PAUSED',
'REBOOT',
'REBUILD',
'RESCUE',
'RESIZE',
'REVERT_RESIZE',
'SHELVED',
'SHELVED_OFFLOADED',
'SHUTOFF',
'SOFT_DELETED',
'SUSPENDED',
'VERIFY_RESIZE',
),
help=_('Search by server status'),
)
parser.add_argument(
'--flavor',
metavar='<flavor>',
help=_('Search by flavor (name or ID)'),
)
parser.add_argument(
'--image',
metavar='<image>',
help=_('Search by image (name or ID)'),
)
parser.add_argument(
'--host',
metavar='<hostname>',
help=_('Search by hostname'),
)
parser.add_argument(
'--all-projects',
action='store_true',
default=boolenv('ALL_PROJECTS'),
help=_(
'Include all projects (admin only) '
'(can be specified using the ALL_PROJECTS envvar)'
),
)
parser.add_argument(
'--project',
metavar='<project>',
help=_("Search by project (admin only) (name or ID)"),
)
identity_common.add_project_domain_option_to_parser(parser)
parser.add_argument(
'--user',
metavar='<user>',
help=_(
'Search by user (name or ID) '
'(admin only before microversion 2.83)'
),
)
identity_common.add_user_domain_option_to_parser(parser)
parser.add_argument(
'--deleted',
action='store_true',
default=False,
help=_('Only display deleted servers (admin only)'),
)
parser.add_argument(
'--availability-zone',
default=None,
help=_(
'Search by availability zone '
'(admin only before microversion 2.83)'
),
)
parser.add_argument(
'--key-name',
help=_(
'Search by keypair name '
'(admin only before microversion 2.83)'
),
)
config_drive_group = parser.add_mutually_exclusive_group()
config_drive_group.add_argument(
'--config-drive',
action='store_true',
dest='has_config_drive',
default=None,
help=_(
'Only display servers with a config drive attached '
'(admin only before microversion 2.83)'
),
)
# NOTE(gibi): this won't actually do anything until bug 1871409 is
# fixed and the REST API is cleaned up regarding the values of
# config_drive
config_drive_group.add_argument(
'--no-config-drive',
action='store_false',
dest='has_config_drive',
help=_(
'Only display servers without a config drive attached '
'(admin only before microversion 2.83)'
),
)
parser.add_argument(
'--progress',
action=PercentAction,
default=None,
help=_(
'Search by progress value (%%) '
'(admin only before microversion 2.83)'
),
)
parser.add_argument(
'--vm-state',
metavar='<state>',
# taken from 'InstanceState' object field in nova
choices=(
'active',
'building',
'deleted',
'error',
'paused',
'stopped',
'suspended',
'rescued',
'resized',
'shelved',
'shelved_offloaded',
'soft-delete',
),
help=_(
'Search by vm_state value '
'(admin only before microversion 2.83)'
),
)
parser.add_argument(
'--task-state',
metavar='<state>',
# taken from 'InstanceTaskState' object field in nova
choices=(
'block_device_mapping',
'deleting',
'image_backup',
'image_pending_upload',
'image_snapshot',
'image_snapshot_pending',
'image_uploading',
'migrating',
'networking',
'pausing',
'powering-off',
'powering-on',
'rebooting',
'reboot_pending',
'reboot_started',
'reboot_pending_hard',
'reboot_started_hard',
'rebooting_hard',
'rebuilding',
'rebuild_block_device_mapping',
'rebuild_spawning',
'rescuing',
'resize_confirming',
'resize_finish',
'resize_migrated',
'resize_migrating',
'resize_prep',
'resize_reverting',
'restoring',
'resuming',
'scheduling',
'shelving',
'shelving_image_pending_upload',
'shelving_image_uploading',
'shelving_offloading',
'soft-deleting',
'spawning',
'suspending',
'updating_password',
'unpausing',
'unrescuing',
'unshelving',
),
help=_(
'Search by task_state value '
'(admin only before microversion 2.83)'
),
)
parser.add_argument(
'--power-state',
metavar='<state>',
# taken from 'InstancePowerState' object field in nova
choices=(
'pending',
'running',
'paused',
'shutdown',
'crashed',
'suspended',
),
help=_(
'Search by power_state value '
'(admin only before microversion 2.83)'
),
)
parser.add_argument(
'--long',
action='store_true',
default=False,
help=_('List additional fields in output'),
)
name_lookup_group = parser.add_mutually_exclusive_group()
name_lookup_group.add_argument(
'-n',
'--no-name-lookup',
action='store_true',
default=False,
help=_(
'Skip flavor and image name lookup. '
'Mutually exclusive with "--name-lookup-one-by-one" option.'
),
)
name_lookup_group.add_argument(
'--name-lookup-one-by-one',
action='store_true',
default=False,
help=_(
'When looking up flavor and image names, look them up '
'one by one as needed instead of all together (default). '
'Mutually exclusive with "--no-name-lookup|-n" option.'
),
)
pagination.add_marker_pagination_option_to_parser(parser)
parser.add_argument(
'--changes-before',
metavar='<changes-before>',
default=None,
help=_(
"List only servers changed before a certain point of time. "
"The provided time should be an ISO 8061 formatted time "
"(e.g., 2016-03-05T06:27:59Z). "
'(supported by --os-compute-api-version 2.66 or above)'
),
)
parser.add_argument(
'--changes-since',
metavar='<changes-since>',
default=None,
help=_(
"List only servers changed after a certain point of time. "
"The provided time should be an ISO 8061 formatted time "
"(e.g., 2016-03-04T06:27:59Z)."
),
)
lock_group = parser.add_mutually_exclusive_group()
lock_group.add_argument(
'--locked',
action='store_true',
default=False,
help=_(
'Only display locked servers '
'(supported by --os-compute-api-version 2.73 or above)'
),
)
lock_group.add_argument(
'--unlocked',
action='store_true',
default=False,
help=_(
'Only display unlocked servers '
'(supported by --os-compute-api-version 2.73 or above)'
),
)
parser.add_argument(
'--tags',
metavar='<tag>',
action='append',
default=[],
dest='tags',
help=_(
'Only list servers with the specified tag. '
'Specify multiple times to filter on multiple tags. '
'(supported by --os-compute-api-version 2.26 or above)'
),
)
parser.add_argument(
'--not-tags',
metavar='<tag>',
action='append',
default=[],
dest='not_tags',
help=_(
'Only list servers without the specified tag. '
'Specify multiple times to filter on multiple tags. '
'(supported by --os-compute-api-version 2.26 or above)'
),
)
return parser
def take_action(self, parsed_args):
compute_client = self.app.client_manager.sdk_connection.compute
identity_client = self.app.client_manager.identity
image_client = self.app.client_manager.image
project_id = None
if parsed_args.project:
project_id = identity_common.find_project(
identity_client,
parsed_args.project,
parsed_args.project_domain,
).id
parsed_args.all_projects = True
user_id = None
if parsed_args.user:
user_id = identity_common.find_user(
identity_client,
parsed_args.user,
parsed_args.user_domain,
).id
# Nova only supports list servers searching by flavor ID. So if a
# flavor name is given, map it to ID.
flavor_id = None
if parsed_args.flavor:
flavor = compute_client.find_flavor(
parsed_args.flavor, ignore_missing=False
)
flavor_id = flavor.id
# Nova only supports list servers searching by image ID. So if a
# image name is given, map it to ID.
image_id = None
if parsed_args.image:
image_id = image_client.find_image(
parsed_args.image,
ignore_missing=False,
).id
search_opts = {
'reservation_id': parsed_args.reservation_id,
'ip': parsed_args.ip,
'ip6': parsed_args.ip6,
'name': parsed_args.name,
'status': parsed_args.status,
'flavor': flavor_id,
'image': image_id,
'host': parsed_args.host,
'project_id': project_id,
'all_projects': parsed_args.all_projects,
'user_id': user_id,
'deleted': parsed_args.deleted,
'changes-before': parsed_args.changes_before,
'changes-since': parsed_args.changes_since,
}
if parsed_args.instance_name is not None:
search_opts['instance_name'] = parsed_args.instance_name
if parsed_args.availability_zone:
search_opts['availability_zone'] = parsed_args.availability_zone
if parsed_args.key_name:
search_opts['key_name'] = parsed_args.key_name
if parsed_args.has_config_drive is not None:
search_opts['config_drive'] = parsed_args.has_config_drive
if parsed_args.progress is not None:
search_opts['progress'] = str(parsed_args.progress)
if parsed_args.vm_state:
search_opts['vm_state'] = parsed_args.vm_state
if parsed_args.task_state:
search_opts['task_state'] = parsed_args.task_state
if parsed_args.power_state:
# taken from 'InstancePowerState' object field in nova
power_state = {
'pending': 0,
'running': 1,
'paused': 3,
'shutdown': 4,
'crashed': 6,
'suspended': 7,
}[parsed_args.power_state]
search_opts['power_state'] = power_state
if parsed_args.tags:
if not sdk_utils.supports_microversion(compute_client, '2.26'):
msg = _(
'--os-compute-api-version 2.26 or greater is required to '
'support the --tag option'
)
raise exceptions.CommandError(msg)
search_opts['tags'] = ','.join(parsed_args.tags)
if parsed_args.not_tags:
if not sdk_utils.supports_microversion(compute_client, '2.26'):
msg = _(
'--os-compute-api-version 2.26 or greater is required to '
'support the --not-tag option'
)
raise exceptions.CommandError(msg)
search_opts['not-tags'] = ','.join(parsed_args.not_tags)
if parsed_args.locked:
if not sdk_utils.supports_microversion(compute_client, '2.73'):
msg = _(
'--os-compute-api-version 2.73 or greater is required to '
'support the --locked option'
)
raise exceptions.CommandError(msg)
search_opts['locked'] = True
elif parsed_args.unlocked:
if not sdk_utils.supports_microversion(compute_client, '2.73'):
msg = _(
'--os-compute-api-version 2.73 or greater is required to '
'support the --unlocked option'
)
raise exceptions.CommandError(msg)
search_opts['locked'] = False
if parsed_args.limit is not None:
search_opts['limit'] = parsed_args.limit
search_opts['paginated'] = False
LOG.debug('search options: %s', search_opts)
if search_opts['changes-before']:
if not sdk_utils.supports_microversion(compute_client, '2.66'):
msg = _('--os-compute-api-version 2.66 or later is required')
raise exceptions.CommandError(msg)
try:
iso8601.parse_date(search_opts['changes-before'])
except (TypeError, iso8601.ParseError):
raise exceptions.CommandError(
_('Invalid changes-before value: %s')
% search_opts['changes-before']
)
if search_opts['changes-since']:
try:
iso8601.parse_date(search_opts['changes-since'])
except (TypeError, iso8601.ParseError):
msg = _('Invalid changes-since value: %s')
raise exceptions.CommandError(
msg % search_opts['changes-since']
)
columns = (
'id',
'name',
'status',
)
column_headers = (
'ID',
'Name',
'Status',
)
if parsed_args.long:
columns += (
'task_state',
'power_state',
)
column_headers += (
'Task State',
'Power State',
)
columns += ('addresses',)
column_headers += ('Networks',)
if parsed_args.long:
columns += (
'image_name',
'image_id',
)
column_headers += (
'Image Name',
'Image ID',
)
else:
if parsed_args.no_name_lookup:
columns += ('image_id',)
else:
columns += ('image_name',)
column_headers += ('Image',)
# microversion 2.47 puts the embedded flavor into the server response
# body but omits the id, so if not present we just expose the original
# flavor name in the output
if sdk_utils.supports_microversion(compute_client, '2.47'):
columns += ('flavor_name',)
column_headers += ('Flavor',)
else:
if parsed_args.long:
columns += (
'flavor_name',
'flavor_id',
)
column_headers += (
'Flavor Name',
'Flavor ID',
)
else:
if parsed_args.no_name_lookup:
columns += ('flavor_id',)
else:
columns += ('flavor_name',)
column_headers += ('Flavor',)
if parsed_args.long:
columns += (
'availability_zone',
'pinned_availability_zone',
'hypervisor_hostname',
'metadata',
)
column_headers += (
'Availability Zone',
'Pinned Availability Zone',
'Host',
'Properties',
)
# support for additional columns
if parsed_args.columns:
for c in parsed_args.columns:
if c in ('Project ID', 'project_id'):
columns += ('project_id',)
column_headers += ('Project ID',)
if c in ('User ID', 'user_id'):
columns += ('user_id',)
column_headers += ('User ID',)
if c in ('Created At', 'created_at'):
columns += ('created_at',)
column_headers += ('Created At',)
if c in ('Security Groups', 'security_groups'):
columns += ('security_groups_name',)
column_headers += ('Security Groups',)
if c in ("Task State", "task_state"):
columns += ('task_state',)
column_headers += ('Task State',)
if c in ("Power State", "power_state"):
columns += ('power_state',)
column_headers += ('Power State',)
if c in ("Image ID", "image_id"):
columns += ('Image ID',)
column_headers += ('Image ID',)
if c in ("Flavor ID", "flavor_id"):
columns += ('flavor_id',)
column_headers += ('Flavor ID',)
if c in ('Availability Zone', "availability_zone"):
columns += ('availability_zone',)
column_headers += ('Availability Zone',)
if c in (
'pinned_availability_zone',
"Pinned Availability Zone",
):
columns += ('Pinned Availability Zone',)
column_headers += ('Pinned Availability Zone',)
if c in ('Host', "host"):
columns += ('hypervisor_hostname',)
column_headers += ('Host',)
if c in ('Properties', "properties"):
columns += ('Metadata',)
column_headers += ('Properties',)
# remove duplicates
column_headers = tuple(dict.fromkeys(column_headers))
columns = tuple(dict.fromkeys(columns))
if parsed_args.marker is not None:
# Check if both "--marker" and "--deleted" are used.
# In that scenario a lookup is not needed as the marker
# needs to be an ID, because find_resource does not
# handle deleted resources
if parsed_args.deleted:
marker_id = parsed_args.marker
else:
marker_id = compute_client.find_server(
parsed_args.marker, ignore_missing=False
).id
search_opts['marker'] = marker_id
data = list(compute_client.servers(**search_opts))
images = {}
flavors = {}
if data and not parsed_args.no_name_lookup:
# partial responses from down cells will not have an image
# attribute so we use getattr
image_ids = {
s.image['id']
for s in data
if getattr(s, 'image', None) and s.image.get('id')
}
# create a dict that maps image_id to image object, which is used
# to display the "Image Name" column. Note that 'image.id' can be
# empty for BFV instances and 'image' can be missing entirely if
# there are infra failures
if parsed_args.name_lookup_one_by_one or image_id:
for image_id in image_ids:
try:
images[image_id] = image_client.get_image(image_id)
except Exception:
# retrieving image names is not crucial, so we swallow
# any exceptions
pass # nosec: B110
else:
try:
# some deployments can have *loads* of images so we only
# want to list the ones we care about. It would be better
# to only return the *fields* we care about (name) but
# glance doesn't support that
# NOTE(stephenfin): This could result in super long URLs
# but it seems unlikely to cause issues. Apache supports
# URL lengths of up to 8190 characters by default, which
# should allow for more than 220 unique image ID (different
# servers are likely use the same image ID) in the filter.
# Who'd need more than that in a single command?
images_list = image_client.images(
id=f"in:{','.join(image_ids)}"
)
for i in images_list:
images[i.id] = i
except Exception:
# retrieving image names is not crucial, so we swallow any
# exceptions
pass # nosec: B110
# create a dict that maps flavor_id to flavor object, which is used
# to display the "Flavor Name" column. Note that 'flavor.id' is not
# present on microversion 2.47 or later and 'flavor' won't be
# present if there are infra failures
if parsed_args.name_lookup_one_by_one or flavor_id:
for f_id in {
s.flavor['id']
for s in data
if s.flavor and s.flavor.get('id')
}:
try:
flavors[f_id] = compute_client.find_flavor(
f_id, ignore_missing=False
)
except Exception:
# retrieving flavor names is not crucial, so we swallow
# any exceptions
pass # nosec: B110
else:
try:
flavors_list = compute_client.flavors(is_public=None)
for i in flavors_list:
flavors[i.id] = i
except Exception:
# retrieving flavor names is not crucial, so we swallow any
# exceptions
pass # nosec: B110
# Populate image_name, image_id, flavor_name and flavor_id attributes
# of server objects so that we can display those columns.
for s in data:
if sdk_utils.supports_microversion(compute_client, '2.69'):
# NOTE(tssurya): From 2.69, we will have the keys 'flavor'
# and 'image' missing in the server response during
# infrastructure failure situations.
# For those servers with partial constructs we just skip the
# processing of the image and flavor information.
if not hasattr(s, 'image') or not hasattr(s, 'flavor'):
continue
if 'id' in s.image and s.image.id is not None:
image = images.get(s.image['id'])
if image:
s.image_name = image.name
s.image_id = s.image['id']
else:
# NOTE(melwitt): An server booted from a volume will have no
# image associated with it. We fill in the Image Name and ID
# with "N/A (booted from volume)" to help users who want to be
# able to grep for boot-from-volume servers when using the CLI.
s.image_name = IMAGE_STRING_FOR_BFV
s.image_id = IMAGE_STRING_FOR_BFV
if not sdk_utils.supports_microversion(compute_client, '2.47'):
flavor = flavors.get(s.flavor['id'])
if flavor:
s.flavor_name = flavor.name
s.flavor_id = s.flavor['id']
else:
s.flavor_name = s.flavor['original_name']
# Add a list with security group name as attribute
for s in data:
if hasattr(s, 'security_groups') and s.security_groups is not None:
s.security_groups_name = [x["name"] for x in s.security_groups]
else:
s.security_groups_name = []
# The host_status field contains the status of the compute host the
# server is on. It is only returned by the API when the nova-api
# policy allows. Users can look at the host_status field when, for
# example, their server has status ACTIVE but is unresponsive. The
# host_status field can indicate a possible problem on the host
# it's on, providing useful information to a user in this
# situation.
if (
sdk_utils.supports_microversion(compute_client, '2.16')
and parsed_args.long
):
if any([s.host_status is not None for s in data]):
columns += ('Host Status',)
column_headers += ('Host Status',)
table = (
column_headers,
(
utils.get_item_properties(
s,
columns,
mixed_case_fields=(
'task_state',
'power_state',
'availability_zone',
'host',
),
formatters={
'power_state': PowerStateColumn,
'addresses': AddressesColumn,
'metadata': format_columns.DictColumn,
'security_groups_name': format_columns.ListColumn,
'hypervisor_hostname': HostColumn,
},
)
for s in data
),
)
return table
class LockServer(command.Command):
_description = _(
"""Lock server(s)
A non-admin user will not be able to execute actions."""
)
def get_parser(self, prog_name):
parser = super().get_parser(prog_name)
parser.add_argument(
'server',
metavar='<server>',
nargs='+',
help=_('Server(s) to lock (name or ID)'),
)
parser.add_argument(
'--reason',
metavar='<reason>',
default=None,
help=_(
'Reason for locking the server(s) '
'(supported by --os-compute-api-version 2.73 or above)'
),
)
return parser
def take_action(self, parsed_args):
compute_client = self.app.client_manager.sdk_connection.compute
kwargs = {}
if parsed_args.reason:
if not sdk_utils.supports_microversion(compute_client, '2.73'):
msg = _(
'--os-compute-api-version 2.73 or greater is required to '
'use the --reason option'
)
raise exceptions.CommandError(msg)
kwargs['locked_reason'] = parsed_args.reason
for server in parsed_args.server:
server_id = compute_client.find_server(
server, ignore_missing=False
).id
compute_client.lock_server(server_id, **kwargs)
# FIXME(dtroyer): Here is what I want, how with argparse/cliff?
# server migrate [--wait] \
# [--live <hostname>
# [--shared-migration | --block-migration]
# [--disk-overcommit | --no-disk-overcommit]]
# <server>
#
# live_parser = parser.add_argument_group(title='Live migration options')
# then adding the groups doesn't seem to work
class MigrateServer(command.Command):
_description = _(
"""Migrate server to different host.
A migrate operation is implemented as a resize operation using the same flavor
as the old server. This means that, like resize, migrate works by creating a
new server using the same flavor and copying the contents of the original disk
into a new one. As with resize, the migrate operation is a two-step process for
the user: the first step is to perform the migrate, and the second step is to
either confirm (verify) success and release the old server, or to declare a
revert to release the new server and restart the old one."""
)
def get_parser(self, prog_name):
parser = super().get_parser(prog_name)
parser.add_argument(
'server',
metavar='<server>',
help=_('Server (name or ID)'),
)
parser.add_argument(
'--live-migration',
dest='live_migration',
action='store_true',
help=_(
'Live migrate the server; use the ``--host`` option to '
'specify a target host for the migration which will be '
'validated by the scheduler'
),
)
parser.add_argument(
'--host',
metavar='<hostname>',
help=_(
'Migrate the server to the specified host. '
'(supported with --os-compute-api-version 2.30 or above '
'when used with the --live-migration option) '
'(supported with --os-compute-api-version 2.56 or above '
'when used without the --live-migration option)'
),
)
migration_group = parser.add_mutually_exclusive_group()
migration_group.add_argument(
'--shared-migration',
dest='block_migration',
action='store_false',
default=None,
help=_(
'Perform a shared live migration '
'(default before --os-compute-api-version 2.25, auto after)'
),
)
migration_group.add_argument(
'--block-migration',
dest='block_migration',
action='store_true',
help=_(
'Perform a block live migration '
'(auto-configured from --os-compute-api-version 2.25)'
),
)
disk_group = parser.add_mutually_exclusive_group()
disk_group.add_argument(
'--disk-overcommit',
action='store_true',
default=None,
help=_(
'Allow disk over-commit on the destination host'
'(supported with --os-compute-api-version 2.24 or below)'
),
)
disk_group.add_argument(
'--no-disk-overcommit',
dest='disk_overcommit',
action='store_false',
help=_(
'Do not over-commit disk on the destination host (default)'
'(supported with --os-compute-api-version 2.24 or below)'
),
)
parser.add_argument(
'--wait',
action='store_true',
help=_('Wait for migrate to complete'),
)
return parser
def take_action(self, parsed_args):
def _show_progress(progress):
if progress:
self.app.stdout.write('\rProgress: %s' % progress)
self.app.stdout.flush()
compute_client = self.app.client_manager.sdk_connection.compute
server = compute_client.find_server(
parsed_args.server, ignore_missing=False
)
if parsed_args.live_migration:
kwargs = {}
block_migration = parsed_args.block_migration
if block_migration is None:
if not sdk_utils.supports_microversion(compute_client, '2.25'):
block_migration = False
else:
block_migration = 'auto'
kwargs['block_migration'] = block_migration
# Technically we could pass a non-None host with
# --os-compute-api-version < 2.30 but that is the same thing
# as the --live option bypassing the scheduler which we don't
# want to support, so if the user is using --live-migration
# and --host, we want to enforce that they are using version
# 2.30 or greater.
if parsed_args.host and not sdk_utils.supports_microversion(
compute_client, '2.30'
):
raise exceptions.CommandError(
'--os-compute-api-version 2.30 or greater is required '
'when using --host'
)
# The host parameter is required in the API even if None.
kwargs['host'] = parsed_args.host
if not sdk_utils.supports_microversion(compute_client, '2.25'):
kwargs['disk_overcommit'] = parsed_args.disk_overcommit
# We can't use an argparse default value because then we can't
# distinguish between explicit 'False' and unset for the below
# case (microversion >= 2.25)
if kwargs['disk_overcommit'] is None:
kwargs['disk_overcommit'] = False
elif parsed_args.disk_overcommit is not None:
# TODO(stephenfin): Raise an error here in OSC 7.0
msg = _(
'The --disk-overcommit and --no-disk-overcommit '
'options are only supported by '
'--os-compute-api-version 2.24 or below; this will '
'be an error in a future release'
)
self.log.warning(msg)
compute_client.live_migrate_server(server, **kwargs)
else: # cold migration
if parsed_args.block_migration or parsed_args.disk_overcommit:
raise exceptions.CommandError(
"--live-migration must be specified if "
"--block-migration or --disk-overcommit is "
"specified"
)
if parsed_args.host:
if not sdk_utils.supports_microversion(compute_client, '2.56'):
msg = _(
'--os-compute-api-version 2.56 or greater is '
'required to use --host without --live-migration.'
)
raise exceptions.CommandError(msg)
kwargs = {'host': parsed_args.host} if parsed_args.host else {}
compute_client.migrate_server(server, **kwargs)
if parsed_args.wait:
if utils.wait_for_status(
compute_client.get_server,
server.id,
success_status=('active', 'verify_resize'),
callback=_show_progress,
):
self.app.stdout.write(
_(
'Complete, check success/failure by '
'openstack server migration/event list/show\n'
)
)
else:
msg = _('Error migrating server: %s') % server.id
raise exceptions.CommandError(msg)
class PauseServer(command.Command):
_description = _("Pause server(s)")
def get_parser(self, prog_name):
parser = super().get_parser(prog_name)
parser.add_argument(
'server',
metavar='<server>',
nargs='+',
help=_('Server(s) to pause (name or ID)'),
)
return parser
def take_action(self, parsed_args):
compute_client = self.app.client_manager.sdk_connection.compute
for server in parsed_args.server:
server_id = compute_client.find_server(
server,
ignore_missing=False,
).id
compute_client.pause_server(server_id)
class RebootServer(command.Command):
_description = _("Perform a hard or soft server reboot")
def get_parser(self, prog_name):
parser = super().get_parser(prog_name)
parser.add_argument(
'server',
metavar='<server>',
help=_('Server (name or ID)'),
)
group = parser.add_mutually_exclusive_group()
group.add_argument(
'--hard',
dest='reboot_type',
action='store_const',
const='HARD',
default='SOFT',
help=_('Perform a hard reboot'),
)
group.add_argument(
'--soft',
dest='reboot_type',
action='store_const',
const='SOFT',
default='SOFT',
help=_('Perform a soft reboot'),
)
parser.add_argument(
'--wait',
action='store_true',
help=_('Wait for reboot to complete'),
)
return parser
def take_action(self, parsed_args):
def _show_progress(progress):
if progress:
self.app.stdout.write('\rProgress: %s' % progress)
self.app.stdout.flush()
compute_client = self.app.client_manager.sdk_connection.compute
server_id = compute_client.find_server(
parsed_args.server,
ignore_missing=False,
).id
compute_client.reboot_server(server_id, parsed_args.reboot_type)
if parsed_args.wait:
# We use osc-lib's wait_for_status since that allows for a callback
if utils.wait_for_status(
compute_client.get_server,
server_id,
callback=_show_progress,
):
self.app.stdout.write(_('Complete\n'))
else:
msg = _('Error rebooting server: %s') % server_id
raise exceptions.CommandError(msg)
class RebuildServer(command.ShowOne):
_description = _("Rebuild server")
def get_parser(self, prog_name):
parser = super().get_parser(prog_name)
parser.add_argument(
'server',
metavar='<server>',
help=_('Server (name or ID)'),
)
parser.add_argument(
'--image',
metavar='<image>',
help=_(
'Recreate server from the specified image (name or ID).'
'Defaults to the currently used one.'
),
)
parser.add_argument(
'--name',
metavar='<name>',
help=_('Set the new name of the rebuilt server'),
)
parser.add_argument(
'--password',
metavar='<password>',
help=_(
'Set the password on the rebuilt server. '
'This option requires cloud support.'
),
)
parser.add_argument(
'--property',
metavar='<key=value>',
action=parseractions.KeyValueAction,
dest='properties',
help=_(
'Set a new property on the rebuilt server '
'(repeat option to set multiple values)'
),
)
parser.add_argument(
'--description',
metavar='<description>',
help=_(
'Set a new description on the rebuilt server '
'(supported by --os-compute-api-version 2.19 or above)'
),
)
preserve_ephemeral_group = parser.add_mutually_exclusive_group()
preserve_ephemeral_group.add_argument(
'--preserve-ephemeral',
action='store_true',
default=None,
help=_(
'Preserve the default ephemeral storage partition on rebuild.'
),
)
preserve_ephemeral_group.add_argument(
'--no-preserve-ephemeral',
action='store_false',
dest='preserve_ephemeral',
help=_(
'Do not preserve the default ephemeral storage partition on '
'rebuild.'
),
)
key_group = parser.add_mutually_exclusive_group()
key_group.add_argument(
'--key-name',
metavar='<key-name>',
help=_(
'Set the key name of key pair on the rebuilt server. '
'Cannot be specified with the --key-unset option. '
'(supported by --os-compute-api-version 2.54 or above)'
),
)
key_group.add_argument(
'--no-key-name',
action='store_true',
dest='no_key_name',
help=_(
'Unset the key name of key pair on the rebuilt server. '
'Cannot be specified with the --key-name option. '
'(supported by --os-compute-api-version 2.54 or above)'
),
)
# TODO(stephenfin): Remove this in a future major version bump
key_group.add_argument(
'--key-unset',
action='store_true',
dest='no_key_name',
help=argparse.SUPPRESS,
)
user_data_group = parser.add_mutually_exclusive_group()
user_data_group.add_argument(
'--user-data',
metavar='<user-data>',
help=_(
'Add a new user data file to the rebuilt server. '
'Cannot be specified with the --no-user-data option. '
'(supported by --os-compute-api-version 2.57 or above)'
),
)
user_data_group.add_argument(
'--no-user-data',
action='store_true',
default=False,
help=_(
'Remove existing user data when rebuilding server. '
'Cannot be specified with the --user-data option. '
'(supported by --os-compute-api-version 2.57 or above)'
),
)
trusted_certs_group = parser.add_mutually_exclusive_group()
trusted_certs_group.add_argument(
'--trusted-image-cert',
metavar='<trusted-cert-id>',
action='append',
dest='trusted_image_certs',
help=_(
'Trusted image certificate IDs used to validate certificates '
'during the image signature verification process. '
'May be specified multiple times to pass multiple trusted '
'image certificate IDs. '
'Cannot be specified with the --no-trusted-certs option. '
'(supported by --os-compute-api-version 2.63 or above)'
),
)
trusted_certs_group.add_argument(
'--no-trusted-image-certs',
action='store_true',
default=False,
help=_(
'Remove any existing trusted image certificates from the '
'server. '
'Cannot be specified with the --trusted-certs option. '
'(supported by --os-compute-api-version 2.63 or above)'
),
)
parser.add_argument(
'--hostname',
metavar='<hostname>',
help=_(
'Hostname configured for the server in the metadata service. '
'A separate utility running in the guest is required to '
'propagate changes to this value to the guest OS itself. '
'(supported by --os-compute-api-version 2.90 or above)'
),
)
parser.add_argument(
'--reimage-boot-volume',
action='store_true',
dest='reimage_boot_volume',
default=None,
help=_(
'Rebuild a volume-backed server. This will wipe the root '
'volume data and overwrite it with the provided image. '
'Defaults to False. '
'(supported by --os-compute-api-version 2.93 or above)'
),
)
parser.add_argument(
'--no-reimage-boot-volume',
action='store_false',
dest='reimage_boot_volume',
default=None,
help=_(
'Do not rebuild a volume-backed server. '
'(supported by --os-compute-api-version 2.93 or above)'
),
)
parser.add_argument(
'--wait',
action='store_true',
help=_('Wait for rebuild to complete'),
)
return parser
def take_action(self, parsed_args):
def _show_progress(progress):
if progress:
self.app.stdout.write('\rProgress: %s' % progress)
self.app.stdout.flush()
compute_client = self.app.client_manager.sdk_connection.compute
image_client = self.app.client_manager.image
server = compute_client.find_server(
parsed_args.server, ignore_missing=False
)
# If parsed_args.image is not set and if the instance is image backed,
# default to the currently used one. If the instance is volume backed,
# it is not trivial to fetch the current image and probably better
# to error out in this case and ask user to supply the image.
if parsed_args.image:
image = image_client.find_image(
parsed_args.image, ignore_missing=False
)
else:
if not server.image or not server.image.id:
msg = _(
'The --image option is required when rebuilding a '
'volume-backed server'
)
raise exceptions.CommandError(msg)
image = image_client.get_image(server.image['id'])
kwargs = {}
if parsed_args.name is not None:
kwargs['name'] = parsed_args.name
if parsed_args.preserve_ephemeral is not None:
kwargs['preserve_ephemeral'] = parsed_args.preserve_ephemeral
if parsed_args.properties:
kwargs['metadata'] = parsed_args.properties
if parsed_args.description:
if not sdk_utils.supports_microversion(compute_client, '2.19'):
msg = _(
'--os-compute-api-version 2.19 or greater is required to '
'support the --description option'
)
raise exceptions.CommandError(msg)
kwargs['description'] = parsed_args.description
if parsed_args.key_name:
if not sdk_utils.supports_microversion(compute_client, '2.54'):
msg = _(
'--os-compute-api-version 2.54 or greater is required to '
'support the --key-name option'
)
raise exceptions.CommandError(msg)
kwargs['key_name'] = parsed_args.key_name
elif parsed_args.no_key_name:
if not sdk_utils.supports_microversion(compute_client, '2.54'):
msg = _(
'--os-compute-api-version 2.54 or greater is required to '
'support the --no-key-name option'
)
raise exceptions.CommandError(msg)
kwargs['key_name'] = None
if parsed_args.user_data:
if not sdk_utils.supports_microversion(compute_client, '2.54'):
msg = _(
'--os-compute-api-version 2.54 or greater is required to '
'support the --user-data option'
)
raise exceptions.CommandError(msg)
try:
with open(parsed_args.user_data, 'rb') as fh:
# TODO(stephenfin): SDK should do this for us
user_data = base64.b64encode(fh.read()).decode('utf-8')
except OSError as e:
msg = _("Can't open '%(data)s': %(exception)s")
raise exceptions.CommandError(
msg % {'data': parsed_args.user_data, 'exception': e}
)
kwargs['user_data'] = user_data
elif parsed_args.no_user_data:
if not sdk_utils.supports_microversion(compute_client, '2.54'):
msg = _(
'--os-compute-api-version 2.54 or greater is required to '
'support the --no-user-data option'
)
raise exceptions.CommandError(msg)
kwargs['user_data'] = None
# TODO(stephenfin): Handle OS_TRUSTED_IMAGE_CERTIFICATE_IDS
if parsed_args.trusted_image_certs:
if not sdk_utils.supports_microversion(compute_client, '2.63'):
msg = _(
'--os-compute-api-version 2.63 or greater is required to '
'support the --trusted-certs option'
)
raise exceptions.CommandError(msg)
certs = parsed_args.trusted_image_certs
kwargs['trusted_image_certificates'] = certs
elif parsed_args.no_trusted_image_certs:
if not sdk_utils.supports_microversion(compute_client, '2.63'):
msg = _(
'--os-compute-api-version 2.63 or greater is required to '
'support the --no-trusted-certs option'
)
raise exceptions.CommandError(msg)
kwargs['trusted_image_certificates'] = None
if parsed_args.hostname:
if not sdk_utils.supports_microversion(compute_client, '2.90'):
msg = _(
'--os-compute-api-version 2.90 or greater is required to '
'support the --hostname option'
)
raise exceptions.CommandError(msg)
kwargs['hostname'] = parsed_args.hostname
if parsed_args.reimage_boot_volume:
if not sdk_utils.supports_microversion(compute_client, '2.93'):
msg = _(
'--os-compute-api-version 2.93 or greater is required to '
'support the --reimage-boot-volume option'
)
raise exceptions.CommandError(msg)
else:
# force user to explicitly request reimaging of volume-backed
# server
if not server.image or not server.image.id:
if sdk_utils.supports_microversion(compute_client, '2.93'):
msg = (
'--reimage-boot-volume is required to rebuild a '
'volume-backed server'
)
raise exceptions.CommandError(msg)
else: # microversion < 2.93
# attempts to rebuild a volume-backed server before API
# microversion 2.93 will fail in all cases except one: if
# the user attempts the rebuild with the exact same image
# that the server was initially built with. We can't check
# for this since we don't have the original image ID to
# hand, so we simply warn the user.
# TODO(stephenfin): Make this a failure in a future
# version
self.log.warning(
'Attempting to rebuild a volume-backed server using '
'--os-compute-api-version 2.92 or earlier, which '
'will only succeed if the image is identical to the '
'one initially used. This will be an error in a '
'future release.'
)
status = getattr(server, 'status', '').lower()
if status == 'shutoff':
success_status = ['shutoff']
elif status in ('error', 'active'):
success_status = ['active']
else:
msg = _("The server status is not ACTIVE, SHUTOFF or ERROR.")
raise exceptions.CommandError(msg)
server = compute_client.rebuild_server(
server, image, admin_password=parsed_args.password, **kwargs
)
if parsed_args.wait:
if utils.wait_for_status(
compute_client.get_server,
server.id,
callback=_show_progress,
success_status=success_status,
):
self.app.stdout.write(_('Complete\n'))
else:
msg = _('Error rebuilding server: %s') % server.id
raise exceptions.CommandError(msg)
data = _prep_server_detail(
compute_client, image_client, server, refresh=False
)
return zip(*sorted(data.items()))
class EvacuateServer(command.ShowOne):
_description = _(
"""Evacuate a server to a different host.
This command is used to recreate a server after the host it was on has failed.
It can only be used if the compute service that manages the server is down.
This command should only be used by an admin after they have confirmed that the
instance is not running on the failed host.
If the server instance was created with an ephemeral root disk on non-shared
storage the server will be rebuilt using the original glance image preserving
the ports and any attached data volumes.
If the server uses boot for volume or has its root disk on shared storage the
root disk will be preserved and reused for the evacuated instance on the new
host."""
)
def get_parser(self, prog_name):
parser = super().get_parser(prog_name)
parser.add_argument(
'server',
metavar='<server>',
help=_('Server (name or ID)'),
)
parser.add_argument(
'--wait',
action='store_true',
help=_('Wait for evacuation to complete'),
)
parser.add_argument(
'--host',
metavar='<host>',
default=None,
help=_(
'Set the preferred host on which to rebuild the evacuated '
'server. The host will be validated by the scheduler. '
'(supported by --os-compute-api-version 2.29 or above)'
),
)
shared_storage_group = parser.add_mutually_exclusive_group()
shared_storage_group.add_argument(
'--password',
metavar='<password>',
default=None,
help=_(
'Set the password on the evacuated instance. This option is '
'mutually exclusive with the --shared-storage option. '
'This option requires cloud support.'
),
)
shared_storage_group.add_argument(
'--shared-storage',
action='store_true',
dest='shared_storage',
help=_(
'Indicate that the instance is on shared storage. '
'This will be auto-calculated with '
'--os-compute-api-version 2.14 and greater and should not '
'be used with later microversions. This option is mutually '
'exclusive with the --password option'
),
)
return parser
def take_action(self, parsed_args):
def _show_progress(progress):
if progress:
self.app.stdout.write('\rProgress: %s' % progress)
self.app.stdout.flush()
compute_client = self.app.client_manager.sdk_connection.compute
image_client = self.app.client_manager.image
if parsed_args.host:
if not sdk_utils.supports_microversion(compute_client, '2.29'):
msg = _(
'--os-compute-api-version 2.29 or later is required '
'to specify a preferred host.'
)
raise exceptions.CommandError(msg)
if parsed_args.shared_storage:
if sdk_utils.supports_microversion(compute_client, '2.14'):
msg = _(
'--os-compute-api-version 2.13 or earlier is required '
'to specify shared-storage.'
)
raise exceptions.CommandError(msg)
kwargs = {
'host': parsed_args.host,
'password': parsed_args.password,
}
if not sdk_utils.supports_microversion(compute_client, '2.14'):
kwargs['on_shared_storage'] = parsed_args.shared_storage
server = compute_client.find_server(
parsed_args.server, ignore_missing=False
)
compute_client.evacuate_server(server, **kwargs)
if parsed_args.wait:
if utils.wait_for_status(
compute_client.get_server,
server.id,
callback=_show_progress,
):
self.app.stdout.write(_('Complete\n'))
else:
msg = _('Error evacuating server: %s') % server.id
raise exceptions.CommandError(msg)
data = _prep_server_detail(compute_client, image_client, server)
return zip(*sorted(data.items()))
class RemoveFixedIP(command.Command):
_description = _("Remove fixed IP address from server")
def get_parser(self, prog_name):
parser = super().get_parser(prog_name)
parser.add_argument(
"server",
metavar="<server>",
help=_("Server to remove the fixed IP address from (name or ID)"),
)
parser.add_argument(
"ip_address",
metavar="<ip-address>",
help=_("Fixed IP address to remove from the server (IP only)"),
)
return parser
def take_action(self, parsed_args):
compute_client = self.app.client_manager.sdk_connection.compute
server = compute_client.find_server(
parsed_args.server, ignore_missing=False
)
compute_client.remove_fixed_ip_from_server(
server, parsed_args.ip_address
)
class RemoveFloatingIP(network_common.NetworkAndComputeCommand):
_description = _("Remove floating IP address from server")
def update_parser_common(self, parser):
parser.add_argument(
"server",
metavar="<server>",
help=_(
"Server to remove the floating IP address from (name or ID)"
),
)
parser.add_argument(
"ip_address",
metavar="<ip-address>",
help=_("Floating IP address to remove from server (IP only)"),
)
return parser
def take_action_network(self, client, parsed_args):
attrs = {}
obj = client.find_ip(
parsed_args.ip_address,
ignore_missing=False,
)
attrs['port_id'] = None
client.update_ip(obj, **attrs)
def take_action_compute(self, client, parsed_args):
server = client.find_server(parsed_args.server, ignore_missing=False)
client.remove_floating_ip_from_server(server, parsed_args.ip_address)
class RemovePort(command.Command):
_description = _("Remove port from server")
def get_parser(self, prog_name):
parser = super().get_parser(prog_name)
parser.add_argument(
"server",
metavar="<server>",
help=_("Server to remove the port from (name or ID)"),
)
parser.add_argument(
"port",
metavar="<port>",
help=_("Port to remove from the server (name or ID)"),
)
return parser
def take_action(self, parsed_args):
compute_client = self.app.client_manager.sdk_connection.compute
server = compute_client.find_server(
parsed_args.server, ignore_missing=False
)
if self.app.client_manager.is_network_endpoint_enabled():
network_client = self.app.client_manager.network
port_id = network_client.find_port(
parsed_args.port, ignore_missing=False
).id
else:
port_id = parsed_args.port
compute_client.delete_server_interface(
port_id,
server=server,
ignore_missing=False,
)
class RemoveNetwork(command.Command):
_description = _("Remove all ports of a network from server")
def get_parser(self, prog_name):
parser = super().get_parser(prog_name)
parser.add_argument(
"server",
metavar="<server>",
help=_("Server to remove the port from (name or ID)"),
)
parser.add_argument(
"network",
metavar="<network>",
help=_("Network to remove from the server (name or ID)"),
)
return parser
def take_action(self, parsed_args):
compute_client = self.app.client_manager.sdk_connection.compute
server = compute_client.find_server(
parsed_args.server, ignore_missing=False
)
if self.app.client_manager.is_network_endpoint_enabled():
network_client = self.app.client_manager.network
net_id = network_client.find_network(
parsed_args.network, ignore_missing=False
).id
else:
net_id = parsed_args.network
for inf in compute_client.server_interfaces(server):
if inf.net_id == net_id:
compute_client.delete_server_interface(
inf.port_id,
server=server,
)
class RemoveServerSecurityGroup(command.Command):
_description = _("Remove security group from server")
def get_parser(self, prog_name):
parser = super().get_parser(prog_name)
parser.add_argument(
'server',
metavar='<server>',
help=_('Server (name or ID)'),
)
parser.add_argument(
'group',
metavar='<group>',
help=_('Security group to remove (name or ID)'),
)
return parser
def take_action(self, parsed_args):
compute_client = self.app.client_manager.sdk_connection.compute
server = compute_client.find_server(
parsed_args.server, ignore_missing=False
)
if self.app.client_manager.is_network_endpoint_enabled():
# the server handles both names and IDs for neutron SGs, so just
# pass things through
security_group = parsed_args.group
else:
# however, if using nova-network then it needs a name, not an ID
security_group = compute_v2.find_security_group(
compute_client, parsed_args.group
)['name']
compute_client.remove_security_group_from_server(
server, security_group
)
class RemoveServerVolume(command.Command):
_description = _(
"""Remove volume from server.
Specify ``--os-compute-api-version 2.20`` or higher to remove a
volume from a server with status ``SHELVED`` or ``SHELVED_OFFLOADED``."""
)
def get_parser(self, prog_name):
parser = super().get_parser(prog_name)
parser.add_argument(
'server',
metavar='<server>',
help=_('Server (name or ID)'),
)
parser.add_argument(
'volume',
metavar='<volume>',
help=_('Volume to remove (name or ID)'),
)
return parser
def take_action(self, parsed_args):
compute_client = self.app.client_manager.sdk_connection.compute
volume_client = self.app.client_manager.sdk_connection.volume
server = compute_client.find_server(
parsed_args.server,
ignore_missing=False,
)
volume = volume_client.find_volume(
parsed_args.volume,
ignore_missing=False,
)
compute_client.delete_volume_attachment(
volume,
server,
ignore_missing=False,
)
class RescueServer(command.Command):
_description = _(
"""Put server in rescue mode.
Specify ``--os-compute-api-version 2.87`` or higher to rescue a
server booted from a volume."""
)
def get_parser(self, prog_name):
parser = super().get_parser(prog_name)
parser.add_argument(
'server',
metavar='<server>',
help=_('Server (name or ID)'),
)
parser.add_argument(
'--image',
metavar='<image>',
help=_(
'Image (name or ID) to use for the rescue mode '
'(defaults to the currently used one)'
),
)
parser.add_argument(
'--password',
metavar='<password>',
help=_(
'Set the password on the rescued instance '
'(requires cloud support)'
),
)
return parser
def take_action(self, parsed_args):
compute_client = self.app.client_manager.sdk_connection.compute
image_client = self.app.client_manager.image
image_ref = None
if parsed_args.image:
image_ref = image_client.find_image(
parsed_args.image, ignore_missing=False
).id
server = compute_client.find_server(
parsed_args.server, ignore_missing=False
)
compute_client.rescue_server(
server, admin_pass=parsed_args.password, image_ref=image_ref
)
class ResizeServer(command.Command):
_description = _(
"""Scale server to a new flavor.
A resize operation is implemented by creating a new server and copying the
contents of the original disk into a new one. It is a two-step process for the
user: the first step is to perform the resize, and the second step is to either
confirm (verify) success and release the old server or to declare a revert to
release the new server and restart the old one."""
)
def get_parser(self, prog_name):
parser = super().get_parser(prog_name)
parser.add_argument(
'server',
metavar='<server>',
help=_('Server (name or ID)'),
)
phase_group = parser.add_mutually_exclusive_group()
phase_group.add_argument(
'--flavor',
metavar='<flavor>',
help=_('Resize server to specified flavor'),
)
phase_group.add_argument(
'--confirm',
action="store_true",
help=_(
"**Deprecated** Confirm server resize is complete. "
"Replaced by the 'openstack server resize confirm' and "
"'openstack server migration confirm' commands"
),
)
phase_group.add_argument(
'--revert',
action="store_true",
help=_(
'**Deprecated** Restore server state before resize'
"Replaced by the 'openstack server resize revert' and "
"'openstack server migration revert' commands"
),
)
parser.add_argument(
'--wait',
action='store_true',
help=_('Wait for resize to complete'),
)
return parser
def take_action(self, parsed_args):
def _show_progress(progress):
if progress:
self.app.stdout.write('\rProgress: %s' % progress)
self.app.stdout.flush()
compute_client = self.app.client_manager.sdk_connection.compute
server = compute_client.find_server(
parsed_args.server, ignore_missing=False
)
if parsed_args.flavor:
if not server.image:
self.log.warning(
_(
"The root disk size in flavor will not be applied "
"while booting from a persistent volume."
)
)
flavor = compute_client.find_flavor(
parsed_args.flavor, ignore_missing=False
)
compute_client.resize_server(server, flavor)
if parsed_args.wait:
if not utils.wait_for_status(
compute_client.get_server,
server.id,
success_status=('active', 'verify_resize'),
callback=_show_progress,
):
msg = _('Error resizing server: %s') % server.id
raise exceptions.CommandError(msg)
self.app.stdout.write(_('Complete\n'))
elif parsed_args.confirm:
self.log.warning(
_(
"The --confirm option has been deprecated. Please use the "
"'openstack server resize confirm' command instead."
)
)
compute_client.confirm_server_resize(server)
elif parsed_args.revert:
self.log.warning(
_(
"The --revert option has been deprecated. Please use the "
"'openstack server resize revert' command instead."
)
)
compute_client.revert_server_resize(server)
class ResizeConfirm(command.Command):
_description = _(
"""Confirm server resize.
Confirm (verify) success of resize operation and release the old server."""
)
def get_parser(self, prog_name):
parser = super().get_parser(prog_name)
parser.add_argument(
'server',
metavar='<server>',
help=_('Server (name or ID)'),
)
return parser
def take_action(self, parsed_args):
compute_client = self.app.client_manager.sdk_connection.compute
server = compute_client.find_server(
parsed_args.server, ignore_missing=False
)
compute_client.confirm_server_resize(server)
# TODO(stephenfin): Remove in OSC 7.0
class MigrateConfirm(ResizeConfirm):
_description = _("DEPRECATED: Use 'server migration confirm' instead.")
def take_action(self, parsed_args):
msg = _(
"The 'server migrate confirm' command has been deprecated in "
"favour of the 'server migration confirm' command."
)
self.log.warning(msg)
super().take_action(parsed_args)
class ConfirmMigration(ResizeConfirm):
_description = _(
"""Confirm server migration.
Confirm (verify) success of the migration operation and release the old
server."""
)
class ResizeRevert(command.Command):
_description = _(
"""Revert server resize.
Revert the resize operation. Release the new server and restart the old
one."""
)
def get_parser(self, prog_name):
parser = super().get_parser(prog_name)
parser.add_argument(
'server',
metavar='<server>',
help=_('Server (name or ID)'),
)
return parser
def take_action(self, parsed_args):
compute_client = self.app.client_manager.sdk_connection.compute
server = compute_client.find_server(
parsed_args.server, ignore_missing=False
)
compute_client.revert_server_resize(server)
# TODO(stephenfin): Remove in OSC 7.0
class MigrateRevert(ResizeRevert):
_description = _("DEPRECATED: Use 'server migration revert' instead.")
def take_action(self, parsed_args):
msg = _(
"The 'server migrate revert' command has been deprecated in "
"favour of the 'server migration revert' command."
)
self.log.warning(msg)
super().take_action(parsed_args)
class RevertMigration(ResizeRevert):
_description = _(
"""Revert server migration.
Revert the migration operation. Release the new server and restart the old
one."""
)
class RestoreServer(command.Command):
_description = _("Restore server(s)")
def get_parser(self, prog_name):
parser = super().get_parser(prog_name)
parser.add_argument(
'server',
metavar='<server>',
nargs='+',
help=_('Server(s) to restore (name or ID)'),
)
return parser
def take_action(self, parsed_args):
compute_client = self.app.client_manager.sdk_connection.compute
for server in parsed_args.server:
server_id = compute_client.find_server(
server,
ignore_missing=False,
).id
compute_client.restore_server(server_id)
class ResumeServer(command.Command):
_description = _("Resume server(s)")
def get_parser(self, prog_name):
parser = super().get_parser(prog_name)
parser.add_argument(
'server',
metavar='<server>',
nargs='+',
help=_('Server(s) to resume (name or ID)'),
)
return parser
def take_action(self, parsed_args):
compute_client = self.app.client_manager.sdk_connection.compute
for server in parsed_args.server:
server_id = compute_client.find_server(
server,
ignore_missing=False,
).id
compute_client.resume_server(server_id)
class SetServer(command.Command):
_description = _("Set server properties")
def get_parser(self, prog_name):
parser = super().get_parser(prog_name)
parser.add_argument(
'server',
metavar='<server>',
help=_('Server (name or ID)'),
)
parser.add_argument(
'--name',
metavar='<new-name>',
help=_('New server name'),
)
password_group = parser.add_mutually_exclusive_group()
password_group.add_argument(
'--password',
help=_(
'Set the server password. '
'This option requires cloud support.'
),
)
password_group.add_argument(
'--no-password',
action='store_true',
help=_(
'Clear the admin password for the server from the metadata '
'service; note that this action does not actually change the '
'server password'
),
)
# TODO(stephenfin): Remove this in a future major version
password_group.add_argument(
'--root-password',
action="store_true",
help=argparse.SUPPRESS,
)
parser.add_argument(
'--property',
metavar='<key=value>',
action=parseractions.KeyValueAction,
dest='properties',
help=_(
'Property to add/change for this server '
'(repeat option to set multiple properties)'
),
)
parser.add_argument(
'--state',
metavar='<state>',
choices=['active', 'error'],
help=_(
'New server state '
'**WARNING** This can result in instances that are no longer '
'usable and should be used with caution '
'(admin only)'
),
)
parser.add_argument(
'--description',
metavar='<description>',
help=_(
'New server description '
'(supported by --os-compute-api-version 2.19 or above)'
),
)
parser.add_argument(
'--tag',
metavar='<tag>',
action='append',
default=[],
dest='tags',
help=_(
'Tag for the server. '
'Specify multiple times to add multiple tags. '
'(supported by --os-compute-api-version 2.26 or above)'
),
)
parser.add_argument(
'--hostname',
metavar='<hostname>',
help=_(
'Hostname configured for the server in the metadata service. '
'A separate utility running in the guest is required to '
'propagate changes to this value to the guest OS itself. '
'(supported by --os-compute-api-version 2.90 or above)'
),
)
return parser
def take_action(self, parsed_args):
compute_client = self.app.client_manager.sdk_connection.compute
server = compute_client.find_server(
parsed_args.server, ignore_missing=False
)
if parsed_args.description:
if not sdk_utils.supports_microversion(compute_client, '2.19'):
msg = _(
'--os-compute-api-version 2.19 or greater is required to '
'support the --description option'
)
raise exceptions.CommandError(msg)
if parsed_args.tags:
if not sdk_utils.supports_microversion(compute_client, '2.26'):
msg = _(
'--os-compute-api-version 2.26 or greater is required to '
'support the --tag option'
)
raise exceptions.CommandError(msg)
if parsed_args.hostname:
if not sdk_utils.supports_microversion(compute_client, '2.90'):
msg = _(
'--os-compute-api-version 2.90 or greater is required to '
'support the --hostname option'
)
raise exceptions.CommandError(msg)
update_kwargs = {}
if parsed_args.name:
update_kwargs['name'] = parsed_args.name
if parsed_args.description:
update_kwargs['description'] = parsed_args.description
if parsed_args.hostname:
update_kwargs['hostname'] = parsed_args.hostname
if update_kwargs:
compute_client.update_server(server, **update_kwargs)
if parsed_args.properties:
compute_client.set_server_metadata(
server, **parsed_args.properties
)
if parsed_args.state:
compute_client.reset_server_state(server, state=parsed_args.state)
if parsed_args.root_password:
p1 = getpass.getpass(_('New password: '))
p2 = getpass.getpass(_('Retype new password: '))
if p1 == p2:
compute_client.change_server_password(server, p1)
else:
msg = _("Passwords do not match, password unchanged")
raise exceptions.CommandError(msg)
elif parsed_args.password:
compute_client.change_server_password(server, parsed_args.password)
elif parsed_args.no_password:
compute_client.clear_server_password(server)
if parsed_args.tags:
for tag in parsed_args.tags:
compute_client.add_tag_to_server(server, tag=tag)
class ShelveServer(command.Command):
"""Shelve and optionally offload server(s).
Shelving a server creates a snapshot of the server and stores this
snapshot before shutting down the server. This shelved server can then be
offloaded or deleted from the host, freeing up remaining resources on the
host, such as network interfaces. Shelved servers can be unshelved,
restoring the server from the snapshot. Shelving is therefore useful where
users wish to retain the UUID and IP of a server, without utilizing other
resources or disks.
Most clouds are configured to automatically offload shelved servers
immediately or after a small delay. For clouds where this is not
configured, or where the delay is larger, offloading can be manually
specified. This is an admin-only operation by default.
"""
def get_parser(self, prog_name):
parser = super().get_parser(prog_name)
parser.add_argument(
'servers',
metavar='<server>',
nargs='+',
help=_('Server(s) to shelve (name or ID)'),
)
parser.add_argument(
'--offload',
action='store_true',
default=False,
help=_(
'Remove the shelved server(s) from the host (admin only). '
'Invoking this option on an unshelved server(s) will result '
'in the server being shelved first'
),
)
parser.add_argument(
'--wait',
action='store_true',
default=False,
help=_('Wait for shelve and/or offload operation to complete'),
)
return parser
def take_action(self, parsed_args):
def _show_progress(progress):
if progress:
self.app.stdout.write('\rProgress: %s' % progress)
self.app.stdout.flush()
compute_client = self.app.client_manager.sdk_connection.compute
server_ids = []
for server in parsed_args.servers:
server_obj = compute_client.find_server(
server,
ignore_missing=False,
)
if server_obj.status.lower() in ('shelved', 'shelved_offloaded'):
continue
server_ids.append(server_obj.id)
compute_client.shelve_server(server_obj.id)
# if we don't have to wait, either because it was requested explicitly
# or is required implicitly, then our job is done
if not parsed_args.wait and not parsed_args.offload:
return
for server_id in server_ids:
# We use osc-lib's wait_for_status since that allows for a callback
# TODO(stephenfin): We should wait for these in parallel using e.g.
# https://review.opendev.org/c/openstack/osc-lib/+/762503/
if not utils.wait_for_status(
compute_client.get_server,
server_id,
success_status=('shelved', 'shelved_offloaded'),
callback=_show_progress,
):
msg = _('Error shelving server: %s') % server_id
raise exceptions.CommandError(msg)
if not parsed_args.offload:
return
for server_id in server_ids:
server_obj = compute_client.get_server(server_id)
if server_obj.status.lower() == 'shelved_offloaded':
continue
compute_client.shelve_offload_server(server_id)
if not parsed_args.wait:
return
for server_id in server_ids:
# We use osc-lib's wait_for_status since that allows for a callback
# TODO(stephenfin): We should wait for these in parallel using e.g.
# https://review.opendev.org/c/openstack/osc-lib/+/762503/
if not utils.wait_for_status(
compute_client.get_server,
server_id,
success_status=('shelved_offloaded',),
callback=_show_progress,
):
msg = _('Error offloading shelved server: %s') % server_id
raise exceptions.CommandError(msg)
class ShowServer(command.ShowOne):
_description = _(
"""Show server details.
Specify ``--os-compute-api-version 2.47`` or higher to see the embedded flavor
information for the server."""
)
def get_parser(self, prog_name):
parser = super().get_parser(prog_name)
parser.add_argument(
'server',
metavar='<server>',
help=_('Server (name or ID)'),
)
# TODO(stephenfin): This should be a separate command, not a flag
diagnostics_group = parser.add_mutually_exclusive_group()
diagnostics_group.add_argument(
'--diagnostics',
action='store_true',
default=False,
help=_('Display server diagnostics information'),
)
diagnostics_group.add_argument(
'--topology',
action='store_true',
default=False,
help=_(
'Include topology information in the output '
'(supported by --os-compute-api-version 2.78 or above)'
),
)
return parser
def take_action(self, parsed_args):
compute_client = self.app.client_manager.sdk_connection.compute
image_client = self.app.client_manager.image
server = compute_client.find_server(
parsed_args.server,
ignore_missing=False,
details=True,
)
if parsed_args.diagnostics:
data = compute_client.get_server_diagnostics(server)
return zip(*sorted(data.items()))
topology = None
if parsed_args.topology:
if not sdk_utils.supports_microversion(compute_client, '2.78'):
msg = _(
'--os-compute-api-version 2.78 or greater is required to '
'support the --topology option'
)
raise exceptions.CommandError(msg)
topology = server.fetch_topology(compute_client)
data = _prep_server_detail(
compute_client, image_client, server, refresh=False
)
if topology:
data['topology'] = format_columns.DictColumn(topology)
return zip(*sorted(data.items()))
class SshServer(command.Command):
_description = _("SSH to server")
def get_parser(self, prog_name):
parser = super().get_parser(prog_name)
parser.add_argument(
'server',
metavar='<server>',
help=_('Server (name or ID)'),
)
# Deprecated during the Yoga cycle
parser.add_argument(
'--login',
'-l',
metavar='<login-name>',
help=argparse.SUPPRESS,
)
# Deprecated during the Yoga cycle
parser.add_argument(
'--port',
'-p',
metavar='<port>',
type=int,
help=argparse.SUPPRESS,
)
# Deprecated during the Yoga cycle
parser.add_argument(
'--identity',
'-i',
metavar='<keyfile>',
help=argparse.SUPPRESS,
)
# Deprecated during the Yoga cycle
parser.add_argument(
'--option',
'-o',
metavar='<config-options>',
help=argparse.SUPPRESS,
)
ip_group = parser.add_mutually_exclusive_group()
ip_group.add_argument(
'-4',
dest='ipv4',
action='store_true',
default=False,
help=_('Use only IPv4 addresses'),
)
ip_group.add_argument(
'-6',
dest='ipv6',
action='store_true',
default=False,
help=_('Use only IPv6 addresses'),
)
type_group = parser.add_mutually_exclusive_group()
type_group.add_argument(
'--public',
dest='address_type',
action='store_const',
const='public',
default='public',
help=_('Use public IP address'),
)
type_group.add_argument(
'--private',
dest='address_type',
action='store_const',
const='private',
default='public',
help=_('Use private IP address'),
)
type_group.add_argument(
'--address-type',
metavar='<address-type>',
dest='address_type',
default='public',
help=_('Use other IP address (public, private, etc)'),
)
# Deprecated during the Yoga cycle
parser.add_argument(
'-v',
dest='verbose',
action='store_true',
default=False,
help=argparse.SUPPRESS,
)
parser.add_argument(
'ssh_args',
nargs='*',
metavar='-- <standard ssh args>',
help=(
'Any argument or option that ssh allows. '
'Use -- once between openstackclient args and SSH args.'
),
)
return parser
def take_action(self, parsed_args):
compute_client = self.app.client_manager.sdk_connection.compute
server = compute_client.find_server(
parsed_args.server, ignore_missing=False
)
# first, handle the deprecated options
if any(
(
parsed_args.port,
parsed_args.identity,
parsed_args.option,
parsed_args.login,
parsed_args.verbose,
)
):
msg = _(
'The ssh options have been deprecated. The ssh equivalent '
'options can be used instead as arguments after "--" on '
'the command line.'
)
self.log.warning(msg)
ip_address_family = [4, 6]
if parsed_args.ipv4:
ip_address_family = [4]
if parsed_args.ipv6:
ip_address_family = [6]
args = parsed_args.ssh_args[:]
if parsed_args.port:
args.extend(['-p', str(parsed_args.port)])
if parsed_args.identity:
args.extend(['-i', parsed_args.identity])
if parsed_args.option:
args.extend(['-o', parsed_args.option])
if parsed_args.login:
login = parsed_args.login
args.extend(['-l', login])
elif '-l' not in args:
login = self.app.client_manager.auth_ref.username
args.extend(['-l', login])
if parsed_args.verbose:
args.append('-v')
ip_address = _get_ip_address(
server.addresses,
parsed_args.address_type,
ip_address_family,
)
cmd = ' '.join(['ssh', ip_address] + args)
LOG.debug(f"ssh command: {cmd}")
# we intentionally pass through user-provided arguments and run this in
# the user's shell
os.system(cmd) # nosec: B605
class StartServer(command.Command):
_description = _("Start server(s)")
def get_parser(self, prog_name):
parser = super().get_parser(prog_name)
parser.add_argument(
'server',
metavar='<server>',
nargs="+",
help=_('Server(s) to start (name or ID)'),
)
parser.add_argument(
'--all-projects',
action='store_true',
default=boolenv('ALL_PROJECTS'),
help=_(
'Start server(s) in another project by name (admin only) '
'(can be specified using the ALL_PROJECTS envvar)'
),
)
return parser
def take_action(self, parsed_args):
compute_client = self.app.client_manager.sdk_connection.compute
for server in parsed_args.server:
server_id = compute_client.find_server(
server,
ignore_missing=False,
details=False,
all_projects=parsed_args.all_projects,
).id
compute_client.start_server(server_id)
class StopServer(command.Command):
_description = _("Stop server(s)")
def get_parser(self, prog_name):
parser = super().get_parser(prog_name)
parser.add_argument(
'server',
metavar='<server>',
nargs="+",
help=_('Server(s) to stop (name or ID)'),
)
parser.add_argument(
'--all-projects',
action='store_true',
default=boolenv('ALL_PROJECTS'),
help=_(
'Stop server(s) in another project by name (admin only)'
'(can be specified using the ALL_PROJECTS envvar)'
),
)
return parser
def take_action(self, parsed_args):
compute_client = self.app.client_manager.sdk_connection.compute
for server in parsed_args.server:
server_id = compute_client.find_server(
server,
ignore_missing=False,
details=False,
all_projects=parsed_args.all_projects,
).id
compute_client.stop_server(server_id)
class SuspendServer(command.Command):
_description = _("Suspend server(s)")
def get_parser(self, prog_name):
parser = super().get_parser(prog_name)
parser.add_argument(
'server',
metavar='<server>',
nargs='+',
help=_('Server(s) to suspend (name or ID)'),
)
return parser
def take_action(self, parsed_args):
compute_client = self.app.client_manager.sdk_connection.compute
for server in parsed_args.server:
server_id = compute_client.find_server(
server,
ignore_missing=False,
).id
compute_client.suspend_server(server_id)
class UnlockServer(command.Command):
_description = _("Unlock server(s)")
def get_parser(self, prog_name):
parser = super().get_parser(prog_name)
parser.add_argument(
'server',
metavar='<server>',
nargs='+',
help=_('Server(s) to unlock (name or ID)'),
)
return parser
def take_action(self, parsed_args):
compute_client = self.app.client_manager.sdk_connection.compute
for server in parsed_args.server:
server_id = compute_client.find_server(
server,
ignore_missing=False,
).id
compute_client.unlock_server(server_id)
class UnpauseServer(command.Command):
_description = _("Unpause server(s)")
def get_parser(self, prog_name):
parser = super().get_parser(prog_name)
parser.add_argument(
'server',
metavar='<server>',
nargs='+',
help=_('Server(s) to unpause (name or ID)'),
)
return parser
def take_action(self, parsed_args):
compute_client = self.app.client_manager.sdk_connection.compute
for server in parsed_args.server:
server_id = compute_client.find_server(
server,
ignore_missing=False,
).id
compute_client.unpause_server(server_id)
class UnrescueServer(command.Command):
_description = _("Restore server from rescue mode")
def get_parser(self, prog_name):
parser = super().get_parser(prog_name)
parser.add_argument(
'server',
metavar='<server>',
help=_('Server (name or ID)'),
)
return parser
def take_action(self, parsed_args):
compute_client = self.app.client_manager.sdk_connection.compute
server = compute_client.find_server(
parsed_args.server, ignore_missing=False
)
compute_client.unrescue_server(server)
class UnsetServer(command.Command):
_description = _("Unset server properties and tags")
def get_parser(self, prog_name):
parser = super().get_parser(prog_name)
parser.add_argument(
'server',
metavar='<server>',
help=_('Server (name or ID)'),
)
property_group = parser.add_mutually_exclusive_group()
property_group.add_argument(
'--property',
metavar='<key>',
action='append',
default=[],
dest='properties',
help=_(
'Property key to remove from server '
'(repeat option to remove multiple values)'
),
)
property_group.add_argument(
'--all-properties',
action='store_true',
help=_('Remove all properties'),
)
parser.add_argument(
'--description',
dest='description',
action='store_true',
help=_(
'Unset server description '
'(supported by --os-compute-api-version 2.19 or above)'
),
)
tag_group = parser.add_mutually_exclusive_group()
tag_group.add_argument(
'--tag',
metavar='<tag>',
action='append',
default=[],
dest='tags',
help=_(
'Tag to remove from the server. '
'Specify multiple times to remove multiple tags. '
'(supported by --os-compute-api-version 2.26 or above)'
),
)
tag_group.add_argument(
'--all-tags',
action='store_true',
help=_(
'Remove all tags '
'(supported by --os-compute-api-version 2.26 or above)'
),
)
return parser
def take_action(self, parsed_args):
compute_client = self.app.client_manager.sdk_connection.compute
server = compute_client.find_server(
parsed_args.server, ignore_missing=False
)
if parsed_args.properties or parsed_args.all_properties:
compute_client.delete_server_metadata(
server, parsed_args.properties or None
)
if parsed_args.description:
if not sdk_utils.supports_microversion(compute_client, '2.19'):
msg = _(
'--os-compute-api-version 2.19 or greater is required to '
'support the --description option'
)
raise exceptions.CommandError(msg)
compute_client.update_server(server, description="")
if parsed_args.tags or parsed_args.all_tags:
if not sdk_utils.supports_microversion(compute_client, '2.26'):
msg = _(
'--os-compute-api-version 2.26 or greater is required to '
'support the --tag option'
)
raise exceptions.CommandError(msg)
for tag in parsed_args.tags:
compute_client.remove_tag_from_server(server, tag)
if parsed_args.all_tags:
compute_client.remove_tags_from_server(server)
class UnshelveServer(command.Command):
_description = _("Unshelve server(s)")
def get_parser(self, prog_name):
parser = super().get_parser(prog_name)
parser.add_argument(
'server',
metavar='<server>',
nargs='+',
help=_('Server(s) to unshelve (name or ID)'),
)
group = parser.add_mutually_exclusive_group()
group.add_argument(
'--availability-zone',
default=None,
help=_(
'Name of the availability zone in which to unshelve a '
'SHELVED_OFFLOADED server '
'(supported by --os-compute-api-version 2.77 or above)'
),
)
group.add_argument(
'--no-availability-zone',
action='store_true',
default=False,
help=_(
'Unpin the availability zone of a SHELVED_OFFLOADED '
'server. Server will be unshelved on a host without '
'availability zone constraint '
'(supported by --os-compute-api-version 2.91 or above)'
),
)
parser.add_argument(
'--host',
default=None,
help=_(
'Name of the destination host in which to unshelve a '
'SHELVED_OFFLOADED server '
'(supported by --os-compute-api-version 2.91 or above)'
),
)
parser.add_argument(
'--wait',
action='store_true',
default=False,
help=_('Wait for unshelve operation to complete'),
)
return parser
def take_action(self, parsed_args):
def _show_progress(progress):
if progress:
self.app.stdout.write('\rProgress: %s' % progress)
self.app.stdout.flush()
compute_client = self.app.client_manager.sdk_connection.compute
kwargs = {}
if parsed_args.availability_zone:
if not sdk_utils.supports_microversion(compute_client, '2.77'):
msg = _(
'--os-compute-api-version 2.77 or greater is required '
'to support the --availability-zone option'
)
raise exceptions.CommandError(msg)
kwargs['availability_zone'] = parsed_args.availability_zone
if parsed_args.host:
if not sdk_utils.supports_microversion(compute_client, '2.91'):
msg = _(
'--os-compute-api-version 2.91 or greater is required '
'to support the --host option'
)
raise exceptions.CommandError(msg)
kwargs['host'] = parsed_args.host
if parsed_args.no_availability_zone:
if not sdk_utils.supports_microversion(compute_client, '2.91'):
msg = _(
'--os-compute-api-version 2.91 or greater is required '
'to support the --no-availability-zone option'
)
raise exceptions.CommandError(msg)
kwargs['availability_zone'] = None
for server in parsed_args.server:
server_obj = compute_client.find_server(
server,
ignore_missing=False,
)
if server_obj.status.lower() not in (
'shelved',
'shelved_offloaded',
):
continue
compute_client.unshelve_server(server_obj.id, **kwargs)
if parsed_args.wait:
if not utils.wait_for_status(
compute_client.get_server,
server_obj.id,
success_status=('active', 'shutoff'),
callback=_show_progress,
):
msg = _('Error unshelving server: %s') % server_obj.id
raise exceptions.CommandError(msg)