image: Split image creation depending on service

The 'image create' operation is actually one of two operations: it can
be either an image service (glance) operation if the '--volume' argument
is *not* passed or a block storage (cinder) operation if it is. Make
this clearer and add a log warning users about options that are
supported by the former but not the latter.

Change-Id: Id153c951a7d18403568bf67e13d5e0a4827428d4
Signed-off-by: Stephen Finucane <sfinucan@redhat.com>
This commit is contained in:
Stephen Finucane 2022-06-30 11:37:13 +01:00
parent 4776e0a5ae
commit 14b93fec76

View File

@ -408,18 +408,10 @@ class CreateImage(command.ShowOne):
) )
return parser return parser
def take_action(self, parsed_args): def _take_action_image(self, parsed_args):
identity_client = self.app.client_manager.identity identity_client = self.app.client_manager.identity
image_client = self.app.client_manager.image image_client = self.app.client_manager.image
for deadopt in self.deadopts:
if getattr(parsed_args, deadopt.replace('-', '_'), None):
msg = _(
"ERROR: --%s was given, which is an Image v1 option "
"that is no longer supported in Image v2"
)
raise exceptions.CommandError(msg % deadopt)
# Build an attribute dict from the parsed args, only include # Build an attribute dict from the parsed args, only include
# attributes that were actually set on the command line # attributes that were actually set on the command line
kwargs = {'allow_duplicates': True} kwargs = {'allow_duplicates': True}
@ -472,7 +464,6 @@ class CreateImage(command.ShowOne):
# image is created. Get the file name (if it is file, and not stdin) # image is created. Get the file name (if it is file, and not stdin)
# for easier further handling. # for easier further handling.
fp, fname = get_data_file(parsed_args) fp, fname = get_data_file(parsed_args)
info = {}
if fp is not None and parsed_args.volume: if fp is not None and parsed_args.volume:
msg = _( msg = _(
@ -552,49 +543,92 @@ class CreateImage(command.ShowOne):
if signer.padding_method: if signer.padding_method:
kwargs['img_signature_key_type'] = signer.padding_method kwargs['img_signature_key_type'] = signer.padding_method
# If a volume is specified. image = image_client.create_image(**kwargs)
if parsed_args.volume: return _format_image(image)
volume_client = self.app.client_manager.volume
source_volume = utils.find_resource( def _take_action_volume(self, parsed_args):
volume_client.volumes, volume_client = self.app.client_manager.volume
parsed_args.volume,
) unsupported_opts = {
mv_kwargs = {} # 'name', # 'name' is a positional argument and will always exist
if volume_client.api_version >= api_versions.APIVersion('3.1'): 'id',
mv_kwargs.update( 'min_disk',
visibility=kwargs.get('visibility', 'private'), 'min_ram',
protected=bool(parsed_args.is_protected), 'file',
'force',
'progress',
'sign_key_path',
'sign_cert_id',
'properties',
'tags',
'project',
'use_import',
}
for unsupported_opt in unsupported_opts:
if getattr(parsed_args, unsupported_opt, None):
opt_name = unsupported_opt.replace('-', '_')
if unsupported_opt == 'use_import':
opt_name = 'import'
msg = _(
"'--%s' was given, which is not supported when "
"creating an image from a volume. "
"This will be an error in a future version."
) )
else: # TODO(stephenfin): These should be an error in a future
if ( # version
parsed_args.visibility or LOG.warning(msg % opt_name)
parsed_args.is_protected is not None
):
msg = _(
'--os-volume-api-version 3.1 or greater is required '
'to support the --public, --private, --community, '
'--shared or --protected option.'
)
raise exceptions.CommandError(msg)
response, body = volume_client.volumes.upload_to_image( source_volume = utils.find_resource(
source_volume.id, volume_client.volumes,
parsed_args.force, parsed_args.volume,
parsed_args.name, )
parsed_args.container_format, kwargs = {}
parsed_args.disk_format, if volume_client.api_version < api_versions.APIVersion('3.1'):
**mv_kwargs if (
) parsed_args.visibility or
info = body['os-volume_upload_image'] parsed_args.is_protected is not None
try: ):
info['volume_type'] = info['volume_type']['name'] msg = _(
except TypeError: '--os-volume-api-version 3.1 or greater is required '
info['volume_type'] = None 'to support the --public, --private, --community, '
'--shared or --protected option.'
)
raise exceptions.CommandError(msg)
else: else:
image = image_client.create_image(**kwargs) kwargs.update(
visibility=parsed_args.visibility or 'private',
protected=parsed_args.is_protected or False,
)
if not info: response, body = volume_client.volumes.upload_to_image(
info = _format_image(image) source_volume.id,
parsed_args.force,
parsed_args.name,
parsed_args.container_format,
parsed_args.disk_format,
**kwargs
)
info = body['os-volume_upload_image']
try:
info['volume_type'] = info['volume_type']['name']
except TypeError:
info['volume_type'] = None
return info
def take_action(self, parsed_args):
for deadopt in self.deadopts:
if getattr(parsed_args, deadopt.replace('-', '_'), None):
msg = _(
"ERROR: --%s was given, which is an Image v1 option "
"that is no longer supported in Image v2"
)
raise exceptions.CommandError(msg % deadopt)
if parsed_args.volume:
info = self._take_action_volume(parsed_args)
else:
info = self._take_action_image(parsed_args)
return zip(*sorted(info.items())) return zip(*sorted(info.items()))