2013-07-08 16:48:10 -05:00
|
|
|
# Copyright 2012-2013 OpenStack Foundation
|
2012-12-26 14:22:23 -06:00
|
|
|
#
|
|
|
|
# Licensed under the Apache License, Version 2.0 (the "License"); you may
|
|
|
|
# not use this file except in compliance with the License. You may obtain
|
|
|
|
# a copy of the License at
|
|
|
|
#
|
|
|
|
# http://www.apache.org/licenses/LICENSE-2.0
|
|
|
|
#
|
|
|
|
# Unless required by applicable law or agreed to in writing, software
|
|
|
|
# distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
|
|
|
|
# WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
|
|
|
|
# License for the specific language governing permissions and limitations
|
|
|
|
# under the License.
|
|
|
|
#
|
|
|
|
|
2013-04-09 13:59:12 -05:00
|
|
|
"""Image V2 Action Implementations"""
|
2012-12-26 14:22:23 -06:00
|
|
|
|
2015-01-21 15:02:58 -06:00
|
|
|
import argparse
|
2012-12-26 14:22:23 -06:00
|
|
|
import logging
|
2013-07-08 16:48:10 -05:00
|
|
|
import six
|
2012-12-26 14:22:23 -06:00
|
|
|
|
|
|
|
from cliff import command
|
|
|
|
from cliff import lister
|
2013-01-24 11:39:18 -06:00
|
|
|
from cliff import show
|
2012-12-26 14:22:23 -06:00
|
|
|
from glanceclient.common import utils as gc_utils
|
2015-09-19 13:04:13 -04:00
|
|
|
|
2015-01-21 15:02:58 -06:00
|
|
|
from openstackclient.api import utils as api_utils
|
2015-09-19 13:04:13 -04:00
|
|
|
from openstackclient.common import exceptions
|
2015-01-21 15:02:58 -06:00
|
|
|
from openstackclient.common import parseractions
|
2012-12-26 14:22:23 -06:00
|
|
|
from openstackclient.common import utils
|
2015-07-16 17:32:42 +08:00
|
|
|
from openstackclient.identity import common
|
|
|
|
|
|
|
|
|
2015-09-19 13:04:13 -04:00
|
|
|
DEFAULT_CONTAINER_FORMAT = 'bare'
|
|
|
|
DEFAULT_DISK_FORMAT = 'raw'
|
|
|
|
|
|
|
|
|
2015-07-16 17:32:42 +08:00
|
|
|
class AddProjectToImage(show.ShowOne):
|
|
|
|
"""Associate project with image"""
|
|
|
|
|
|
|
|
log = logging.getLogger(__name__ + ".AddProjectToImage")
|
|
|
|
|
|
|
|
def get_parser(self, prog_name):
|
|
|
|
parser = super(AddProjectToImage, self).get_parser(prog_name)
|
|
|
|
parser.add_argument(
|
|
|
|
"image",
|
|
|
|
metavar="<image>",
|
|
|
|
help="Image to share (name or ID)",
|
|
|
|
)
|
|
|
|
parser.add_argument(
|
|
|
|
"project",
|
|
|
|
metavar="<project>",
|
|
|
|
help="Project to associate with image (name or ID)",
|
|
|
|
)
|
|
|
|
common.add_project_domain_option_to_parser(parser)
|
|
|
|
return parser
|
|
|
|
|
|
|
|
def take_action(self, parsed_args):
|
|
|
|
self.log.debug("take_action(%s)", parsed_args)
|
|
|
|
|
|
|
|
image_client = self.app.client_manager.image
|
|
|
|
identity_client = self.app.client_manager.identity
|
|
|
|
|
|
|
|
project_id = common.find_project(identity_client,
|
|
|
|
parsed_args.project,
|
|
|
|
parsed_args.project_domain).id
|
|
|
|
|
|
|
|
image_id = utils.find_resource(
|
|
|
|
image_client.images,
|
|
|
|
parsed_args.image).id
|
|
|
|
|
|
|
|
image_member = image_client.image_members.create(
|
|
|
|
image_id,
|
|
|
|
project_id,
|
|
|
|
)
|
|
|
|
|
|
|
|
return zip(*sorted(six.iteritems(image_member._info)))
|
2012-12-26 14:22:23 -06:00
|
|
|
|
|
|
|
|
2015-09-19 13:04:13 -04:00
|
|
|
class CreateImage(show.ShowOne):
|
|
|
|
"""Create/upload an image"""
|
|
|
|
|
|
|
|
log = logging.getLogger(__name__ + ".CreateImage")
|
|
|
|
deadopts = ('owner', 'size', 'location', 'copy-from', 'checksum', 'store')
|
|
|
|
|
|
|
|
def get_parser(self, prog_name):
|
|
|
|
parser = super(CreateImage, self).get_parser(prog_name)
|
|
|
|
# TODO(mordred): add --volume and --force parameters and support
|
|
|
|
# TODO(bunting): There are additional arguments that v1 supported
|
|
|
|
# that v2 either doesn't support or supports weirdly.
|
|
|
|
# --checksum - could be faked clientside perhaps?
|
|
|
|
# --owner - could be set as an update after the put?
|
|
|
|
# --location - maybe location add?
|
|
|
|
# --size - passing image size is actually broken in python-glanceclient
|
|
|
|
# --copy-from - does not exist in v2
|
|
|
|
# --store - does not exits in v2
|
|
|
|
parser.add_argument(
|
|
|
|
"name",
|
|
|
|
metavar="<image-name>",
|
|
|
|
help="New image name",
|
|
|
|
)
|
|
|
|
parser.add_argument(
|
|
|
|
"--id",
|
|
|
|
metavar="<id>",
|
|
|
|
help="Image ID to reserve",
|
|
|
|
)
|
|
|
|
parser.add_argument(
|
|
|
|
"--container-format",
|
|
|
|
default=DEFAULT_CONTAINER_FORMAT,
|
|
|
|
metavar="<container-format>",
|
|
|
|
help="Image container format "
|
|
|
|
"(default: %s)" % DEFAULT_CONTAINER_FORMAT,
|
|
|
|
)
|
|
|
|
parser.add_argument(
|
|
|
|
"--disk-format",
|
|
|
|
default=DEFAULT_DISK_FORMAT,
|
|
|
|
metavar="<disk-format>",
|
|
|
|
help="Image disk format "
|
|
|
|
"(default: %s)" % DEFAULT_DISK_FORMAT,
|
|
|
|
)
|
|
|
|
parser.add_argument(
|
|
|
|
"--min-disk",
|
|
|
|
metavar="<disk-gb>",
|
|
|
|
type=int,
|
|
|
|
help="Minimum disk size needed to boot image, in gigabytes",
|
|
|
|
)
|
|
|
|
parser.add_argument(
|
|
|
|
"--min-ram",
|
|
|
|
metavar="<ram-mb>",
|
|
|
|
type=int,
|
|
|
|
help="Minimum RAM size needed to boot image, in megabytes",
|
|
|
|
)
|
|
|
|
parser.add_argument(
|
|
|
|
"--file",
|
|
|
|
metavar="<file>",
|
|
|
|
help="Upload image from local file",
|
|
|
|
)
|
|
|
|
protected_group = parser.add_mutually_exclusive_group()
|
|
|
|
protected_group.add_argument(
|
|
|
|
"--protected",
|
|
|
|
action="store_true",
|
|
|
|
help="Prevent image from being deleted",
|
|
|
|
)
|
|
|
|
protected_group.add_argument(
|
|
|
|
"--unprotected",
|
|
|
|
action="store_true",
|
|
|
|
help="Allow image to be deleted (default)",
|
|
|
|
)
|
|
|
|
public_group = parser.add_mutually_exclusive_group()
|
|
|
|
public_group.add_argument(
|
|
|
|
"--public",
|
|
|
|
action="store_true",
|
|
|
|
help="Image is accessible to the public",
|
|
|
|
)
|
|
|
|
public_group.add_argument(
|
|
|
|
"--private",
|
|
|
|
action="store_true",
|
|
|
|
help="Image is inaccessible to the public (default)",
|
|
|
|
)
|
|
|
|
parser.add_argument(
|
|
|
|
"--property",
|
|
|
|
dest="properties",
|
|
|
|
metavar="<key=value>",
|
|
|
|
action=parseractions.KeyValueAction,
|
|
|
|
help="Set a property on this image "
|
|
|
|
"(repeat option to set multiple properties)",
|
|
|
|
)
|
|
|
|
parser.add_argument(
|
|
|
|
"--tag",
|
|
|
|
dest="tags",
|
|
|
|
metavar="<tag>",
|
|
|
|
action='append',
|
|
|
|
help="Set a tag on this image "
|
|
|
|
"(repeat option to set multiple tags)",
|
|
|
|
)
|
|
|
|
for deadopt in self.deadopts:
|
|
|
|
parser.add_argument(
|
|
|
|
"--%s" % deadopt,
|
|
|
|
metavar="<%s>" % deadopt,
|
|
|
|
dest=deadopt.replace('-', '_'),
|
|
|
|
help=argparse.SUPPRESS
|
|
|
|
)
|
|
|
|
return parser
|
|
|
|
|
|
|
|
def take_action(self, parsed_args):
|
|
|
|
self.log.debug("take_action(%s)", parsed_args)
|
|
|
|
image_client = self.app.client_manager.image
|
|
|
|
|
|
|
|
for deadopt in self.deadopts:
|
|
|
|
if getattr(parsed_args, deadopt.replace('-', '_'), None):
|
|
|
|
raise exceptions.CommandError(
|
|
|
|
"ERROR: --%s was given, which is an Image v1 option"
|
|
|
|
" that is no longer supported in Image v2" % deadopt)
|
|
|
|
|
|
|
|
# Build an attribute dict from the parsed args, only include
|
|
|
|
# attributes that were actually set on the command line
|
|
|
|
kwargs = {}
|
|
|
|
copy_attrs = ('name', 'id',
|
|
|
|
'container_format', 'disk_format',
|
|
|
|
'min_disk', 'min_ram',
|
|
|
|
'tags')
|
|
|
|
for attr in copy_attrs:
|
|
|
|
if attr in parsed_args:
|
|
|
|
val = getattr(parsed_args, attr, None)
|
|
|
|
if val:
|
|
|
|
# Only include a value in kwargs for attributes that
|
|
|
|
# are actually present on the command line
|
|
|
|
kwargs[attr] = val
|
|
|
|
# properties should get flattened into the general kwargs
|
|
|
|
if getattr(parsed_args, 'properties', None):
|
|
|
|
for k, v in six.iteritems(parsed_args.properties):
|
|
|
|
kwargs[k] = str(v)
|
|
|
|
# Handle exclusive booleans with care
|
|
|
|
# Avoid including attributes in kwargs if an option is not
|
|
|
|
# present on the command line. These exclusive booleans are not
|
|
|
|
# a single value for the pair of options because the default must be
|
|
|
|
# to do nothing when no options are present as opposed to always
|
|
|
|
# setting a default.
|
|
|
|
if parsed_args.protected:
|
|
|
|
kwargs['protected'] = True
|
|
|
|
if parsed_args.unprotected:
|
|
|
|
kwargs['protected'] = False
|
|
|
|
if parsed_args.public:
|
|
|
|
kwargs['visibility'] = 'public'
|
|
|
|
if parsed_args.private:
|
|
|
|
kwargs['visibility'] = 'private'
|
|
|
|
|
|
|
|
# open the file first to ensure any failures are handled before the
|
|
|
|
# image is created
|
|
|
|
fp = gc_utils.get_data_file(parsed_args)
|
|
|
|
|
|
|
|
if fp is None and parsed_args.file:
|
|
|
|
self.log.warning("Failed to get an image file.")
|
|
|
|
return {}, {}
|
|
|
|
|
|
|
|
image = image_client.images.create(**kwargs)
|
|
|
|
|
|
|
|
if fp is not None:
|
|
|
|
with fp:
|
|
|
|
try:
|
|
|
|
image_client.images.upload(image.id, fp)
|
|
|
|
except Exception as e:
|
|
|
|
# If the upload fails for some reason attempt to remove the
|
|
|
|
# dangling queued image made by the create() call above but
|
|
|
|
# only if the user did not specify an id which indicates
|
|
|
|
# the Image already exists and should be left alone.
|
|
|
|
try:
|
|
|
|
if 'id' not in kwargs:
|
|
|
|
image_client.images.delete(image.id)
|
|
|
|
except Exception:
|
|
|
|
pass # we don't care about this one
|
|
|
|
raise e # now, throw the upload exception again
|
|
|
|
|
|
|
|
# update the image after the data has been uploaded
|
|
|
|
image = image_client.images.get(image.id)
|
|
|
|
|
|
|
|
return zip(*sorted(six.iteritems(image)))
|
|
|
|
|
|
|
|
|
2013-04-09 13:59:12 -05:00
|
|
|
class DeleteImage(command.Command):
|
2014-12-10 11:47:54 +08:00
|
|
|
"""Delete image(s)"""
|
2013-04-09 13:59:12 -05:00
|
|
|
|
|
|
|
log = logging.getLogger(__name__ + ".DeleteImage")
|
|
|
|
|
|
|
|
def get_parser(self, prog_name):
|
|
|
|
parser = super(DeleteImage, self).get_parser(prog_name)
|
|
|
|
parser.add_argument(
|
2014-12-10 11:47:54 +08:00
|
|
|
"images",
|
2013-07-08 16:48:10 -05:00
|
|
|
metavar="<image>",
|
2014-12-10 11:47:54 +08:00
|
|
|
nargs="+",
|
|
|
|
help="Image(s) to delete (name or ID)",
|
2013-07-08 16:48:10 -05:00
|
|
|
)
|
2013-04-09 13:59:12 -05:00
|
|
|
return parser
|
|
|
|
|
|
|
|
def take_action(self, parsed_args):
|
2014-05-20 13:11:19 +02:00
|
|
|
self.log.debug("take_action(%s)", parsed_args)
|
2013-04-09 13:59:12 -05:00
|
|
|
|
|
|
|
image_client = self.app.client_manager.image
|
2014-12-10 11:47:54 +08:00
|
|
|
for image in parsed_args.images:
|
|
|
|
image_obj = utils.find_resource(
|
|
|
|
image_client.images,
|
|
|
|
image,
|
|
|
|
)
|
|
|
|
image_client.images.delete(image_obj.id)
|
2013-04-09 13:59:12 -05:00
|
|
|
|
|
|
|
|
2012-12-26 14:22:23 -06:00
|
|
|
class ListImage(lister.Lister):
|
2013-07-08 16:48:10 -05:00
|
|
|
"""List available images"""
|
2012-12-26 14:22:23 -06:00
|
|
|
|
|
|
|
log = logging.getLogger(__name__ + ".ListImage")
|
|
|
|
|
|
|
|
def get_parser(self, prog_name):
|
|
|
|
parser = super(ListImage, self).get_parser(prog_name)
|
2014-09-18 00:56:38 -05:00
|
|
|
public_group = parser.add_mutually_exclusive_group()
|
|
|
|
public_group.add_argument(
|
|
|
|
"--public",
|
|
|
|
dest="public",
|
|
|
|
action="store_true",
|
|
|
|
default=False,
|
|
|
|
help="List only public images",
|
|
|
|
)
|
|
|
|
public_group.add_argument(
|
|
|
|
"--private",
|
|
|
|
dest="private",
|
|
|
|
action="store_true",
|
|
|
|
default=False,
|
|
|
|
help="List only private images",
|
|
|
|
)
|
2015-01-21 15:02:58 -06:00
|
|
|
public_group.add_argument(
|
|
|
|
"--shared",
|
|
|
|
dest="shared",
|
|
|
|
action="store_true",
|
|
|
|
default=False,
|
|
|
|
help="List only shared images",
|
|
|
|
)
|
|
|
|
parser.add_argument(
|
|
|
|
'--property',
|
|
|
|
metavar='<key=value>',
|
|
|
|
action=parseractions.KeyValueAction,
|
|
|
|
help='Filter output based on property',
|
|
|
|
)
|
2014-07-25 01:09:59 -04:00
|
|
|
parser.add_argument(
|
|
|
|
'--long',
|
|
|
|
action='store_true',
|
|
|
|
default=False,
|
|
|
|
help='List additional fields in output',
|
|
|
|
)
|
2015-01-21 15:02:58 -06:00
|
|
|
|
2015-04-30 17:21:45 +08:00
|
|
|
# --page-size has never worked, leave here for silent compatibility
|
2015-01-21 15:02:58 -06:00
|
|
|
# We'll implement limit/marker differently later
|
|
|
|
parser.add_argument(
|
|
|
|
"--page-size",
|
|
|
|
metavar="<size>",
|
|
|
|
help=argparse.SUPPRESS,
|
|
|
|
)
|
2015-01-21 17:19:46 +08:00
|
|
|
parser.add_argument(
|
|
|
|
'--sort',
|
|
|
|
metavar="<key>[:<direction>]",
|
|
|
|
help="Sort output by selected keys and directions(asc or desc) "
|
|
|
|
"(default: asc), multiple keys and directions can be "
|
|
|
|
"specified separated by comma",
|
|
|
|
)
|
2012-12-26 14:22:23 -06:00
|
|
|
return parser
|
|
|
|
|
|
|
|
def take_action(self, parsed_args):
|
2014-05-20 13:11:19 +02:00
|
|
|
self.log.debug("take_action(%s)", parsed_args)
|
2013-01-24 11:39:18 -06:00
|
|
|
|
2012-12-26 14:22:23 -06:00
|
|
|
image_client = self.app.client_manager.image
|
|
|
|
|
|
|
|
kwargs = {}
|
2014-09-18 00:56:38 -05:00
|
|
|
if parsed_args.public:
|
|
|
|
kwargs['public'] = True
|
|
|
|
if parsed_args.private:
|
|
|
|
kwargs['private'] = True
|
2015-01-21 15:02:58 -06:00
|
|
|
if parsed_args.shared:
|
|
|
|
kwargs['shared'] = True
|
2012-12-26 14:22:23 -06:00
|
|
|
|
2014-07-25 01:09:59 -04:00
|
|
|
if parsed_args.long:
|
2015-01-21 15:02:58 -06:00
|
|
|
columns = (
|
|
|
|
'ID',
|
|
|
|
'Name',
|
|
|
|
'Disk Format',
|
|
|
|
'Container Format',
|
|
|
|
'Size',
|
|
|
|
'Status',
|
|
|
|
'visibility',
|
|
|
|
'protected',
|
|
|
|
'owner',
|
|
|
|
'tags',
|
|
|
|
)
|
|
|
|
column_headers = (
|
|
|
|
'ID',
|
|
|
|
'Name',
|
|
|
|
'Disk Format',
|
|
|
|
'Container Format',
|
|
|
|
'Size',
|
|
|
|
'Status',
|
|
|
|
'Visibility',
|
|
|
|
'Protected',
|
|
|
|
'Owner',
|
|
|
|
'Tags',
|
|
|
|
)
|
2014-07-25 01:09:59 -04:00
|
|
|
else:
|
|
|
|
columns = ("ID", "Name")
|
2015-01-21 15:02:58 -06:00
|
|
|
column_headers = columns
|
2012-12-26 14:22:23 -06:00
|
|
|
|
2015-04-14 11:03:22 -06:00
|
|
|
# List of image data received
|
|
|
|
data = []
|
|
|
|
# No pages received yet, so start the page marker at None.
|
|
|
|
marker = None
|
|
|
|
while True:
|
|
|
|
page = image_client.api.image_list(marker=marker, **kwargs)
|
|
|
|
if not page:
|
|
|
|
break
|
|
|
|
data.extend(page)
|
|
|
|
# Set the marker to the id of the last item we received
|
|
|
|
marker = page[-1]['id']
|
2014-09-18 00:56:38 -05:00
|
|
|
|
2015-01-21 15:02:58 -06:00
|
|
|
if parsed_args.property:
|
|
|
|
# NOTE(dtroyer): coerce to a list to subscript it in py3
|
|
|
|
attr, value = list(parsed_args.property.items())[0]
|
|
|
|
api_utils.simple_filter(
|
|
|
|
data,
|
|
|
|
attr=attr,
|
|
|
|
value=value,
|
|
|
|
property_field='properties',
|
|
|
|
)
|
2015-01-21 17:19:46 +08:00
|
|
|
|
|
|
|
data = utils.sort_items(data, parsed_args.sort)
|
|
|
|
|
2015-01-21 15:02:58 -06:00
|
|
|
return (
|
|
|
|
column_headers,
|
|
|
|
(utils.get_dict_properties(
|
|
|
|
s,
|
|
|
|
columns,
|
|
|
|
formatters={
|
|
|
|
'tags': utils.format_dict,
|
|
|
|
},
|
|
|
|
) for s in data)
|
|
|
|
)
|
2012-12-26 14:22:23 -06:00
|
|
|
|
|
|
|
|
2015-07-16 17:32:42 +08:00
|
|
|
class RemoveProjectImage(command.Command):
|
|
|
|
"""Disassociate project with image"""
|
|
|
|
|
|
|
|
log = logging.getLogger(__name__ + ".RemoveProjectImage")
|
|
|
|
|
|
|
|
def get_parser(self, prog_name):
|
|
|
|
parser = super(RemoveProjectImage, self).get_parser(prog_name)
|
|
|
|
parser.add_argument(
|
|
|
|
"image",
|
|
|
|
metavar="<image>",
|
|
|
|
help="Image to unshare (name or ID)",
|
|
|
|
)
|
|
|
|
parser.add_argument(
|
|
|
|
"project",
|
|
|
|
metavar="<project>",
|
|
|
|
help="Project to disassociate with image (name or ID)",
|
|
|
|
)
|
|
|
|
common.add_project_domain_option_to_parser(parser)
|
|
|
|
return parser
|
|
|
|
|
|
|
|
def take_action(self, parsed_args):
|
|
|
|
self.log.debug("take_action(%s)", parsed_args)
|
|
|
|
|
|
|
|
image_client = self.app.client_manager.image
|
|
|
|
identity_client = self.app.client_manager.identity
|
|
|
|
|
|
|
|
project_id = common.find_project(identity_client,
|
|
|
|
parsed_args.project,
|
|
|
|
parsed_args.project_domain).id
|
|
|
|
|
|
|
|
image_id = utils.find_resource(
|
|
|
|
image_client.images,
|
|
|
|
parsed_args.image).id
|
|
|
|
|
|
|
|
image_client.image_members.delete(image_id, project_id)
|
|
|
|
|
|
|
|
|
2012-12-26 14:22:23 -06:00
|
|
|
class SaveImage(command.Command):
|
2013-07-08 16:48:10 -05:00
|
|
|
"""Save an image locally"""
|
2012-12-26 14:22:23 -06:00
|
|
|
|
|
|
|
log = logging.getLogger(__name__ + ".SaveImage")
|
|
|
|
|
|
|
|
def get_parser(self, prog_name):
|
|
|
|
parser = super(SaveImage, self).get_parser(prog_name)
|
|
|
|
parser.add_argument(
|
|
|
|
"--file",
|
2013-07-08 16:48:10 -05:00
|
|
|
metavar="<filename>",
|
2015-01-08 02:54:26 -05:00
|
|
|
help="Downloaded image save filename (default: stdout)",
|
2013-07-08 16:48:10 -05:00
|
|
|
)
|
2012-12-26 14:22:23 -06:00
|
|
|
parser.add_argument(
|
2013-07-08 16:48:10 -05:00
|
|
|
"image",
|
|
|
|
metavar="<image>",
|
2015-01-08 02:54:26 -05:00
|
|
|
help="Image to save (name or ID)",
|
2013-07-08 16:48:10 -05:00
|
|
|
)
|
2012-12-26 14:22:23 -06:00
|
|
|
return parser
|
|
|
|
|
|
|
|
def take_action(self, parsed_args):
|
2014-05-20 13:11:19 +02:00
|
|
|
self.log.debug("take_action(%s)", parsed_args)
|
2012-12-26 14:22:23 -06:00
|
|
|
|
2013-01-24 11:39:18 -06:00
|
|
|
image_client = self.app.client_manager.image
|
2013-07-08 16:48:10 -05:00
|
|
|
image = utils.find_resource(
|
|
|
|
image_client.images,
|
|
|
|
parsed_args.image,
|
|
|
|
)
|
2015-07-15 15:40:51 -04:00
|
|
|
data = image_client.images.data(image.id)
|
2013-01-24 11:39:18 -06:00
|
|
|
|
2012-12-26 14:22:23 -06:00
|
|
|
gc_utils.save_image(data, parsed_args.file)
|
|
|
|
|
|
|
|
|
2013-01-24 11:39:18 -06:00
|
|
|
class ShowImage(show.ShowOne):
|
2015-01-08 02:54:26 -05:00
|
|
|
"""Display image details"""
|
2012-12-26 14:22:23 -06:00
|
|
|
|
|
|
|
log = logging.getLogger(__name__ + ".ShowImage")
|
|
|
|
|
|
|
|
def get_parser(self, prog_name):
|
|
|
|
parser = super(ShowImage, self).get_parser(prog_name)
|
|
|
|
parser.add_argument(
|
2013-07-08 16:48:10 -05:00
|
|
|
"image",
|
|
|
|
metavar="<image>",
|
2015-01-08 02:54:26 -05:00
|
|
|
help="Image to display (name or ID)",
|
2013-07-08 16:48:10 -05:00
|
|
|
)
|
2012-12-26 14:22:23 -06:00
|
|
|
return parser
|
|
|
|
|
|
|
|
def take_action(self, parsed_args):
|
2014-05-20 13:11:19 +02:00
|
|
|
self.log.debug("take_action(%s)", parsed_args)
|
2013-01-24 11:39:18 -06:00
|
|
|
|
2012-12-26 14:22:23 -06:00
|
|
|
image_client = self.app.client_manager.image
|
2013-07-08 16:48:10 -05:00
|
|
|
image = utils.find_resource(
|
|
|
|
image_client.images,
|
|
|
|
parsed_args.image,
|
|
|
|
)
|
|
|
|
|
|
|
|
info = {}
|
2015-04-30 16:38:19 -07:00
|
|
|
info.update(image)
|
2013-07-08 16:48:10 -05:00
|
|
|
return zip(*sorted(six.iteritems(info)))
|
2015-05-12 17:01:07 -07:00
|
|
|
|
|
|
|
|
|
|
|
class SetImage(show.ShowOne):
|
|
|
|
"""Set image properties"""
|
|
|
|
|
|
|
|
log = logging.getLogger(__name__ + ".SetImage")
|
|
|
|
|
|
|
|
def get_parser(self, prog_name):
|
|
|
|
parser = super(SetImage, self).get_parser(prog_name)
|
|
|
|
parser.add_argument(
|
|
|
|
"image",
|
|
|
|
metavar="<image>",
|
|
|
|
help="Image to modify (name or ID)"
|
|
|
|
)
|
|
|
|
parser.add_argument(
|
|
|
|
"--name",
|
|
|
|
metavar="<name>",
|
|
|
|
help="New image name"
|
|
|
|
)
|
|
|
|
parser.add_argument(
|
|
|
|
"--architecture",
|
|
|
|
metavar="<architecture>",
|
|
|
|
help="Operating system Architecture"
|
|
|
|
)
|
|
|
|
parser.add_argument(
|
|
|
|
"--protected",
|
|
|
|
dest="protected",
|
|
|
|
action="store_true",
|
|
|
|
help="Prevent image from being deleted"
|
|
|
|
)
|
|
|
|
parser.add_argument(
|
|
|
|
"--instance-uuid",
|
|
|
|
metavar="<instance_uuid>",
|
|
|
|
help="ID of instance used to create this image"
|
|
|
|
)
|
|
|
|
parser.add_argument(
|
|
|
|
"--min-disk",
|
|
|
|
type=int,
|
|
|
|
metavar="<disk-gb>",
|
|
|
|
help="Minimum disk size needed to boot image, in gigabytes"
|
|
|
|
)
|
|
|
|
visibility_choices = ["public", "private"]
|
|
|
|
parser.add_argument(
|
|
|
|
"--visibility",
|
|
|
|
metavar="<visibility>",
|
|
|
|
choices=visibility_choices,
|
|
|
|
help="Scope of image accessibility. Valid values: %s"
|
|
|
|
% visibility_choices
|
|
|
|
)
|
|
|
|
help_msg = ("ID of image in Glance that should be used as the kernel"
|
|
|
|
" when booting an AMI-style image")
|
|
|
|
parser.add_argument(
|
|
|
|
"--kernel-id",
|
|
|
|
metavar="<kernel-id>",
|
|
|
|
help=help_msg
|
|
|
|
)
|
|
|
|
parser.add_argument(
|
|
|
|
"--os-version",
|
|
|
|
metavar="<os-version>",
|
|
|
|
help="Operating system version as specified by the distributor"
|
|
|
|
)
|
|
|
|
disk_choices = ["None", "ami", "ari", "aki", "vhd", "vmdk", "raw",
|
|
|
|
"qcow2", "vdi", "iso"]
|
|
|
|
help_msg = ("Format of the disk. Valid values: %s" % disk_choices)
|
|
|
|
parser.add_argument(
|
|
|
|
"--disk-format",
|
|
|
|
metavar="<disk-format>",
|
|
|
|
choices=disk_choices,
|
|
|
|
help=help_msg
|
|
|
|
)
|
|
|
|
parser.add_argument(
|
|
|
|
"--os-distro",
|
|
|
|
metavar="<os-distro>",
|
|
|
|
help="Common name of operating system distribution"
|
|
|
|
)
|
|
|
|
parser.add_argument(
|
|
|
|
"--owner",
|
|
|
|
metavar="<owner>",
|
|
|
|
help="New Owner of the image"
|
|
|
|
)
|
|
|
|
msg = ("ID of image stored in Glance that should be used as the "
|
|
|
|
"ramdisk when booting an AMI-style image")
|
|
|
|
parser.add_argument(
|
|
|
|
"--ramdisk-id",
|
|
|
|
metavar="<ramdisk-id>",
|
|
|
|
help=msg
|
|
|
|
)
|
|
|
|
parser.add_argument(
|
|
|
|
"--min-ram",
|
|
|
|
type=int,
|
|
|
|
metavar="<ram-mb>",
|
|
|
|
help="Amount of RAM (in MB) required to boot image"
|
|
|
|
)
|
|
|
|
container_choices = ["None", "ami", "ari", "aki", "bare", "ovf", "ova"]
|
|
|
|
help_msg = ("Format of the container. Valid values: %s"
|
|
|
|
% container_choices)
|
|
|
|
parser.add_argument(
|
|
|
|
"--container-format",
|
|
|
|
metavar="<container-format>",
|
|
|
|
choices=container_choices,
|
|
|
|
help=help_msg
|
|
|
|
)
|
|
|
|
return parser
|
|
|
|
|
|
|
|
def take_action(self, parsed_args):
|
|
|
|
self.log.debug("take_action(%s)", parsed_args)
|
|
|
|
image_client = self.app.client_manager.image
|
|
|
|
|
|
|
|
kwargs = {}
|
|
|
|
copy_attrs = ('architecture', 'container_format', 'disk_format',
|
|
|
|
'file', 'kernel_id', 'locations', 'name',
|
|
|
|
'min_disk', 'min_ram', 'name', 'os_distro', 'os_version',
|
|
|
|
'owner', 'prefix', 'progress', 'ramdisk_id',
|
|
|
|
'visibility')
|
|
|
|
for attr in copy_attrs:
|
|
|
|
if attr in parsed_args:
|
|
|
|
val = getattr(parsed_args, attr, None)
|
|
|
|
if val:
|
|
|
|
# Only include a value in kwargs for attributes that are
|
|
|
|
# actually present on the command line
|
|
|
|
kwargs[attr] = val
|
|
|
|
if parsed_args.protected:
|
|
|
|
kwargs['protected'] = True
|
|
|
|
else:
|
|
|
|
kwargs['protected'] = False
|
|
|
|
|
|
|
|
if not kwargs:
|
|
|
|
self.log.warning("No arguments specified")
|
|
|
|
return {}, {}
|
|
|
|
|
|
|
|
image = utils.find_resource(
|
|
|
|
image_client.images, parsed_args.image)
|
|
|
|
|
|
|
|
image = image_client.images.update(image.id, **kwargs)
|
|
|
|
info = {}
|
|
|
|
info.update(image)
|
|
|
|
return zip(*sorted(six.iteritems(info)))
|