Fix: create image from volume command
Currently the command ``openstack image create --volume`` calls cinderclient to upload the volume to image service (glance) but OSC passes ``visibility`` and ``protected`` fields which are only available in microversion 3.1 or greater. This generates an error if the user is using volume microversion < 3.1 and wants to create an image from volume. This patch fixes that by only passing ``visibility`` and ``protected`` fields when the volume microversion is 3.1 or greater and fail otherwise i.e. the following 3 cases: 1) visibility/protected argument + mv >= 3.1 = pass 2) visibility/protected argument + mv < 3.1 = fail 3) not visibility/protected argument + any mv = pass Story: 2010060 Task: 45511 Change-Id: I568a0ea0af8f7f82b16d49a6a1bb0391b99c50dc
This commit is contained in:
parent
20e7b01af8
commit
9eea28ba59
@ -21,6 +21,7 @@ import logging
|
|||||||
import os
|
import os
|
||||||
import sys
|
import sys
|
||||||
|
|
||||||
|
from cinderclient import api_versions
|
||||||
from openstack.image import image_signer
|
from openstack.image import image_signer
|
||||||
from osc_lib.api import utils as api_utils
|
from osc_lib.api import utils as api_utils
|
||||||
from osc_lib.cli import format_columns
|
from osc_lib.cli import format_columns
|
||||||
@ -483,14 +484,27 @@ class CreateImage(command.ShowOne):
|
|||||||
volume_client.volumes,
|
volume_client.volumes,
|
||||||
parsed_args.volume,
|
parsed_args.volume,
|
||||||
)
|
)
|
||||||
|
mv_kwargs = {}
|
||||||
|
if volume_client.api_version >= api_versions.APIVersion('3.1'):
|
||||||
|
mv_kwargs.update(
|
||||||
|
visibility=kwargs.get('visibility', 'private'),
|
||||||
|
protected=bool(parsed_args.protected)
|
||||||
|
)
|
||||||
|
else:
|
||||||
|
if kwargs.get('visibility') or parsed_args.protected:
|
||||||
|
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(
|
response, body = volume_client.volumes.upload_to_image(
|
||||||
source_volume.id,
|
source_volume.id,
|
||||||
parsed_args.force,
|
parsed_args.force,
|
||||||
parsed_args.name,
|
parsed_args.name,
|
||||||
parsed_args.container_format,
|
parsed_args.container_format,
|
||||||
parsed_args.disk_format,
|
parsed_args.disk_format,
|
||||||
visibility=kwargs.get('visibility', 'private'),
|
**mv_kwargs
|
||||||
protected=True if parsed_args.protected else False
|
|
||||||
)
|
)
|
||||||
info = body['os-volume_upload_image']
|
info = body['os-volume_upload_image']
|
||||||
try:
|
try:
|
||||||
|
@ -18,6 +18,7 @@ import os
|
|||||||
import tempfile
|
import tempfile
|
||||||
from unittest import mock
|
from unittest import mock
|
||||||
|
|
||||||
|
from cinderclient import api_versions
|
||||||
from openstack import exceptions as sdk_exceptions
|
from openstack import exceptions as sdk_exceptions
|
||||||
from osc_lib.cli import format_columns
|
from osc_lib.cli import format_columns
|
||||||
from osc_lib import exceptions
|
from osc_lib import exceptions
|
||||||
@ -25,9 +26,10 @@ from osc_lib import exceptions
|
|||||||
from openstackclient.image.v2 import image
|
from openstackclient.image.v2 import image
|
||||||
from openstackclient.tests.unit.identity.v3 import fakes as identity_fakes
|
from openstackclient.tests.unit.identity.v3 import fakes as identity_fakes
|
||||||
from openstackclient.tests.unit.image.v2 import fakes as image_fakes
|
from openstackclient.tests.unit.image.v2 import fakes as image_fakes
|
||||||
|
from openstackclient.tests.unit.volume.v3 import fakes as volume_fakes
|
||||||
|
|
||||||
|
|
||||||
class TestImage(image_fakes.TestImagev2):
|
class TestImage(image_fakes.TestImagev2, volume_fakes.TestVolume):
|
||||||
|
|
||||||
def setUp(self):
|
def setUp(self):
|
||||||
super(TestImage, self).setUp()
|
super(TestImage, self).setUp()
|
||||||
@ -40,6 +42,13 @@ class TestImage(image_fakes.TestImagev2):
|
|||||||
self.project_mock.reset_mock()
|
self.project_mock.reset_mock()
|
||||||
self.domain_mock = self.app.client_manager.identity.domains
|
self.domain_mock = self.app.client_manager.identity.domains
|
||||||
self.domain_mock.reset_mock()
|
self.domain_mock.reset_mock()
|
||||||
|
self.volumes_mock = self.app.client_manager.volume.volumes
|
||||||
|
fake_body = {
|
||||||
|
'os-volume_upload_image':
|
||||||
|
{'volume_type': {'name': 'fake_type'}}}
|
||||||
|
self.volumes_mock.upload_to_image.return_value = (
|
||||||
|
200, fake_body)
|
||||||
|
self.volumes_mock.reset_mock()
|
||||||
|
|
||||||
def setup_images_mock(self, count):
|
def setup_images_mock(self, count):
|
||||||
images = image_fakes.create_images(count=count)
|
images = image_fakes.create_images(count=count)
|
||||||
@ -287,6 +296,101 @@ class TestImageCreate(TestImage):
|
|||||||
use_import=True
|
use_import=True
|
||||||
)
|
)
|
||||||
|
|
||||||
|
@mock.patch('osc_lib.utils.find_resource')
|
||||||
|
@mock.patch('openstackclient.image.v2.image.get_data_file')
|
||||||
|
def test_image_create_from_volume(self, mock_get_data_f, mock_get_vol):
|
||||||
|
|
||||||
|
fake_vol_id = 'fake-volume-id'
|
||||||
|
mock_get_data_f.return_value = (None, None)
|
||||||
|
|
||||||
|
class FakeVolume:
|
||||||
|
id = fake_vol_id
|
||||||
|
|
||||||
|
mock_get_vol.return_value = FakeVolume()
|
||||||
|
|
||||||
|
arglist = [
|
||||||
|
'--volume', fake_vol_id,
|
||||||
|
self.new_image.name,
|
||||||
|
]
|
||||||
|
verifylist = [
|
||||||
|
('name', self.new_image.name),
|
||||||
|
]
|
||||||
|
parsed_args = self.check_parser(self.cmd, arglist, verifylist)
|
||||||
|
|
||||||
|
columns, data = self.cmd.take_action(parsed_args)
|
||||||
|
|
||||||
|
self.volumes_mock.upload_to_image.assert_called_with(
|
||||||
|
fake_vol_id,
|
||||||
|
False,
|
||||||
|
self.new_image.name,
|
||||||
|
'bare',
|
||||||
|
'raw'
|
||||||
|
)
|
||||||
|
|
||||||
|
@mock.patch('osc_lib.utils.find_resource')
|
||||||
|
@mock.patch('openstackclient.image.v2.image.get_data_file')
|
||||||
|
def test_image_create_from_volume_fail(self, mock_get_data_f,
|
||||||
|
mock_get_vol):
|
||||||
|
|
||||||
|
fake_vol_id = 'fake-volume-id'
|
||||||
|
mock_get_data_f.return_value = (None, None)
|
||||||
|
|
||||||
|
class FakeVolume:
|
||||||
|
id = fake_vol_id
|
||||||
|
|
||||||
|
mock_get_vol.return_value = FakeVolume()
|
||||||
|
|
||||||
|
arglist = [
|
||||||
|
'--volume', fake_vol_id,
|
||||||
|
self.new_image.name,
|
||||||
|
'--public'
|
||||||
|
]
|
||||||
|
verifylist = [
|
||||||
|
('name', self.new_image.name),
|
||||||
|
]
|
||||||
|
parsed_args = self.check_parser(self.cmd, arglist, verifylist)
|
||||||
|
|
||||||
|
self.assertRaises(exceptions.CommandError, self.cmd.take_action,
|
||||||
|
parsed_args)
|
||||||
|
|
||||||
|
@mock.patch('osc_lib.utils.find_resource')
|
||||||
|
@mock.patch('openstackclient.image.v2.image.get_data_file')
|
||||||
|
def test_image_create_from_volume_v31(self, mock_get_data_f,
|
||||||
|
mock_get_vol):
|
||||||
|
|
||||||
|
self.app.client_manager.volume.api_version = (
|
||||||
|
api_versions.APIVersion('3.1'))
|
||||||
|
|
||||||
|
fake_vol_id = 'fake-volume-id'
|
||||||
|
mock_get_data_f.return_value = (None, None)
|
||||||
|
|
||||||
|
class FakeVolume:
|
||||||
|
id = fake_vol_id
|
||||||
|
|
||||||
|
mock_get_vol.return_value = FakeVolume()
|
||||||
|
|
||||||
|
arglist = [
|
||||||
|
'--volume', fake_vol_id,
|
||||||
|
self.new_image.name,
|
||||||
|
'--public'
|
||||||
|
]
|
||||||
|
verifylist = [
|
||||||
|
('name', self.new_image.name),
|
||||||
|
]
|
||||||
|
parsed_args = self.check_parser(self.cmd, arglist, verifylist)
|
||||||
|
|
||||||
|
columns, data = self.cmd.take_action(parsed_args)
|
||||||
|
|
||||||
|
self.volumes_mock.upload_to_image.assert_called_with(
|
||||||
|
fake_vol_id,
|
||||||
|
False,
|
||||||
|
self.new_image.name,
|
||||||
|
'bare',
|
||||||
|
'raw',
|
||||||
|
visibility='public',
|
||||||
|
protected=False
|
||||||
|
)
|
||||||
|
|
||||||
|
|
||||||
class TestAddProjectToImage(TestImage):
|
class TestAddProjectToImage(TestImage):
|
||||||
|
|
||||||
|
@ -0,0 +1,7 @@
|
|||||||
|
---
|
||||||
|
fixes:
|
||||||
|
- |
|
||||||
|
Fixed create image from volume command. If user wants to
|
||||||
|
pass ``visibility`` and ``protected`` fields, they need to
|
||||||
|
specify volume microversion 3.1 or greater by passing
|
||||||
|
``os-volume-api-version 3.1`` with the command.
|
Loading…
Reference in New Issue
Block a user