Add a 'meta' passthrough parameter for glance images

create_image tries to do data type conversion for you so that what you
mean is correct 90% of the time. However, inferring intent is hard on
people who do know what they want.

New parameter 'meta' is a vehicle for non-converted key/value pairs.

Change-Id: I99c1a104f6eb8fe72dd4ebab5b3aac8231068eb7
This commit is contained in:
Monty Taylor 2016-08-02 10:15:13 -05:00
parent 3ef3864785
commit a98be6a666
No known key found for this signature in database
GPG Key ID: 7BAE94BC7141A594
3 changed files with 177 additions and 30 deletions

View File

@ -0,0 +1,7 @@
---
features:
- Added a parameter to create_image 'meta' which allows
for providing parameters to the API that will not have
any type conversions performed. For the simple case,
the existing kwargs approach to image metadata is still
the best bet.

View File

@ -2547,7 +2547,7 @@ class OpenStackCloud(object):
disk_format=None, container_format=None, disk_format=None, container_format=None,
disable_vendor_agent=True, disable_vendor_agent=True,
wait=False, timeout=3600, wait=False, timeout=3600,
allow_duplicates=False, **kwargs): allow_duplicates=False, meta=None, **kwargs):
"""Upload an image to Glance. """Upload an image to Glance.
:param str name: Name of the image to create. If it is a pathname :param str name: Name of the image to create. If it is a pathname
@ -2581,9 +2581,19 @@ class OpenStackCloud(object):
:param timeout: Seconds to wait for image creation. None is forever. :param timeout: Seconds to wait for image creation. None is forever.
:param allow_duplicates: If true, skips checks that enforce unique :param allow_duplicates: If true, skips checks that enforce unique
image name. (optional, defaults to False) image name. (optional, defaults to False)
:param meta: A dict of key/value pairs to use for metadata that
bypasses automatic type conversion.
Additional kwargs will be passed to the image creation as additional Additional kwargs will be passed to the image creation as additional
metadata for the image. metadata for the image and will have all values converted to string
except for min_disk, min_ram, size and virtual_size which will be
converted to int.
If you are sure you have all of your data types correct or have an
advanced need to be explicit, use meta. If you are just a normal
consumer, using kwargs is likely the right choice.
If a value is in meta and kwargs, meta wins.
:returns: A ``munch.Munch`` of the Image object :returns: A ``munch.Munch`` of the Image object
@ -2593,6 +2603,9 @@ class OpenStackCloud(object):
if not disk_format: if not disk_format:
disk_format = self.cloud_config.config['image_format'] disk_format = self.cloud_config.config['image_format']
if not meta:
meta = {}
# If there is no filename, see if name is actually the filename # If there is no filename, see if name is actually the filename
if not filename: if not filename:
name, filename = self._get_name_and_filename(name) name, filename = self._get_name_and_filename(name)
@ -2640,15 +2653,21 @@ class OpenStackCloud(object):
return self._upload_image_task( return self._upload_image_task(
name, filename, container, name, filename, container,
current_image=current_image, current_image=current_image,
wait=wait, timeout=timeout, **kwargs) wait=wait, timeout=timeout,
meta=meta, **kwargs)
else: else:
# If a user used the v1 calling format, they will have
# passed a dict called properties along
properties = kwargs.pop('properties', {})
kwargs.update(properties)
image_kwargs = dict(properties=kwargs) image_kwargs = dict(properties=kwargs)
if disk_format: if disk_format:
image_kwargs['disk_format'] = disk_format image_kwargs['disk_format'] = disk_format
if container_format: if container_format:
image_kwargs['container_format'] = container_format image_kwargs['container_format'] = container_format
return self._upload_image_put(name, filename, **image_kwargs) return self._upload_image_put(
name, filename, meta=meta, **image_kwargs)
except OpenStackCloudException: except OpenStackCloudException:
self.log.debug("Image creation failed", exc_info=True) self.log.debug("Image creation failed", exc_info=True)
raise raise
@ -2656,15 +2675,25 @@ class OpenStackCloud(object):
raise OpenStackCloudException( raise OpenStackCloudException(
"Image creation failed: {message}".format(message=str(e))) "Image creation failed: {message}".format(message=str(e)))
def _upload_image_put_v2(self, name, image_data, **image_kwargs): def _make_v2_image_params(self, meta, properties):
if 'properties' in image_kwargs: ret = {}
img_props = image_kwargs.pop('properties') for k, v in iter(properties.items()):
for k, v in iter(img_props.items()): if k in ('min_disk', 'min_ram', 'size', 'virtual_size'):
image_kwargs[k] = str(v) ret[k] = int(v)
# some MUST be integer else:
for k in ('min_disk', 'min_ram'): if v is None:
if k in image_kwargs: ret[k] = None
image_kwargs[k] = int(image_kwargs[k]) else:
ret[k] = str(v)
ret.update(meta)
return ret
def _upload_image_put_v2(self, name, image_data, meta, **image_kwargs):
properties = image_kwargs.pop('properties', {})
image_kwargs.update(self._make_v2_image_params(meta, properties))
image = self.manager.submitTask(_tasks.ImageCreate( image = self.manager.submitTask(_tasks.ImageCreate(
name=name, **image_kwargs)) name=name, **image_kwargs))
try: try:
@ -2678,7 +2707,10 @@ class OpenStackCloud(object):
return image return image
def _upload_image_put_v1(self, name, image_data, **image_kwargs): def _upload_image_put_v1(
self, name, image_data, meta, **image_kwargs):
image_kwargs['properties'].update(meta)
image = self.manager.submitTask(_tasks.ImageCreate( image = self.manager.submitTask(_tasks.ImageCreate(
name=name, **image_kwargs)) name=name, **image_kwargs))
try: try:
@ -2691,19 +2723,25 @@ class OpenStackCloud(object):
raise raise
return image return image
def _upload_image_put(self, name, filename, **image_kwargs): def _upload_image_put(
self, name, filename, meta, **image_kwargs):
image_data = open(filename, 'rb') image_data = open(filename, 'rb')
# Because reasons and crying bunnies # Because reasons and crying bunnies
if self.cloud_config.get_api_version('image') == '2': if self.cloud_config.get_api_version('image') == '2':
image = self._upload_image_put_v2(name, image_data, **image_kwargs) image = self._upload_image_put_v2(
name, image_data, meta, **image_kwargs)
else: else:
image = self._upload_image_put_v1(name, image_data, **image_kwargs) image = self._upload_image_put_v1(
name, image_data, meta, **image_kwargs)
self._cache.invalidate() self._cache.invalidate()
return self.get_image(image.id) return self.get_image(image.id)
def _upload_image_task( def _upload_image_task(
self, name, filename, container, current_image, self, name, filename, container, current_image,
wait, timeout, **image_properties): wait, timeout, meta, **image_kwargs):
parameters = image_kwargs.pop('parameters', {})
image_kwargs.update(parameters)
# get new client sessions # get new client sessions
with self._swift_client_lock: with self._swift_client_lock:
@ -2713,8 +2751,8 @@ class OpenStackCloud(object):
self.create_object( self.create_object(
container, name, filename, container, name, filename,
md5=image_properties.get('md5', None), md5=image_kwargs.get('md5', None),
sha256=image_properties.get('sha256', None)) sha256=image_kwargs.get('sha256', None))
if not current_image: if not current_image:
current_image = self.get_image(name) current_image = self.get_image(name)
# TODO(mordred): Can we do something similar to what nodepool does # TODO(mordred): Can we do something similar to what nodepool does
@ -2752,8 +2790,7 @@ class OpenStackCloud(object):
if image is None: if image is None:
continue continue
self.update_image_properties( self.update_image_properties(
image=image, image=image, meta=meta, **image_kwargs)
**image_properties)
return self.get_image(status.result['image_id']) return self.get_image(status.result['image_id'])
if status.status == 'failure': if status.status == 'failure':
if status.message == IMAGE_ERROR_396: if status.message == IMAGE_ERROR_396:
@ -2769,9 +2806,11 @@ class OpenStackCloud(object):
return glance_task return glance_task
def update_image_properties( def update_image_properties(
self, image=None, name_or_id=None, **properties): self, image=None, name_or_id=None, meta=None, **properties):
if image is None: if image is None:
image = self.get_image(name_or_id) image = self.get_image(name_or_id)
if not meta:
meta = {}
img_props = {} img_props = {}
for k, v in iter(properties.items()): for k, v in iter(properties.items()):
@ -2782,15 +2821,15 @@ class OpenStackCloud(object):
# This makes me want to die inside # This makes me want to die inside
if self.cloud_config.get_api_version('image') == '2': if self.cloud_config.get_api_version('image') == '2':
return self._update_image_properties_v2(image, img_props) return self._update_image_properties_v2(image, meta, img_props)
else: else:
return self._update_image_properties_v1(image, img_props) return self._update_image_properties_v1(image, meta, img_props)
def _update_image_properties_v2(self, image, properties): def _update_image_properties_v2(self, image, meta, properties):
img_props = {} img_props = {}
for k, v in iter(properties.items()): for k, v in iter(self._make_v2_image_params(meta, properties).items()):
if image.get(k, None) != v: if image.get(k, None) != v:
img_props[k] = str(v) img_props[k] = v
if not img_props: if not img_props:
return False return False
self.manager.submitTask(_tasks.ImageUpdate( self.manager.submitTask(_tasks.ImageUpdate(
@ -2798,7 +2837,8 @@ class OpenStackCloud(object):
self.list_images.invalidate(self) self.list_images.invalidate(self)
return True return True
def _update_image_properties_v1(self, image, properties): def _update_image_properties_v1(self, image, meta, properties):
properties.update(meta)
img_props = {} img_props = {}
for k, v in iter(properties.items()): for k, v in iter(properties.items()):
if image.properties.get(k, None) != v: if image.properties.get(k, None) != v:

View File

@ -386,7 +386,7 @@ class TestMemoryCache(base.TestCase):
fake_image = fakes.FakeImage('42', '42 name', 'success') fake_image = fakes.FakeImage('42', '42 name', 'success')
glance_mock.images.create.return_value = fake_image glance_mock.images.create.return_value = fake_image
glance_mock.images.list.return_value = [fake_image] glance_mock.images.list.return_value = [fake_image]
self._call_create_image('42 name', min_disk=0, min_ram=0) self._call_create_image('42 name', min_disk='0', min_ram=0)
args = {'name': '42 name', args = {'name': '42 name',
'container_format': 'bare', 'disk_format': 'qcow2', 'container_format': 'bare', 'disk_format': 'qcow2',
'owner_specified.shade.md5': mock.ANY, 'owner_specified.shade.md5': mock.ANY,
@ -400,6 +400,106 @@ class TestMemoryCache(base.TestCase):
fake_image_dict = meta.obj_to_dict(fake_image) fake_image_dict = meta.obj_to_dict(fake_image)
self.assertEqual([fake_image_dict], self.cloud.list_images()) self.assertEqual([fake_image_dict], self.cloud.list_images())
@mock.patch.object(occ.cloud_config.CloudConfig, 'get_api_version')
@mock.patch.object(shade.OpenStackCloud, 'glance_client')
def test_create_image_put_bad_int(self, glance_mock, mock_api_version):
mock_api_version.return_value = '2'
self.cloud.image_api_use_tasks = False
glance_mock.images.list.return_value = []
self.assertEqual([], self.cloud.list_images())
fake_image = fakes.FakeImage('42', '42 name', 'success')
glance_mock.images.create.return_value = fake_image
glance_mock.images.list.return_value = [fake_image]
self.assertRaises(
exc.OpenStackCloudException,
self._call_create_image, '42 name', min_disk='fish', min_ram=0)
@mock.patch.object(occ.cloud_config.CloudConfig, 'get_api_version')
@mock.patch.object(shade.OpenStackCloud, 'glance_client')
def test_create_image_put_user_int(self, glance_mock, mock_api_version):
mock_api_version.return_value = '2'
self.cloud.image_api_use_tasks = False
glance_mock.images.list.return_value = []
self.assertEqual([], self.cloud.list_images())
fake_image = fakes.FakeImage('42', '42 name', 'success')
glance_mock.images.create.return_value = fake_image
glance_mock.images.list.return_value = [fake_image]
self._call_create_image(
'42 name', min_disk='0', min_ram=0, int_v=12345)
args = {'name': '42 name',
'container_format': 'bare', 'disk_format': u'qcow2',
'owner_specified.shade.md5': mock.ANY,
'owner_specified.shade.sha256': mock.ANY,
'owner_specified.shade.object': 'images/42 name',
'int_v': '12345',
'visibility': 'private',
'min_disk': 0, 'min_ram': 0}
glance_mock.images.create.assert_called_with(**args)
glance_mock.images.upload.assert_called_with(
image_data=mock.ANY, image_id=fake_image.id)
fake_image_dict = meta.obj_to_dict(fake_image)
self.assertEqual([fake_image_dict], self.cloud.list_images())
@mock.patch.object(occ.cloud_config.CloudConfig, 'get_api_version')
@mock.patch.object(shade.OpenStackCloud, 'glance_client')
def test_create_image_put_meta_int(self, glance_mock, mock_api_version):
mock_api_version.return_value = '2'
self.cloud.image_api_use_tasks = False
glance_mock.images.list.return_value = []
self.assertEqual([], self.cloud.list_images())
fake_image = fakes.FakeImage('42', '42 name', 'success')
glance_mock.images.create.return_value = fake_image
glance_mock.images.list.return_value = [fake_image]
self._call_create_image(
'42 name', min_disk='0', min_ram=0, meta={'int_v': 12345})
args = {'name': '42 name',
'container_format': 'bare', 'disk_format': u'qcow2',
'owner_specified.shade.md5': mock.ANY,
'owner_specified.shade.sha256': mock.ANY,
'owner_specified.shade.object': 'images/42 name',
'int_v': 12345,
'visibility': 'private',
'min_disk': 0, 'min_ram': 0}
glance_mock.images.create.assert_called_with(**args)
glance_mock.images.upload.assert_called_with(
image_data=mock.ANY, image_id=fake_image.id)
fake_image_dict = meta.obj_to_dict(fake_image)
self.assertEqual([fake_image_dict], self.cloud.list_images())
@mock.patch.object(occ.cloud_config.CloudConfig, 'get_api_version')
@mock.patch.object(shade.OpenStackCloud, 'glance_client')
def test_create_image_put_user_prop(self, glance_mock, mock_api_version):
mock_api_version.return_value = '2'
self.cloud.image_api_use_tasks = False
glance_mock.images.list.return_value = []
self.assertEqual([], self.cloud.list_images())
fake_image = fakes.FakeImage('42', '42 name', 'success')
glance_mock.images.create.return_value = fake_image
glance_mock.images.list.return_value = [fake_image]
self._call_create_image(
'42 name', min_disk='0', min_ram=0, properties={'int_v': 12345})
args = {'name': '42 name',
'container_format': 'bare', 'disk_format': u'qcow2',
'owner_specified.shade.md5': mock.ANY,
'owner_specified.shade.sha256': mock.ANY,
'owner_specified.shade.object': 'images/42 name',
'int_v': '12345',
'visibility': 'private',
'min_disk': 0, 'min_ram': 0}
glance_mock.images.create.assert_called_with(**args)
glance_mock.images.upload.assert_called_with(
image_data=mock.ANY, image_id=fake_image.id)
fake_image_dict = meta.obj_to_dict(fake_image)
self.assertEqual([fake_image_dict], self.cloud.list_images())
@mock.patch.object(occ.cloud_config.CloudConfig, 'get_api_version') @mock.patch.object(occ.cloud_config.CloudConfig, 'get_api_version')
@mock.patch.object(shade.OpenStackCloud, '_get_file_hashes') @mock.patch.object(shade.OpenStackCloud, '_get_file_hashes')
@mock.patch.object(shade.OpenStackCloud, 'glance_client') @mock.patch.object(shade.OpenStackCloud, 'glance_client')