Adds image create
and delete
functionality.
We use the V1 API for `create` since it does not yet exist in the V2 API in glanceclient. For blueprint glance-client. Change-Id: Ifa819c14f6a013f4530d16247a671e5a1c740a28
This commit is contained in:
parent
95bf187a4f
commit
6da61e03f0
@ -22,7 +22,7 @@ LOG = logging.getLogger(__name__)
|
|||||||
|
|
||||||
API_NAME = "image"
|
API_NAME = "image"
|
||||||
API_VERSIONS = {
|
API_VERSIONS = {
|
||||||
"1.0": "glanceclient.v2.client.Client",
|
"1": "glanceclient.v1.client.Client",
|
||||||
"2": "glanceclient.v2.client.Client"
|
"2": "glanceclient.v2.client.Client"
|
||||||
}
|
}
|
||||||
|
|
||||||
|
14
openstackclient/image/v1/__init__.py
Normal file
14
openstackclient/image/v1/__init__.py
Normal file
@ -0,0 +1,14 @@
|
|||||||
|
# Copyright 2013 OpenStack, LLC.
|
||||||
|
#
|
||||||
|
# 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.
|
||||||
|
#
|
156
openstackclient/image/v1/image.py
Normal file
156
openstackclient/image/v1/image.py
Normal file
@ -0,0 +1,156 @@
|
|||||||
|
# Copyright 2013 OpenStack, LLC.
|
||||||
|
#
|
||||||
|
# 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.
|
||||||
|
#
|
||||||
|
|
||||||
|
"""Image V1 Action Implementations"""
|
||||||
|
|
||||||
|
import logging
|
||||||
|
import os
|
||||||
|
import sys
|
||||||
|
|
||||||
|
if os.name == "nt":
|
||||||
|
import msvcrt
|
||||||
|
else:
|
||||||
|
msvcrt = None
|
||||||
|
|
||||||
|
from cliff import show
|
||||||
|
|
||||||
|
|
||||||
|
class CreateImage(show.ShowOne):
|
||||||
|
"""Create image command"""
|
||||||
|
|
||||||
|
api = "image"
|
||||||
|
log = logging.getLogger(__name__ + ".CreateImage")
|
||||||
|
|
||||||
|
def get_parser(self, prog_name):
|
||||||
|
parser = super(CreateImage, self).get_parser(prog_name)
|
||||||
|
parser.add_argument(
|
||||||
|
"name",
|
||||||
|
metavar="<name>",
|
||||||
|
help="Name of image.")
|
||||||
|
parser.add_argument(
|
||||||
|
"--disk_format",
|
||||||
|
default="raw",
|
||||||
|
metavar="<disk_format>",
|
||||||
|
help="Disk format of image.")
|
||||||
|
parser.add_argument(
|
||||||
|
"--id",
|
||||||
|
metavar="<id>",
|
||||||
|
help="ID of image to reserve.")
|
||||||
|
parser.add_argument(
|
||||||
|
"--store",
|
||||||
|
metavar="<store>",
|
||||||
|
help="Store to upload image to.")
|
||||||
|
parser.add_argument(
|
||||||
|
"--container-format",
|
||||||
|
default="bare",
|
||||||
|
metavar="<container_format>",
|
||||||
|
help="Container format of image.")
|
||||||
|
parser.add_argument(
|
||||||
|
"--owner",
|
||||||
|
metavar="<tenant_id>",
|
||||||
|
help="Owner of the image.")
|
||||||
|
parser.add_argument(
|
||||||
|
"--size",
|
||||||
|
metavar="<size>",
|
||||||
|
help="Size of image in bytes. Only used with --location and"
|
||||||
|
" --copy-from.")
|
||||||
|
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(
|
||||||
|
"--location",
|
||||||
|
metavar="<image_url>",
|
||||||
|
help="URL where the data for this image already resides.")
|
||||||
|
parser.add_argument(
|
||||||
|
"--file",
|
||||||
|
metavar="<file>",
|
||||||
|
help="Local file that contains disk image.")
|
||||||
|
parser.add_argument(
|
||||||
|
"--checksum",
|
||||||
|
metavar="<checksum>",
|
||||||
|
help="Hash of image data used for verification.")
|
||||||
|
parser.add_argument(
|
||||||
|
"--copy-from",
|
||||||
|
metavar="<image_url>",
|
||||||
|
help="Similar to --location, but this indicates that the image"
|
||||||
|
" should immediately be copied from the data store.")
|
||||||
|
parser.add_argument(
|
||||||
|
"--metadata",
|
||||||
|
metavar="<key=value>",
|
||||||
|
default=[],
|
||||||
|
action="append",
|
||||||
|
help="Arbitrary metadata to associate with image.")
|
||||||
|
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")
|
||||||
|
|
||||||
|
args["properties"] = {}
|
||||||
|
for _metadata in args.pop("metadata"):
|
||||||
|
key, value = _metadata.split("=", 1)
|
||||||
|
args["properties"][key] = value
|
||||||
|
|
||||||
|
if "location" not in args and "copy_from" not in args:
|
||||||
|
if "file" in args:
|
||||||
|
args["data"] = open(args.pop("file"), "rb")
|
||||||
|
else:
|
||||||
|
args["data"] = None
|
||||||
|
if sys.stdin.isatty() is not True:
|
||||||
|
if msvcrt:
|
||||||
|
msvcrt.setmode(sys.stdin.fileno(), os.O_BINARY)
|
||||||
|
args["data"] = sys.stdin
|
||||||
|
|
||||||
|
image_client = self.app.client_manager.image
|
||||||
|
data = image_client.images.create(**args)._info.copy()
|
||||||
|
|
||||||
|
return zip(*sorted(data.iteritems()))
|
@ -13,7 +13,7 @@
|
|||||||
# under the License.
|
# under the License.
|
||||||
#
|
#
|
||||||
|
|
||||||
"""Image Action Implementations"""
|
"""Image V2 Action Implementations"""
|
||||||
|
|
||||||
import logging
|
import logging
|
||||||
|
|
||||||
@ -25,6 +25,27 @@ from glanceclient.common import utils as gc_utils
|
|||||||
from openstackclient.common import utils
|
from openstackclient.common import utils
|
||||||
|
|
||||||
|
|
||||||
|
class DeleteImage(command.Command):
|
||||||
|
"""Delete image command"""
|
||||||
|
|
||||||
|
api = "image"
|
||||||
|
log = logging.getLogger(__name__ + ".DeleteImage")
|
||||||
|
|
||||||
|
def get_parser(self, prog_name):
|
||||||
|
parser = super(DeleteImage, self).get_parser(prog_name)
|
||||||
|
parser.add_argument(
|
||||||
|
"id",
|
||||||
|
metavar="<image_id>",
|
||||||
|
help="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_client.images.delete(parsed_args.id)
|
||||||
|
|
||||||
|
|
||||||
class ListImage(lister.Lister):
|
class ListImage(lister.Lister):
|
||||||
"""List image command"""
|
"""List image command"""
|
||||||
|
|
||||||
|
4
setup.py
4
setup.py
@ -173,7 +173,11 @@ setuptools.setup(
|
|||||||
'set_user=openstackclient.identity.v3.user:SetUser',
|
'set_user=openstackclient.identity.v3.user:SetUser',
|
||||||
'show_user=openstackclient.identity.v3.user:ShowUser',
|
'show_user=openstackclient.identity.v3.user:ShowUser',
|
||||||
],
|
],
|
||||||
|
'openstack.image.v1': [
|
||||||
|
'create_image=openstackclient.image.v1.image:CreateImage',
|
||||||
|
],
|
||||||
'openstack.image.v2': [
|
'openstack.image.v2': [
|
||||||
|
'delete_image=openstackclient.image.v2.image:DeleteImage',
|
||||||
'list_image=openstackclient.image.v2.image:ListImage',
|
'list_image=openstackclient.image.v2.image:ListImage',
|
||||||
'save_image=openstackclient.image.v2.image:SaveImage',
|
'save_image=openstackclient.image.v2.image:SaveImage',
|
||||||
'show_image=openstackclient.image.v2.image:ShowImage',
|
'show_image=openstackclient.image.v2.image:ShowImage',
|
||||||
|
@ -1,7 +1,7 @@
|
|||||||
cliff
|
cliff
|
||||||
keyring
|
keyring
|
||||||
pycrypto
|
pycrypto
|
||||||
python-glanceclient>=0.5.1
|
python-glanceclient>=0.9.0,<2
|
||||||
python-keystoneclient>=0.2,<1.0
|
python-keystoneclient>=0.2,<1.0
|
||||||
python-novaclient>=2
|
python-novaclient>=2
|
||||||
python-cinderclient>=1
|
python-cinderclient>=1
|
||||||
|
Loading…
Reference in New Issue
Block a user