Merge "Complete Image v1"

This commit is contained in:
Jenkins 2013-07-20 20:17:00 +00:00 committed by Gerrit Code Review
commit 7b47579dad
6 changed files with 382 additions and 59 deletions

@ -1,4 +1,4 @@
# Copyright 2012-2013 OpenStack, LLC. # Copyright 2012-2013 OpenStack Foundation
# #
# Licensed under the Apache License, Version 2.0 (the "License"); you may # 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 # not use this file except in compliance with the License. You may obtain
@ -15,6 +15,9 @@
import logging import logging
from glanceclient import exc as gc_exceptions
from glanceclient.v1 import client as gc_v1_client
from glanceclient.v1 import images as gc_v1_images
from openstackclient.common import utils from openstackclient.common import utils
@ -22,8 +25,8 @@ LOG = logging.getLogger(__name__)
API_NAME = "image" API_NAME = "image"
API_VERSIONS = { API_VERSIONS = {
"1": "glanceclient.v1.client.Client", "1": "openstackclient.image.client.Client_v1",
"2": "glanceclient.v2.client.Client" "2": "glanceclient.v2.client.Client",
} }
@ -38,3 +41,54 @@ def make_client(instance):
instance._url = instance.get_endpoint_for_service_type(API_NAME) instance._url = instance.get_endpoint_for_service_type(API_NAME)
return image_client(instance._url, token=instance._token) return image_client(instance._url, token=instance._token)
# NOTE(dtroyer): glanceclient.v1.image.ImageManager() doesn't have a find()
# method so add one here until the common client libs arrive
# A similar subclass will be required for v2
class Client_v1(gc_v1_client.Client):
"""An image v1 client that uses ImageManager_v1"""
def __init__(self, *args, **kwargs):
super(Client_v1, self).__init__(*args, **kwargs)
self.images = ImageManager_v1(self)
class ImageManager_v1(gc_v1_images.ImageManager):
"""Add find() and findall() to the ImageManager class"""
def find(self, **kwargs):
"""Find a single item with attributes matching ``**kwargs``.
This isn't very efficient: it loads the entire list then filters on
the Python side.
"""
rl = self.findall(**kwargs)
num = len(rl)
if num == 0:
raise gc_exceptions.NotFound
elif num > 1:
raise gc_exceptions.NoUniqueMatch
else:
return rl[0]
def findall(self, **kwargs):
"""Find all items with attributes matching ``**kwargs``.
This isn't very efficient: it loads the entire list then filters on
the Python side.
"""
found = []
searches = kwargs.items()
for obj in self.list():
try:
if all(getattr(obj, attr) == value
for (attr, value) in searches):
found.append(obj)
except AttributeError:
continue
return found

@ -1,4 +1,4 @@
# Copyright 2013 OpenStack, LLC. # Copyright 2012-2013 OpenStack Foundation
# #
# Licensed under the Apache License, Version 2.0 (the "License"); you may # 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 # not use this file except in compliance with the License. You may obtain
@ -17,6 +17,7 @@
import logging import logging
import os import os
import six
import sys import sys
if os.name == "nt": if os.name == "nt":
@ -24,11 +25,18 @@ if os.name == "nt":
else: else:
msvcrt = None msvcrt = None
from cliff import command
from cliff import lister
from cliff import show from cliff import show
from glanceclient.common import utils as gc_utils
from openstackclient.common import exceptions
from openstackclient.common import parseractions
from openstackclient.common import utils
class CreateImage(show.ShowOne): class CreateImage(show.ShowOne):
"""Create image command""" """Create/upload an image"""
log = logging.getLogger(__name__ + ".CreateImage") log = logging.getLogger(__name__ + ".CreateImage")
@ -37,89 +45,108 @@ class CreateImage(show.ShowOne):
parser.add_argument( parser.add_argument(
"name", "name",
metavar="<name>", metavar="<name>",
help="Name of image.") help="Name of image",
)
parser.add_argument( parser.add_argument(
"--disk_format", "--disk_format",
default="raw", default="raw",
metavar="<disk_format>", metavar="<disk_format>",
help="Disk format of image.") help="Disk format of image",
)
parser.add_argument( parser.add_argument(
"--id", "--id",
metavar="<id>", metavar="<id>",
help="ID of image to reserve.") help="ID of image to reserve",
)
parser.add_argument( parser.add_argument(
"--store", "--store",
metavar="<store>", metavar="<store>",
help="Store to upload image to.") help="Store to upload image to",
)
parser.add_argument( parser.add_argument(
"--container-format", "--container-format",
default="bare", default="bare",
metavar="<container_format>", metavar="<container_format>",
help="Container format of image.") help="Container format of image",
)
parser.add_argument( parser.add_argument(
"--owner", "--owner",
metavar="<tenant_id>", metavar="<tenant>",
help="Owner of the image.") help="Owner of the image",
)
parser.add_argument( parser.add_argument(
"--size", "--size",
metavar="<size>", metavar="<size>",
help="Size of image in bytes. Only used with --location and" help="Size of image in bytes. Only used with --location and"
" --copy-from.") " --copy-from",
)
parser.add_argument( parser.add_argument(
"--min-disk", "--min-disk",
metavar="<disk_gb>", metavar="<disk_gb>",
help="Minimum size of disk needed to boot image in gigabytes.") help="Minimum size of disk needed to boot image in gigabytes",
)
parser.add_argument( parser.add_argument(
"--min-ram", "--min-ram",
metavar="<disk_ram>", metavar="<disk_ram>",
help="Minimum amount of ram needed to boot image in megabytes.") help="Minimum amount of ram needed to boot image in megabytes",
)
parser.add_argument( parser.add_argument(
"--location", "--location",
metavar="<image_url>", metavar="<image_url>",
help="URL where the data for this image already resides.") help="URL where the data for this image already resides",
)
parser.add_argument( parser.add_argument(
"--file", "--file",
metavar="<file>", metavar="<file>",
help="Local file that contains disk image.") help="Local file that contains disk image",
)
parser.add_argument( parser.add_argument(
"--checksum", "--checksum",
metavar="<checksum>", metavar="<checksum>",
help="Hash of image data used for verification.") help="Hash of image data used for verification",
)
parser.add_argument( parser.add_argument(
"--copy-from", "--copy-from",
metavar="<image_url>", metavar="<image_url>",
help="Similar to --location, but this indicates that the image" help="Similar to --location, but this indicates that the image"
" should immediately be copied from the data store.") " should immediately be copied from the data store",
)
parser.add_argument( parser.add_argument(
"--property", "--property",
dest="properties",
metavar="<key=value>", metavar="<key=value>",
default=[], action=parseractions.KeyValueAction,
action="append", help="Set property on this image "
help="Arbitrary property to associate with image.") '(repeat option to set multiple properties)',
)
protected_group = parser.add_mutually_exclusive_group() protected_group = parser.add_mutually_exclusive_group()
protected_group.add_argument( protected_group.add_argument(
"--protected", "--protected",
dest="protected", dest="protected",
action="store_true", action="store_true",
help="Prevent image from being deleted (default: False).") help="Prevent image from being deleted (default: False)",
)
protected_group.add_argument( protected_group.add_argument(
"--unprotected", "--unprotected",
dest="protected", dest="protected",
action="store_false", action="store_false",
default=False, default=False,
help="Allow images to be deleted (default: True).") help="Allow images to be deleted (default: True)",
)
public_group = parser.add_mutually_exclusive_group() public_group = parser.add_mutually_exclusive_group()
public_group.add_argument( public_group.add_argument(
"--public", "--public",
dest="is_public", dest="is_public",
action="store_true", action="store_true",
default=True, default=True,
help="Image is accessible to the public (default).") help="Image is accessible to the public (default)",
)
public_group.add_argument( public_group.add_argument(
"--private", "--private",
dest="is_public", dest="is_public",
action="store_false", action="store_false",
help="Image is inaccessible to the public.") help="Image is inaccessible to the public",
)
return parser return parser
def take_action(self, parsed_args): def take_action(self, parsed_args):
@ -134,11 +161,6 @@ class CreateImage(show.ShowOne):
args.pop("prefix") args.pop("prefix")
args.pop("variables") args.pop("variables")
args["properties"] = {}
for _property in args.pop("property"):
key, value = _property.split("=", 1)
args["properties"][key] = value
if "location" not in args and "copy_from" not in args: if "location" not in args and "copy_from" not in args:
if "file" in args: if "file" in args:
args["data"] = open(args.pop("file"), "rb") args["data"] = open(args.pop("file"), "rb")
@ -150,6 +172,231 @@ class CreateImage(show.ShowOne):
args["data"] = sys.stdin args["data"] = sys.stdin
image_client = self.app.client_manager.image image_client = self.app.client_manager.image
data = image_client.images.create(**args)._info.copy() try:
image = utils.find_resource(
image_client.images,
parsed_args.name,
)
except exceptions.CommandError:
# This is normal for a create or reserve (create w/o an image)
image = image_client.images.create(**args)
else:
# It must be an update
# If an image is specified via --file, --location or --copy-from
# let the API handle it
image = image_client.images.update(image, **args)
return zip(*sorted(data.iteritems())) info = {}
info.update(image._info)
return zip(*sorted(six.iteritems(info)))
class DeleteImage(command.Command):
"""Delete an image"""
log = logging.getLogger(__name__ + ".DeleteImage")
def get_parser(self, prog_name):
parser = super(DeleteImage, self).get_parser(prog_name)
parser.add_argument(
"image",
metavar="<image>",
help="Name or ID of image to delete",
)
return parser
def take_action(self, parsed_args):
self.log.debug("take_action(%s)" % parsed_args)
image_client = self.app.client_manager.image
image = utils.find_resource(
image_client.images,
parsed_args.image,
)
image_client.images.delete(image)
class ListImage(lister.Lister):
"""List available images"""
log = logging.getLogger(__name__ + ".ListImage")
def get_parser(self, prog_name):
parser = super(ListImage, self).get_parser(prog_name)
parser.add_argument(
"--page-size",
metavar="<size>",
help="Number of images to request in each paginated request",
)
return parser
def take_action(self, parsed_args):
self.log.debug("take_action(%s)" % parsed_args)
image_client = self.app.client_manager.image
kwargs = {}
if parsed_args.page_size is not None:
kwargs["page_size"] = parsed_args.page_size
data = image_client.images.list(**kwargs)
columns = ["ID", "Name"]
return (columns, (utils.get_item_properties(s, columns) for s in data))
class SaveImage(command.Command):
"""Save an image locally"""
log = logging.getLogger(__name__ + ".SaveImage")
def get_parser(self, prog_name):
parser = super(SaveImage, self).get_parser(prog_name)
parser.add_argument(
"--file",
metavar="<filename>",
help="Downloaded image save filename [default: stdout]",
)
parser.add_argument(
"image",
metavar="<image>",
help="Name or ID of image to delete",
)
return parser
def take_action(self, parsed_args):
self.log.debug("take_action(%s)" % parsed_args)
image_client = self.app.client_manager.image
image = utils.find_resource(
image_client.images,
parsed_args.image,
)
data = image_client.images.data(image)
gc_utils.save_image(data, parsed_args.file)
class SetImage(show.ShowOne):
"""Change 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="Name or ID of image to change",
)
parser.add_argument(
"--name",
metavar="<name>",
help="Name of image",
)
parser.add_argument(
"--owner",
metavar="<tenant>",
help="Owner of the image",
)
parser.add_argument(
"--min-disk",
metavar="<disk_gb>",
help="Minimum size of disk needed to boot image in gigabytes",
)
parser.add_argument(
"--min-ram",
metavar="<disk_ram>",
help="Minimum amount of ram needed to boot image in megabytes",
)
parser.add_argument(
"--property",
dest="properties",
metavar="<key=value>",
action=parseractions.KeyValueAction,
help="Set property on this image "
'(repeat option to set multiple properties)',
)
protected_group = parser.add_mutually_exclusive_group()
protected_group.add_argument(
"--protected",
dest="protected",
action="store_true",
help="Prevent image from being deleted (default: False)",
)
protected_group.add_argument(
"--unprotected",
dest="protected",
action="store_false",
default=False,
help="Allow images to be deleted (default: True)",
)
public_group = parser.add_mutually_exclusive_group()
public_group.add_argument(
"--public",
dest="is_public",
action="store_true",
default=True,
help="Image is accessible to the public (default)",
)
public_group.add_argument(
"--private",
dest="is_public",
action="store_false",
help="Image is inaccessible to the public",
)
return parser
def take_action(self, parsed_args):
self.log.debug("take_action(%s)" % parsed_args)
# NOTE(jk0): Since create() takes kwargs, it's easiest to just make a
# copy of parsed_args and remove what we don't need.
args = vars(parsed_args)
args = dict(filter(lambda x: x[1] is not None, args.items()))
args.pop("columns")
args.pop("formatter")
args.pop("prefix")
args.pop("variables")
image_arg = args.pop("image")
image_client = self.app.client_manager.image
image = utils.find_resource(
image_client.images,
image_arg,
)
# Merge properties
args["properties"].update(image.properties)
image = image_client.images.update(image, **args)
info = {}
info.update(image._info)
return zip(*sorted(six.iteritems(info)))
class ShowImage(show.ShowOne):
"""Show image details"""
log = logging.getLogger(__name__ + ".ShowImage")
def get_parser(self, prog_name):
parser = super(ShowImage, self).get_parser(prog_name)
parser.add_argument(
"image",
metavar="<image>",
help="Name or ID of image to display",
)
return parser
def take_action(self, parsed_args):
self.log.debug("take_action(%s)" % parsed_args)
image_client = self.app.client_manager.image
image = utils.find_resource(
image_client.images,
parsed_args.image,
)
info = {}
info.update(image._info)
return zip(*sorted(six.iteritems(info)))

@ -1,4 +1,4 @@
# Copyright 2012-2013 OpenStack, LLC. # Copyright 2012-2013 OpenStack Foundation
# #
# Licensed under the Apache License, Version 2.0 (the "License"); you may # 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 # not use this file except in compliance with the License. You may obtain
@ -16,6 +16,7 @@
"""Image V2 Action Implementations""" """Image V2 Action Implementations"""
import logging import logging
import six
from cliff import command from cliff import command
from cliff import lister from cliff import lister
@ -26,27 +27,32 @@ from openstackclient.common import utils
class DeleteImage(command.Command): class DeleteImage(command.Command):
"""Delete image command""" """Delete an image"""
log = logging.getLogger(__name__ + ".DeleteImage") log = logging.getLogger(__name__ + ".DeleteImage")
def get_parser(self, prog_name): def get_parser(self, prog_name):
parser = super(DeleteImage, self).get_parser(prog_name) parser = super(DeleteImage, self).get_parser(prog_name)
parser.add_argument( parser.add_argument(
"id", "image",
metavar="<image_id>", metavar="<image>",
help="ID of image to delete.") help="Name or ID of image to delete",
)
return parser return parser
def take_action(self, parsed_args): def take_action(self, parsed_args):
self.log.debug("take_action(%s)" % parsed_args) self.log.debug("take_action(%s)" % parsed_args)
image_client = self.app.client_manager.image image_client = self.app.client_manager.image
image_client.images.delete(parsed_args.id) image = utils.find_resource(
image_client.images,
parsed_args.image,
)
image_client.images.delete(image)
class ListImage(lister.Lister): class ListImage(lister.Lister):
"""List image command""" """List available images"""
log = logging.getLogger(__name__ + ".ListImage") log = logging.getLogger(__name__ + ".ListImage")
@ -55,7 +61,8 @@ class ListImage(lister.Lister):
parser.add_argument( parser.add_argument(
"--page-size", "--page-size",
metavar="<size>", metavar="<size>",
help="Number of images to request in each paginated request.") help="Number of images to request in each paginated request",
)
return parser return parser
def take_action(self, parsed_args): def take_action(self, parsed_args):
@ -74,7 +81,7 @@ class ListImage(lister.Lister):
class SaveImage(command.Command): class SaveImage(command.Command):
"""Save image command""" """Save an image locally"""
log = logging.getLogger(__name__ + ".SaveImage") log = logging.getLogger(__name__ + ".SaveImage")
@ -82,42 +89,52 @@ class SaveImage(command.Command):
parser = super(SaveImage, self).get_parser(prog_name) parser = super(SaveImage, self).get_parser(prog_name)
parser.add_argument( parser.add_argument(
"--file", "--file",
metavar="<file>", metavar="<filename>",
help="Local file to save downloaded image data " help="Downloaded image save filename [default: stdout]",
"to. If this is not specified the image " )
"data will be written to stdout.")
parser.add_argument( parser.add_argument(
"id", "image",
metavar="<image_id>", metavar="<image>",
help="ID of image to describe.") help="Name or ID of image to delete",
)
return parser return parser
def take_action(self, parsed_args): def take_action(self, parsed_args):
self.log.debug("take_action(%s)" % parsed_args) self.log.debug("take_action(%s)" % parsed_args)
image_client = self.app.client_manager.image image_client = self.app.client_manager.image
data = image_client.images.data(parsed_args.id) image = utils.find_resource(
image_client.images,
parsed_args.image,
)
data = image_client.images.data(image)
gc_utils.save_image(data, parsed_args.file) gc_utils.save_image(data, parsed_args.file)
class ShowImage(show.ShowOne): class ShowImage(show.ShowOne):
"""Show image command""" """Show image details"""
log = logging.getLogger(__name__ + ".ShowImage") log = logging.getLogger(__name__ + ".ShowImage")
def get_parser(self, prog_name): def get_parser(self, prog_name):
parser = super(ShowImage, self).get_parser(prog_name) parser = super(ShowImage, self).get_parser(prog_name)
parser.add_argument( parser.add_argument(
"id", "image",
metavar="<image_id>", metavar="<image>",
help="ID of image to describe.") help="Name or ID of image to display",
)
return parser return parser
def take_action(self, parsed_args): def take_action(self, parsed_args):
self.log.debug("take_action(%s)" % parsed_args) self.log.debug("take_action(%s)" % parsed_args)
image_client = self.app.client_manager.image image_client = self.app.client_manager.image
data = image_client.images.get(parsed_args.id) image = utils.find_resource(
image_client.images,
parsed_args.image,
)
return zip(*sorted(data.iteritems())) info = {}
info.update(image._info)
return zip(*sorted(six.iteritems(info)))

@ -1,4 +1,4 @@
# Copyright 2012-2013 OpenStack, LLC. # Copyright 2012-2013 OpenStack Foundation
# #
# Licensed under the Apache License, Version 2.0 (the "License"); you may # 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 # not use this file except in compliance with the License. You may obtain
@ -35,7 +35,7 @@ KEYRING_SERVICE = 'openstack'
DEFAULT_COMPUTE_API_VERSION = '2' DEFAULT_COMPUTE_API_VERSION = '2'
DEFAULT_IDENTITY_API_VERSION = '2.0' DEFAULT_IDENTITY_API_VERSION = '2.0'
DEFAULT_IMAGE_API_VERSION = '2' DEFAULT_IMAGE_API_VERSION = '1'
DEFAULT_VOLUME_API_VERSION = '1' DEFAULT_VOLUME_API_VERSION = '1'
DEFAULT_DOMAIN = 'default' DEFAULT_DOMAIN = 'default'

@ -36,7 +36,7 @@ DEFAULT_VOLUME_API_VERSION = "1"
LIB_COMPUTE_API_VERSION = "2" LIB_COMPUTE_API_VERSION = "2"
LIB_IDENTITY_API_VERSION = "2.0" LIB_IDENTITY_API_VERSION = "2.0"
LIB_IMAGE_API_VERSION = "2" LIB_IMAGE_API_VERSION = "1"
LIB_VOLUME_API_VERSION = "1" LIB_VOLUME_API_VERSION = "1"

@ -149,6 +149,11 @@ openstack.identity.v3 =
openstack.image.v1 = openstack.image.v1 =
image_create = openstackclient.image.v1.image:CreateImage image_create = openstackclient.image.v1.image:CreateImage
image_delete = openstackclient.image.v1.image:DeleteImage
image_list = openstackclient.image.v1.image:ListImage
image_save = openstackclient.image.v1.image:SaveImage
image_set = openstackclient.image.v1.image:SetImage
image_show = openstackclient.image.v1.image:ShowImage
openstack.image.v2 = openstack.image.v2 =
image_delete = openstackclient.image.v2.image:DeleteImage image_delete = openstackclient.image.v2.image:DeleteImage