image: Unset properties rather than setting to None

Currently, we attempt to unset an image property by setting it to None.
This doesn't work for known properties and is rightly rejected by the
Glance API with the following error:

  BadRequestException: 400: Client Error for url:
  http://172.20.4.87/image/v2/images/368c5751-2b0b-4a38-a255-fd146fe52d31,
  Bad Request

The solution is to actually unset the field by deleting it.

Change-Id: Ie156bedbe0f9244f82c81401679706f484caf9aa
Signed-off-by: Stephen Finucane <sfinucan@redhat.com>
Story: #2008463
Task: #41493
This commit is contained in:
Stephen Finucane 2020-12-22 16:46:33 +00:00
parent f5b185c357
commit 29a7c9afce
3 changed files with 16 additions and 17 deletions

View File

@ -1166,7 +1166,7 @@ class UnsetImage(command.Command):
if parsed_args.properties: if parsed_args.properties:
for k in parsed_args.properties: for k in parsed_args.properties:
if k in image: if k in image:
kwargs[k] = None delattr(image, k)
elif k in image.properties: elif k in image.properties:
# Since image is an "evil" object from SDK POV we need to # Since image is an "evil" object from SDK POV we need to
# pass modified properties object, so that SDK can figure # pass modified properties object, so that SDK can figure

View File

@ -119,6 +119,7 @@ class ImageTests(base.BaseImageTests):
'image set ' + 'image set ' +
'--property a=b ' + '--property a=b ' +
'--property c=d ' + '--property c=d ' +
'--property hw_rng_model=virtio ' +
'--public ' + '--public ' +
self.name self.name
) )
@ -133,6 +134,7 @@ class ImageTests(base.BaseImageTests):
'image unset ' + 'image unset ' +
'--property a ' + '--property a ' +
'--property c ' + '--property c ' +
'--property hw_rng_model ' +
self.name self.name
) )
json_output = json.loads(self.openstack( json_output = json.loads(self.openstack(

View File

@ -1501,15 +1501,16 @@ class TestImageShow(TestImage):
class TestImageUnset(TestImage): class TestImageUnset(TestImage):
attrs = {}
attrs['tags'] = ['test']
attrs['prop'] = 'test'
attrs['prop2'] = 'fake'
image = image_fakes.FakeImage.create_one_image(attrs)
def setUp(self): def setUp(self):
super(TestImageUnset, self).setUp() super(TestImageUnset, self).setUp()
attrs = {}
attrs['tags'] = ['test']
attrs['hw_rng_model'] = 'virtio'
attrs['prop'] = 'test'
attrs['prop2'] = 'fake'
self.image = image_fakes.FakeImage.create_one_image(attrs)
self.client.find_image.return_value = self.image self.client.find_image.return_value = self.image
self.client.remove_tag.return_value = self.image self.client.remove_tag.return_value = self.image
self.client.update_image.return_value = self.image self.client.update_image.return_value = self.image
@ -1552,22 +1553,20 @@ class TestImageUnset(TestImage):
def test_image_unset_property_option(self): def test_image_unset_property_option(self):
arglist = [ arglist = [
'--property', 'hw_rng_model',
'--property', 'prop', '--property', 'prop',
self.image.id, self.image.id,
] ]
verifylist = [ verifylist = [
('properties', ['prop']), ('properties', ['hw_rng_model', 'prop']),
('image', self.image.id) ('image', self.image.id)
] ]
parsed_args = self.check_parser(self.cmd, arglist, verifylist) parsed_args = self.check_parser(self.cmd, arglist, verifylist)
result = self.cmd.take_action(parsed_args) result = self.cmd.take_action(parsed_args)
kwargs = {}
self.client.update_image.assert_called_with( self.client.update_image.assert_called_with(
self.image, self.image, properties={'prop2': 'fake'})
properties={'prop2': 'fake'},
**kwargs)
self.assertIsNone(result) self.assertIsNone(result)
@ -1575,23 +1574,21 @@ class TestImageUnset(TestImage):
arglist = [ arglist = [
'--tag', 'test', '--tag', 'test',
'--property', 'hw_rng_model',
'--property', 'prop', '--property', 'prop',
self.image.id, self.image.id,
] ]
verifylist = [ verifylist = [
('tags', ['test']), ('tags', ['test']),
('properties', ['prop']), ('properties', ['hw_rng_model', 'prop']),
('image', self.image.id) ('image', self.image.id)
] ]
parsed_args = self.check_parser(self.cmd, arglist, verifylist) parsed_args = self.check_parser(self.cmd, arglist, verifylist)
result = self.cmd.take_action(parsed_args) result = self.cmd.take_action(parsed_args)
kwargs = {}
self.client.update_image.assert_called_with( self.client.update_image.assert_called_with(
self.image, self.image, properties={'prop2': 'fake'})
properties={'prop2': 'fake'},
**kwargs)
self.client.remove_tag.assert_called_with( self.client.remove_tag.assert_called_with(
self.image.id, 'test' self.image.id, 'test'