2012-03-26 22:48:48 -07:00
|
|
|
# Copyright 2012 OpenStack LLC.
|
|
|
|
# All Rights Reserved.
|
|
|
|
#
|
|
|
|
# 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.
|
|
|
|
|
2012-08-01 16:04:37 +00:00
|
|
|
import errno
|
|
|
|
import hashlib
|
2012-03-26 22:48:48 -07:00
|
|
|
import os
|
2012-07-13 22:03:22 +00:00
|
|
|
import sys
|
2012-02-29 16:42:26 -05:00
|
|
|
import uuid
|
|
|
|
|
|
|
|
import prettytable
|
|
|
|
|
2012-07-12 18:30:54 -07:00
|
|
|
from glanceclient import exc
|
2012-05-17 14:33:43 -07:00
|
|
|
from glanceclient.openstack.common import importutils
|
2013-05-22 11:31:25 +02:00
|
|
|
from glanceclient.openstack.common import strutils
|
2012-02-29 16:42:26 -05:00
|
|
|
|
|
|
|
|
|
|
|
# Decorator for cli-args
|
|
|
|
def arg(*args, **kwargs):
|
|
|
|
def _decorator(func):
|
|
|
|
# Because of the sematics of decorator composition if we just append
|
|
|
|
# to the options list positional options will appear to be backwards.
|
|
|
|
func.__dict__.setdefault('arguments', []).insert(0, (args, kwargs))
|
|
|
|
return func
|
|
|
|
return _decorator
|
|
|
|
|
|
|
|
|
|
|
|
def pretty_choice_list(l):
|
|
|
|
return ', '.join("'%s'" % i for i in l)
|
|
|
|
|
|
|
|
|
|
|
|
def print_list(objs, fields, formatters={}):
|
|
|
|
pt = prettytable.PrettyTable([f for f in fields], caching=False)
|
2012-06-07 14:41:14 -07:00
|
|
|
pt.align = 'l'
|
2012-02-29 16:42:26 -05:00
|
|
|
|
|
|
|
for o in objs:
|
|
|
|
row = []
|
|
|
|
for field in fields:
|
|
|
|
if field in formatters:
|
|
|
|
row.append(formatters[field](o))
|
|
|
|
else:
|
|
|
|
field_name = field.lower().replace(' ', '_')
|
2012-07-13 22:03:22 +00:00
|
|
|
data = getattr(o, field_name, None) or ''
|
2012-02-29 16:42:26 -05:00
|
|
|
row.append(data)
|
|
|
|
pt.add_row(row)
|
|
|
|
|
2013-05-22 11:31:25 +02:00
|
|
|
print strutils.safe_encode(pt.get_string())
|
2012-02-29 16:42:26 -05:00
|
|
|
|
|
|
|
|
|
|
|
def print_dict(d):
|
|
|
|
pt = prettytable.PrettyTable(['Property', 'Value'], caching=False)
|
2012-07-14 01:54:29 +00:00
|
|
|
pt.align = 'l'
|
2012-02-29 16:42:26 -05:00
|
|
|
[pt.add_row(list(r)) for r in d.iteritems()]
|
2013-05-22 11:31:25 +02:00
|
|
|
print strutils.safe_encode(pt.get_string(sortby='Property'))
|
2012-02-29 16:42:26 -05:00
|
|
|
|
|
|
|
|
|
|
|
def find_resource(manager, name_or_id):
|
|
|
|
"""Helper for the _find_* methods."""
|
|
|
|
# first try to get entity as integer id
|
|
|
|
try:
|
|
|
|
if isinstance(name_or_id, int) or name_or_id.isdigit():
|
|
|
|
return manager.get(int(name_or_id))
|
2012-07-12 18:30:54 -07:00
|
|
|
except exc.NotFound:
|
2012-02-29 16:42:26 -05:00
|
|
|
pass
|
|
|
|
|
|
|
|
# now try to get entity as uuid
|
|
|
|
try:
|
2013-05-22 11:31:25 +02:00
|
|
|
uuid.UUID(strutils.safe_encode(name_or_id))
|
2012-02-29 16:42:26 -05:00
|
|
|
return manager.get(name_or_id)
|
2012-07-12 18:30:54 -07:00
|
|
|
except (ValueError, exc.NotFound):
|
2012-02-29 16:42:26 -05:00
|
|
|
pass
|
|
|
|
|
|
|
|
# finally try to find entity by name
|
2013-01-05 07:32:25 +09:00
|
|
|
matches = list(manager.list(filters={'name': name_or_id}))
|
|
|
|
num_matches = len(matches)
|
|
|
|
if num_matches == 0:
|
2012-02-29 16:42:26 -05:00
|
|
|
msg = "No %s with a name or ID of '%s' exists." % \
|
|
|
|
(manager.resource_class.__name__.lower(), name_or_id)
|
2012-07-12 18:30:54 -07:00
|
|
|
raise exc.CommandError(msg)
|
2013-01-05 07:32:25 +09:00
|
|
|
elif num_matches > 1:
|
|
|
|
msg = ("Multiple %s matches found for '%s', use an ID to be more"
|
|
|
|
" specific." % (manager.resource_class.__name__.lower(),
|
|
|
|
name_or_id))
|
|
|
|
raise exc.CommandError(msg)
|
|
|
|
else:
|
|
|
|
return matches[0]
|
2012-02-29 16:42:26 -05:00
|
|
|
|
|
|
|
|
2012-03-26 22:48:48 -07:00
|
|
|
def skip_authentication(f):
|
|
|
|
"""Function decorator used to indicate a caller may be unauthenticated."""
|
|
|
|
f.require_authentication = False
|
2012-02-29 16:42:26 -05:00
|
|
|
return f
|
|
|
|
|
|
|
|
|
2012-03-26 22:48:48 -07:00
|
|
|
def is_authentication_required(f):
|
|
|
|
"""Checks to see if the function requires authentication.
|
|
|
|
|
|
|
|
Use the skip_authentication decorator to indicate a caller may
|
|
|
|
skip the authentication step.
|
2012-02-29 16:42:26 -05:00
|
|
|
"""
|
2012-03-26 22:48:48 -07:00
|
|
|
return getattr(f, 'require_authentication', True)
|
2012-02-29 16:42:26 -05:00
|
|
|
|
|
|
|
|
|
|
|
def string_to_bool(arg):
|
|
|
|
return arg.strip().lower() in ('t', 'true', 'yes', '1')
|
2012-03-26 22:48:48 -07:00
|
|
|
|
|
|
|
|
|
|
|
def env(*vars, **kwargs):
|
|
|
|
"""Search for the first defined of possibly many env vars
|
|
|
|
|
|
|
|
Returns the first environment variable defined in vars, or
|
|
|
|
returns the default defined in kwargs.
|
|
|
|
"""
|
|
|
|
for v in vars:
|
|
|
|
value = os.environ.get(v, None)
|
|
|
|
if value:
|
|
|
|
return value
|
|
|
|
return kwargs.get('default', '')
|
2012-05-17 14:33:43 -07:00
|
|
|
|
|
|
|
|
|
|
|
def import_versioned_module(version, submodule=None):
|
|
|
|
module = 'glanceclient.v%s' % version
|
|
|
|
if submodule:
|
|
|
|
module = '.'.join((module, submodule))
|
|
|
|
return importutils.import_module(module)
|
2012-07-13 22:03:22 +00:00
|
|
|
|
|
|
|
|
|
|
|
def exit(msg=''):
|
|
|
|
if msg:
|
2013-05-22 11:31:25 +02:00
|
|
|
print >> sys.stderr, strutils.safe_encode(msg)
|
2012-07-13 22:03:22 +00:00
|
|
|
sys.exit(1)
|
2012-08-01 16:04:37 +00:00
|
|
|
|
|
|
|
|
|
|
|
def save_image(data, path):
|
|
|
|
"""
|
|
|
|
Save an image to the specified path.
|
|
|
|
|
|
|
|
:param data: binary data of the image
|
|
|
|
:param path: path to save the image to
|
|
|
|
"""
|
|
|
|
if path is None:
|
|
|
|
image = sys.stdout
|
|
|
|
else:
|
|
|
|
image = open(path, 'wb')
|
|
|
|
try:
|
|
|
|
for chunk in data:
|
|
|
|
image.write(chunk)
|
|
|
|
finally:
|
|
|
|
if path is not None:
|
|
|
|
image.close()
|
|
|
|
|
|
|
|
|
|
|
|
def integrity_iter(iter, checksum):
|
|
|
|
"""
|
|
|
|
Check image data integrity.
|
|
|
|
|
|
|
|
:raises: IOError
|
|
|
|
"""
|
|
|
|
md5sum = hashlib.md5()
|
|
|
|
for chunk in iter:
|
|
|
|
yield chunk
|
|
|
|
md5sum.update(chunk)
|
|
|
|
md5sum = md5sum.hexdigest()
|
|
|
|
if md5sum != checksum:
|
|
|
|
raise IOError(errno.EPIPE,
|
|
|
|
'Corrupt image download. Checksum was %s expected %s' %
|
|
|
|
(md5sum, checksum))
|
2012-11-07 19:39:43 +01:00
|
|
|
|
|
|
|
|
|
|
|
def make_size_human_readable(size):
|
|
|
|
suffix = ['B', 'kB', 'MB', 'GB', 'TB', 'PB', 'EB', 'ZB']
|
|
|
|
base = 1024.0
|
|
|
|
|
|
|
|
index = 0
|
2012-11-19 10:35:04 -08:00
|
|
|
while size >= base:
|
2012-11-07 19:39:43 +01:00
|
|
|
index = index + 1
|
|
|
|
size = size / base
|
|
|
|
|
2012-11-19 10:35:04 -08:00
|
|
|
padded = '%.1f' % size
|
|
|
|
stripped = padded.rstrip('0').rstrip('.')
|
|
|
|
|
|
|
|
return '%s%s' % (stripped, suffix[index])
|
2013-01-30 15:18:44 +01:00
|
|
|
|
|
|
|
|
2013-03-20 18:00:39 +00:00
|
|
|
def getsockopt(self, *args, **kwargs):
|
|
|
|
"""
|
|
|
|
A function which allows us to monkey patch eventlet's
|
|
|
|
GreenSocket, adding a required 'getsockopt' method.
|
|
|
|
TODO: (mclaren) we can remove this once the eventlet fix
|
|
|
|
(https://bitbucket.org/eventlet/eventlet/commits/609f230)
|
|
|
|
lands in mainstream packages.
|
|
|
|
"""
|
|
|
|
return self.fd.getsockopt(*args, **kwargs)
|
2013-07-11 15:28:49 +02:00
|
|
|
|
|
|
|
|
|
|
|
def exception_to_str(exc):
|
|
|
|
try:
|
|
|
|
error = unicode(exc)
|
|
|
|
except UnicodeError:
|
|
|
|
try:
|
|
|
|
error = str(exc)
|
|
|
|
except UnicodeError:
|
|
|
|
error = ("Caught '%(exception)s' exception." %
|
|
|
|
{"exception": exc.__class__.__name__})
|
|
|
|
return strutils.safe_encode(error, errors='ignore')
|