Fix for referencing cloud image by ID

For pre-existing cloud images (not managed by nodepool), referencing
them by ID was failing since they could not be found with this data,
only by name.

Current code expects the shade get_image() call to accept a dict with
an 'id' key, which will return that same dict without any provider API
calls. This dict can then be used in createServer() to bypass looking
up the image to get the image ID. However, shade does not accept a dict
for this purpose, but an object with an 'id' attribute. This is
possibly a bug in shade to not accept a dict. But since nodepool knows
whether or not it has an ID (image-id) vs. an image name (image-name),
it can bypass shade altogether when image-id is used in the config.

Note: There is currently no image ID validation done before image
creation when an image-id value is supplied. Not even shade validated
the image ID with a passed in object. Server creation will fail with
an easily identifiable message about this, though.

Change-Id: I732026d1a305c71af53917285f4ebb2beaf3341d
Story: 2002013
Task: 19653
This commit is contained in:
David Shrewsbury 2018-07-02 14:46:39 -04:00
parent e66e5a56b6
commit d39cc6d7ce
4 changed files with 23 additions and 10 deletions

View File

@ -69,14 +69,6 @@ class ProviderCloudImage(ConfigValue):
def __repr__(self): def __repr__(self):
return "<ProviderCloudImage %s>" % self.name return "<ProviderCloudImage %s>" % self.name
@property
def external(self):
'''External identifier to pass to the cloud.'''
if self.image_id:
return dict(id=self.image_id)
else:
return self.image_name or self.name
@property @property
def external_name(self): def external_name(self):
'''Human readable version of external.''' '''Human readable version of external.'''

View File

@ -76,6 +76,8 @@ class OpenStackNodeLauncher(NodeLauncher):
) )
config_drive = diskimage.config_drive config_drive = diskimage.config_drive
# Using a dict with the ID bypasses an image search during
# server creation.
image_external = dict(id=cloud_image.external_id) image_external = dict(id=cloud_image.external_id)
image_id = "{path}/{upload_id}".format( image_id = "{path}/{upload_id}".format(
path=self.handler.zk._imageUploadPath( path=self.handler.zk._imageUploadPath(
@ -92,7 +94,13 @@ class OpenStackNodeLauncher(NodeLauncher):
# launch using unmanaged cloud image # launch using unmanaged cloud image
config_drive = self.label.cloud_image.config_drive config_drive = self.label.cloud_image.config_drive
image_external = self.label.cloud_image.external if self.label.cloud_image.image_id:
# Using a dict with the ID bypasses an image search during
# server creation.
image_external = dict(id=self.label.cloud_image.image_id)
else:
image_external = self.label.cloud_image.external_name
image_id = self.label.cloud_image.name image_id = self.label.cloud_image.name
image_name = self.label.cloud_image.name image_name = self.label.cloud_image.name
username = self.label.cloud_image.username username = self.label.cloud_image.username

View File

@ -388,7 +388,14 @@ class OpenStackProvider(Provider):
def labelReady(self, label): def labelReady(self, label):
if not label.cloud_image: if not label.cloud_image:
return False return False
image = self.getImage(label.cloud_image.external)
# If an image ID was supplied, we'll assume it is ready since
# we don't currently have a way of validating that (except during
# server creation).
if label.cloud_image.image_id:
return True
image = self.getImage(label.cloud_image.external_name)
if not image: if not image:
self.log.warning( self.log.warning(
"Provider %s is configured to use %s as the" "Provider %s is configured to use %s as the"

View File

@ -0,0 +1,6 @@
---
fixes:
- |
For pre-existing cloud images (not managed by nodepool), referencing
them by ID was failing since they could not be found with this data,
only by name.