2013-09-20 04:05:51 +08:00
|
|
|
# Copyright 2012 OpenStack Foundation
|
2012-03-26 22:48:48 -07:00
|
|
|
# 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.
|
|
|
|
|
2013-08-04 16:12:23 +02:00
|
|
|
from __future__ import print_function
|
|
|
|
|
2012-08-01 16:04:37 +00:00
|
|
|
import errno
|
2014-07-01 14:45:12 +05:30
|
|
|
import hashlib
|
2014-09-03 07:37:45 -04:00
|
|
|
import json
|
2012-03-26 22:48:48 -07:00
|
|
|
import os
|
2013-10-22 15:51:49 +00:00
|
|
|
import re
|
2012-07-13 22:03:22 +00:00
|
|
|
import sys
|
2014-09-09 14:51:14 -07:00
|
|
|
import threading
|
2012-02-29 16:42:26 -05:00
|
|
|
import uuid
|
|
|
|
|
2015-02-05 22:27:16 +00:00
|
|
|
from oslo_utils import importutils
|
2014-01-10 11:13:21 +01:00
|
|
|
import six
|
|
|
|
|
2013-08-19 17:27:07 -04:00
|
|
|
if os.name == 'nt':
|
|
|
|
import msvcrt
|
|
|
|
else:
|
|
|
|
msvcrt = None
|
|
|
|
|
2015-02-05 22:27:16 +00:00
|
|
|
from oslo_utils import encodeutils
|
|
|
|
from oslo_utils import strutils
|
2012-02-29 16:42:26 -05:00
|
|
|
import prettytable
|
|
|
|
|
2012-07-12 18:30:54 -07:00
|
|
|
from glanceclient import exc
|
2012-02-29 16:42:26 -05:00
|
|
|
|
2014-09-09 14:51:14 -07:00
|
|
|
_memoized_property_lock = threading.Lock()
|
|
|
|
|
2014-09-15 16:17:18 -06:00
|
|
|
SENSITIVE_HEADERS = ('X-Auth-Token', )
|
|
|
|
|
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
|
|
|
|
|
|
|
|
|
2014-07-01 16:34:16 +08:00
|
|
|
def schema_args(schema_getter, omit=None):
|
|
|
|
omit = omit or []
|
2013-08-19 17:27:07 -04:00
|
|
|
typemap = {
|
|
|
|
'string': str,
|
|
|
|
'integer': int,
|
2014-01-22 11:45:42 +08:00
|
|
|
'boolean': strutils.bool_from_string,
|
2013-08-19 17:27:07 -04:00
|
|
|
'array': list
|
|
|
|
}
|
|
|
|
|
|
|
|
def _decorator(func):
|
|
|
|
schema = schema_getter()
|
|
|
|
if schema is None:
|
|
|
|
param = '<unavailable>'
|
|
|
|
kwargs = {
|
|
|
|
'help': ("Please run with connection parameters set to "
|
|
|
|
"retrieve the schema for generating help for this "
|
|
|
|
"command")
|
|
|
|
}
|
|
|
|
func.__dict__.setdefault('arguments', []).insert(0, ((param, ),
|
|
|
|
kwargs))
|
|
|
|
else:
|
|
|
|
properties = schema.get('properties', {})
|
2014-01-10 11:13:21 +01:00
|
|
|
for name, property in six.iteritems(properties):
|
2013-08-19 17:27:07 -04:00
|
|
|
if name in omit:
|
|
|
|
continue
|
|
|
|
param = '--' + name.replace('_', '-')
|
|
|
|
kwargs = {}
|
|
|
|
|
|
|
|
type_str = property.get('type', 'string')
|
2014-12-10 10:47:40 +01:00
|
|
|
|
|
|
|
if isinstance(type_str, list):
|
|
|
|
# NOTE(flaper87): This means the server has
|
|
|
|
# returned something like `['null', 'string']`,
|
|
|
|
# therfore we use the first non-`null` type as
|
|
|
|
# the valid type.
|
|
|
|
for t in type_str:
|
|
|
|
if t != 'null':
|
|
|
|
type_str = t
|
|
|
|
break
|
|
|
|
|
2013-08-19 17:27:07 -04:00
|
|
|
if type_str == 'array':
|
|
|
|
items = property.get('items')
|
|
|
|
kwargs['type'] = typemap.get(items.get('type'))
|
|
|
|
kwargs['nargs'] = '+'
|
|
|
|
else:
|
|
|
|
kwargs['type'] = typemap.get(type_str)
|
|
|
|
|
|
|
|
if type_str == 'boolean':
|
|
|
|
kwargs['metavar'] = '[True|False]'
|
|
|
|
else:
|
|
|
|
kwargs['metavar'] = '<%s>' % name.upper()
|
|
|
|
|
|
|
|
description = property.get('description', "")
|
|
|
|
if 'enum' in property:
|
|
|
|
if len(description):
|
|
|
|
description += " "
|
2014-12-10 10:47:40 +01:00
|
|
|
|
|
|
|
# NOTE(flaper87): Make sure all values are `str/unicode`
|
|
|
|
# for the `join` to succeed. Enum types can also be `None`
|
|
|
|
# therfore, join's call would fail without the following
|
|
|
|
# list comprehension
|
|
|
|
vals = [six.text_type(val) for val in property.get('enum')]
|
|
|
|
description += ('Valid values: ' + ', '.join(vals))
|
2013-08-19 17:27:07 -04:00
|
|
|
kwargs['help'] = description
|
|
|
|
|
|
|
|
func.__dict__.setdefault('arguments',
|
|
|
|
[]).insert(0, ((param, ), kwargs))
|
|
|
|
return func
|
|
|
|
|
|
|
|
return _decorator
|
|
|
|
|
|
|
|
|
2012-02-29 16:42:26 -05:00
|
|
|
def pretty_choice_list(l):
|
|
|
|
return ', '.join("'%s'" % i for i in l)
|
|
|
|
|
|
|
|
|
2014-09-03 07:37:45 -04:00
|
|
|
def print_list(objs, fields, formatters=None, field_settings=None):
|
2014-07-01 16:34:16 +08:00
|
|
|
formatters = formatters or {}
|
2014-09-03 07:37:45 -04:00
|
|
|
field_settings = field_settings or {}
|
2012-02-29 16:42:26 -05:00
|
|
|
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:
|
2014-09-03 07:37:45 -04:00
|
|
|
if field in field_settings:
|
|
|
|
for setting, value in six.iteritems(field_settings[field]):
|
|
|
|
setting_dict = getattr(pt, setting)
|
|
|
|
setting_dict[field] = value
|
|
|
|
|
2012-02-29 16:42:26 -05:00
|
|
|
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)
|
|
|
|
|
2015-01-06 14:32:05 +00:00
|
|
|
print(encodeutils.safe_decode(pt.get_string()))
|
2012-02-29 16:42:26 -05:00
|
|
|
|
|
|
|
|
2013-11-15 16:02:46 +08:00
|
|
|
def print_dict(d, max_column_width=80):
|
2012-02-29 16:42:26 -05:00
|
|
|
pt = prettytable.PrettyTable(['Property', 'Value'], caching=False)
|
2012-07-14 01:54:29 +00:00
|
|
|
pt.align = 'l'
|
2013-11-15 16:02:46 +08:00
|
|
|
pt.max_width = max_column_width
|
2014-09-03 07:37:45 -04:00
|
|
|
for k, v in six.iteritems(d):
|
|
|
|
if isinstance(v, (dict, list)):
|
|
|
|
v = json.dumps(v)
|
|
|
|
pt.add_row([k, v])
|
2015-01-06 14:32:05 +00:00
|
|
|
print(encodeutils.safe_decode(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:
|
2015-01-06 14:32:05 +00:00
|
|
|
# This must be unicode for Python 3 compatibility.
|
|
|
|
# If you pass a bytestring to uuid.UUID, you will get a TypeError
|
|
|
|
uuid.UUID(encodeutils.safe_decode(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
|
|
|
|
|
|
|
|
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
|
|
|
|
|
|
|
|
2014-09-25 11:19:31 +05:30
|
|
|
def exit(msg='', exit_code=1):
|
2012-07-13 22:03:22 +00:00
|
|
|
if msg:
|
2015-01-06 14:32:05 +00:00
|
|
|
print(encodeutils.safe_decode(msg), file=sys.stderr)
|
2014-09-25 11:19:31 +05:30
|
|
|
sys.exit(exit_code)
|
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()
|
|
|
|
|
|
|
|
|
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:
|
2014-02-25 16:03:34 +01:00
|
|
|
error = six.text_type(exc)
|
2013-07-11 15:28:49 +02:00
|
|
|
except UnicodeError:
|
|
|
|
try:
|
|
|
|
error = str(exc)
|
|
|
|
except UnicodeError:
|
|
|
|
error = ("Caught '%(exception)s' exception." %
|
|
|
|
{"exception": exc.__class__.__name__})
|
2015-01-06 14:32:05 +00:00
|
|
|
return encodeutils.safe_decode(error, errors='ignore')
|
2013-07-08 21:18:16 +02:00
|
|
|
|
|
|
|
|
|
|
|
def get_file_size(file_obj):
|
|
|
|
"""
|
|
|
|
Analyze file-like object and attempt to determine its size.
|
|
|
|
|
|
|
|
:param file_obj: file-like object.
|
|
|
|
:retval The file's size or None if it cannot be determined.
|
|
|
|
"""
|
2014-04-26 23:34:58 +02:00
|
|
|
if (hasattr(file_obj, 'seek') and hasattr(file_obj, 'tell') and
|
|
|
|
(six.PY2 or six.PY3 and file_obj.seekable())):
|
2013-07-08 21:18:16 +02:00
|
|
|
try:
|
|
|
|
curr = file_obj.tell()
|
|
|
|
file_obj.seek(0, os.SEEK_END)
|
|
|
|
size = file_obj.tell()
|
|
|
|
file_obj.seek(curr)
|
|
|
|
return size
|
2013-08-05 18:03:37 -03:00
|
|
|
except IOError as e:
|
2013-07-08 21:18:16 +02:00
|
|
|
if e.errno == errno.ESPIPE:
|
|
|
|
# Illegal seek. This means the file object
|
2014-05-02 16:17:43 +02:00
|
|
|
# is a pipe (e.g. the user is trying
|
2013-07-08 21:18:16 +02:00
|
|
|
# to pipe image data to the client,
|
|
|
|
# echo testdata | bin/glance add blah...), or
|
|
|
|
# that file object is empty, or that a file-like
|
|
|
|
# object which doesn't support 'seek/tell' has
|
|
|
|
# been supplied.
|
|
|
|
return
|
|
|
|
else:
|
|
|
|
raise
|
2013-08-19 17:27:07 -04:00
|
|
|
|
|
|
|
|
|
|
|
def get_data_file(args):
|
|
|
|
if args.file:
|
|
|
|
return open(args.file, 'rb')
|
|
|
|
else:
|
|
|
|
# distinguish cases where:
|
|
|
|
# (1) stdin is not valid (as in cron jobs):
|
|
|
|
# glance ... <&-
|
|
|
|
# (2) image data is provided through standard input:
|
|
|
|
# glance ... < /tmp/file or cat /tmp/file | glance ...
|
|
|
|
# (3) no image data provided:
|
|
|
|
# glance ...
|
|
|
|
try:
|
|
|
|
os.fstat(0)
|
|
|
|
except OSError:
|
|
|
|
# (1) stdin is not valid (closed...)
|
|
|
|
return None
|
|
|
|
if not sys.stdin.isatty():
|
|
|
|
# (2) image data is provided through standard input
|
|
|
|
if msvcrt:
|
|
|
|
msvcrt.setmode(sys.stdin.fileno(), os.O_BINARY)
|
|
|
|
return sys.stdin
|
|
|
|
else:
|
|
|
|
# (3) no image data provided
|
|
|
|
return None
|
2013-10-22 15:51:49 +00:00
|
|
|
|
|
|
|
|
|
|
|
def strip_version(endpoint):
|
|
|
|
"""Strip version from the last component of endpoint if present."""
|
2014-11-24 15:12:39 +01:00
|
|
|
# NOTE(flaper87): This shouldn't be necessary if
|
|
|
|
# we make endpoint the first argument. However, we
|
|
|
|
# can't do that just yet because we need to keep
|
|
|
|
# backwards compatibility.
|
|
|
|
if not isinstance(endpoint, six.string_types):
|
|
|
|
raise ValueError("Expected endpoint")
|
|
|
|
|
|
|
|
version = None
|
2013-10-22 15:51:49 +00:00
|
|
|
# Get rid of trailing '/' if present
|
2014-11-24 15:12:39 +01:00
|
|
|
endpoint = endpoint.rstrip('/')
|
2013-10-22 15:51:49 +00:00
|
|
|
url_bits = endpoint.split('/')
|
|
|
|
# regex to match 'v1' or 'v2.0' etc
|
|
|
|
if re.match('v\d+\.?\d*', url_bits[-1]):
|
2014-11-24 15:12:39 +01:00
|
|
|
version = float(url_bits[-1].lstrip('v'))
|
2013-10-22 15:51:49 +00:00
|
|
|
endpoint = '/'.join(url_bits[:-1])
|
2014-11-24 15:12:39 +01:00
|
|
|
return endpoint, version
|
2014-02-11 11:06:02 +08:00
|
|
|
|
|
|
|
|
|
|
|
def print_image(image_obj, max_col_width=None):
|
|
|
|
ignore = ['self', 'access', 'file', 'schema']
|
|
|
|
image = dict([item for item in six.iteritems(image_obj)
|
|
|
|
if item[0] not in ignore])
|
|
|
|
if str(max_col_width).isdigit():
|
|
|
|
print_dict(image, max_column_width=max_col_width)
|
|
|
|
else:
|
|
|
|
print_dict(image)
|
2014-07-01 14:45:12 +05:30
|
|
|
|
|
|
|
|
|
|
|
def integrity_iter(iter, checksum):
|
|
|
|
"""
|
|
|
|
Check image data integrity.
|
|
|
|
|
|
|
|
:raises: IOError
|
|
|
|
"""
|
|
|
|
md5sum = hashlib.md5()
|
|
|
|
for chunk in iter:
|
|
|
|
yield chunk
|
|
|
|
if isinstance(chunk, six.string_types):
|
|
|
|
chunk = six.b(chunk)
|
|
|
|
md5sum.update(chunk)
|
|
|
|
md5sum = md5sum.hexdigest()
|
|
|
|
if md5sum != checksum:
|
|
|
|
raise IOError(errno.EPIPE,
|
|
|
|
'Corrupt image download. Checksum was %s expected %s' %
|
|
|
|
(md5sum, checksum))
|
2014-09-09 14:51:14 -07:00
|
|
|
|
|
|
|
|
|
|
|
def memoized_property(fn):
|
|
|
|
attr_name = '_lazy_once_' + fn.__name__
|
|
|
|
|
|
|
|
@property
|
|
|
|
def _memoized_property(self):
|
|
|
|
if hasattr(self, attr_name):
|
|
|
|
return getattr(self, attr_name)
|
|
|
|
else:
|
|
|
|
with _memoized_property_lock:
|
|
|
|
if not hasattr(self, attr_name):
|
|
|
|
setattr(self, attr_name, fn(self))
|
|
|
|
return getattr(self, attr_name)
|
|
|
|
return _memoized_property
|
2014-09-15 16:17:18 -06:00
|
|
|
|
|
|
|
|
|
|
|
def safe_header(name, value):
|
|
|
|
if name in SENSITIVE_HEADERS:
|
|
|
|
v = value.encode('utf-8')
|
|
|
|
h = hashlib.sha1(v)
|
|
|
|
d = h.hexdigest()
|
|
|
|
return name, "{SHA1}%s" % d
|
|
|
|
else:
|
|
|
|
return name, value
|
2014-11-01 20:49:03 +00:00
|
|
|
|
|
|
|
|
|
|
|
class IterableWithLength(object):
|
|
|
|
def __init__(self, iterable, length):
|
|
|
|
self.iterable = iterable
|
|
|
|
self.length = length
|
|
|
|
|
|
|
|
def __iter__(self):
|
|
|
|
return self.iterable
|
|
|
|
|
|
|
|
def next(self):
|
|
|
|
return next(self.iterable)
|
|
|
|
|
|
|
|
def __len__(self):
|
|
|
|
return self.length
|