2013-09-20 04:05:51 +08:00
|
|
|
# Copyright 2012 OpenStack Foundation
|
2012-08-01 16:04:37 +00: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-12-16 15:28:05 +01:00
|
|
|
import sys
|
2012-08-01 16:04:37 +00:00
|
|
|
|
2015-06-11 09:21:53 +12:00
|
|
|
import mock
|
2015-07-27 15:21:32 +03:00
|
|
|
from oslo_utils import encodeutils
|
2014-01-10 17:25:25 +01:00
|
|
|
import six
|
2015-02-25 12:17:24 +00:00
|
|
|
# NOTE(jokke): simplified transition to py3, behaves like py2 xrange
|
|
|
|
from six.moves import range
|
2013-07-08 21:18:16 +02:00
|
|
|
import testtools
|
|
|
|
|
2012-08-01 16:04:37 +00:00
|
|
|
from glanceclient.common import utils
|
|
|
|
|
|
|
|
|
2012-12-24 22:37:01 -06:00
|
|
|
class TestUtils(testtools.TestCase):
|
2012-08-01 16:04:37 +00:00
|
|
|
|
2012-11-07 19:39:43 +01:00
|
|
|
def test_make_size_human_readable(self):
|
2012-11-19 10:35:04 -08:00
|
|
|
self.assertEqual("106B", utils.make_size_human_readable(106))
|
|
|
|
self.assertEqual("1000kB", utils.make_size_human_readable(1024000))
|
|
|
|
self.assertEqual("1MB", utils.make_size_human_readable(1048576))
|
|
|
|
self.assertEqual("1.4GB", utils.make_size_human_readable(1476395008))
|
|
|
|
self.assertEqual("9.3MB", utils.make_size_human_readable(9761280))
|
2015-09-14 17:27:43 +00:00
|
|
|
self.assertEqual("0B", utils.make_size_human_readable(None))
|
2013-01-30 15:18:44 +01:00
|
|
|
|
2013-07-08 21:18:16 +02:00
|
|
|
def test_get_new_file_size(self):
|
|
|
|
size = 98304
|
2014-01-10 17:25:25 +01:00
|
|
|
file_obj = six.StringIO('X' * size)
|
2013-07-08 21:18:16 +02:00
|
|
|
try:
|
2014-02-24 18:47:59 +09:00
|
|
|
self.assertEqual(size, utils.get_file_size(file_obj))
|
2013-07-08 21:18:16 +02:00
|
|
|
# Check that get_file_size didn't change original file position.
|
2014-02-24 18:47:59 +09:00
|
|
|
self.assertEqual(0, file_obj.tell())
|
2013-07-08 21:18:16 +02:00
|
|
|
finally:
|
|
|
|
file_obj.close()
|
|
|
|
|
|
|
|
def test_get_consumed_file_size(self):
|
|
|
|
size, consumed = 98304, 304
|
2014-01-10 17:25:25 +01:00
|
|
|
file_obj = six.StringIO('X' * size)
|
2013-07-08 21:18:16 +02:00
|
|
|
file_obj.seek(consumed)
|
|
|
|
try:
|
2014-02-24 18:47:59 +09:00
|
|
|
self.assertEqual(size, utils.get_file_size(file_obj))
|
2013-07-08 21:18:16 +02:00
|
|
|
# Check that get_file_size didn't change original file position.
|
2014-02-24 18:47:59 +09:00
|
|
|
self.assertEqual(consumed, file_obj.tell())
|
2013-07-08 21:18:16 +02:00
|
|
|
finally:
|
|
|
|
file_obj.close()
|
|
|
|
|
2013-02-22 16:11:12 +01:00
|
|
|
def test_prettytable(self):
|
2015-07-20 17:29:49 +03:00
|
|
|
class Struct(object):
|
2013-02-22 16:11:12 +01:00
|
|
|
def __init__(self, **entries):
|
|
|
|
self.__dict__.update(entries)
|
|
|
|
|
|
|
|
# test that the prettytable output is wellformatted (left-aligned)
|
|
|
|
columns = ['ID', 'Name']
|
|
|
|
val = ['Name1', 'another', 'veeeery long']
|
|
|
|
images = [Struct(**{'id': i ** 16, 'name': val[i]})
|
|
|
|
for i in range(len(val))]
|
|
|
|
|
|
|
|
saved_stdout = sys.stdout
|
|
|
|
try:
|
2014-01-10 17:25:25 +01:00
|
|
|
sys.stdout = output_list = six.StringIO()
|
2013-02-22 16:11:12 +01:00
|
|
|
utils.print_list(images, columns)
|
|
|
|
|
2014-01-10 17:25:25 +01:00
|
|
|
sys.stdout = output_dict = six.StringIO()
|
2013-11-15 16:02:46 +08:00
|
|
|
utils.print_dict({'K': 'k', 'Key': 'veeeeeeeeeeeeeeeeeeeeeeee'
|
|
|
|
'eeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeee'
|
|
|
|
'eeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeee'
|
|
|
|
'eeeeeeeeeeeery long value'},
|
|
|
|
max_column_width=60)
|
2013-02-22 16:11:12 +01:00
|
|
|
|
|
|
|
finally:
|
|
|
|
sys.stdout = saved_stdout
|
|
|
|
|
2014-02-24 18:47:59 +09:00
|
|
|
self.assertEqual('''\
|
2013-02-22 16:11:12 +01:00
|
|
|
+-------+--------------+
|
|
|
|
| ID | Name |
|
|
|
|
+-------+--------------+
|
|
|
|
| | Name1 |
|
|
|
|
| 1 | another |
|
|
|
|
| 65536 | veeeery long |
|
|
|
|
+-------+--------------+
|
2014-02-24 18:47:59 +09:00
|
|
|
''',
|
|
|
|
output_list.getvalue())
|
2013-02-22 16:11:12 +01:00
|
|
|
|
2014-02-24 18:47:59 +09:00
|
|
|
self.assertEqual('''\
|
2013-11-15 16:02:46 +08:00
|
|
|
+----------+--------------------------------------------------------------+
|
|
|
|
| Property | Value |
|
|
|
|
+----------+--------------------------------------------------------------+
|
|
|
|
| K | k |
|
|
|
|
| Key | veeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeee |
|
|
|
|
| | eeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeee |
|
|
|
|
| | ery long value |
|
|
|
|
+----------+--------------------------------------------------------------+
|
2014-02-24 18:47:59 +09:00
|
|
|
''',
|
|
|
|
output_dict.getvalue())
|
2013-07-11 15:28:49 +02:00
|
|
|
|
2014-12-10 10:47:40 +01:00
|
|
|
def test_schema_args_with_list_types(self):
|
|
|
|
# NOTE(flaper87): Regression for bug
|
|
|
|
# https://bugs.launchpad.net/python-glanceclient/+bug/1401032
|
|
|
|
|
|
|
|
def schema_getter(_type='string', enum=False):
|
|
|
|
prop = {
|
|
|
|
'type': ['null', _type],
|
|
|
|
'description': 'Test schema (READ-ONLY)',
|
|
|
|
}
|
|
|
|
|
|
|
|
if enum:
|
|
|
|
prop['enum'] = [None, 'opt-1', 'opt-2']
|
|
|
|
|
|
|
|
def actual_getter():
|
|
|
|
return {
|
|
|
|
'additionalProperties': False,
|
|
|
|
'required': ['name'],
|
|
|
|
'name': 'test_schema',
|
|
|
|
'properties': {
|
|
|
|
'test': prop,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return actual_getter
|
|
|
|
|
|
|
|
def dummy_func():
|
|
|
|
pass
|
|
|
|
|
|
|
|
decorated = utils.schema_args(schema_getter())(dummy_func)
|
|
|
|
arg, opts = decorated.__dict__['arguments'][0]
|
|
|
|
self.assertIn('--test', arg)
|
2015-07-27 15:21:32 +03:00
|
|
|
self.assertEqual(encodeutils.safe_decode, opts['type'])
|
2014-12-10 10:47:40 +01:00
|
|
|
|
|
|
|
decorated = utils.schema_args(schema_getter('integer'))(dummy_func)
|
|
|
|
arg, opts = decorated.__dict__['arguments'][0]
|
|
|
|
self.assertIn('--test', arg)
|
|
|
|
self.assertEqual(int, opts['type'])
|
|
|
|
|
|
|
|
decorated = utils.schema_args(schema_getter(enum=True))(dummy_func)
|
|
|
|
arg, opts = decorated.__dict__['arguments'][0]
|
|
|
|
self.assertIn('--test', arg)
|
2015-07-27 15:21:32 +03:00
|
|
|
self.assertEqual(encodeutils.safe_decode, opts['type'])
|
2014-12-10 10:47:40 +01:00
|
|
|
self.assertIn('None, opt-1, opt-2', opts['help'])
|
2015-06-11 09:21:53 +12:00
|
|
|
|
|
|
|
def test_iterable_closes(self):
|
|
|
|
# Regression test for bug 1461678.
|
|
|
|
def _iterate(i):
|
|
|
|
for chunk in i:
|
|
|
|
raise(IOError)
|
|
|
|
|
|
|
|
data = six.moves.StringIO('somestring')
|
|
|
|
data.close = mock.Mock()
|
|
|
|
i = utils.IterableWithLength(data, 10)
|
|
|
|
self.assertRaises(IOError, _iterate, i)
|
|
|
|
data.close.assert_called_with()
|
2015-09-02 14:50:47 +03:00
|
|
|
|
|
|
|
def test_safe_header(self):
|
|
|
|
self.assertEqual(('somekey', 'somevalue'),
|
|
|
|
utils.safe_header('somekey', 'somevalue'))
|
|
|
|
self.assertEqual(('somekey', None),
|
|
|
|
utils.safe_header('somekey', None))
|
|
|
|
|
|
|
|
for sensitive_header in utils.SENSITIVE_HEADERS:
|
2015-04-24 13:29:12 +02:00
|
|
|
(name, value) = utils.safe_header(
|
|
|
|
sensitive_header,
|
|
|
|
encodeutils.safe_encode('somestring'))
|
2015-09-02 14:50:47 +03:00
|
|
|
self.assertEqual(sensitive_header, name)
|
|
|
|
self.assertTrue(value.startswith("{SHA1}"))
|
|
|
|
|
|
|
|
(name, value) = utils.safe_header(sensitive_header, None)
|
|
|
|
self.assertEqual(sensitive_header, name)
|
|
|
|
self.assertIsNone(value)
|