Merge "Separate out routine for getting qemu_img_info"
This commit is contained in:
commit
3ecfc460b3
@ -221,14 +221,56 @@ def fetch(context, image_service, image_id, path, _user_id, _project_id):
|
|||||||
LOG.info(msg, {"sz": fsz_mb, "mbps": mbps})
|
LOG.info(msg, {"sz": fsz_mb, "mbps": mbps})
|
||||||
|
|
||||||
|
|
||||||
|
def get_qemu_data(image_id, has_meta, disk_format_raw, dest, run_as_root):
|
||||||
|
# We may be on a system that doesn't have qemu-img installed. That
|
||||||
|
# is ok if we are working with a RAW image. This logic checks to see
|
||||||
|
# if qemu-img is installed. If not we make sure the image is RAW and
|
||||||
|
# throw an exception if not. Otherwise we stop before needing
|
||||||
|
# qemu-img. Systems with qemu-img will always progress through the
|
||||||
|
# whole function.
|
||||||
|
try:
|
||||||
|
# Use the empty tmp file to make sure qemu_img_info works.
|
||||||
|
data = qemu_img_info(dest, run_as_root=run_as_root)
|
||||||
|
# There are a lot of cases that can cause a process execution
|
||||||
|
# error, but until we do more work to separate out the various
|
||||||
|
# cases we'll keep the general catch here
|
||||||
|
except processutils.ProcessExecutionError:
|
||||||
|
data = None
|
||||||
|
if has_meta:
|
||||||
|
if not disk_format_raw:
|
||||||
|
raise exception.ImageUnacceptable(
|
||||||
|
reason=_("qemu-img is not installed and image is of "
|
||||||
|
"type %s. Only RAW images can be used if "
|
||||||
|
"qemu-img is not installed.") %
|
||||||
|
disk_format_raw,
|
||||||
|
image_id=image_id)
|
||||||
|
else:
|
||||||
|
raise exception.ImageUnacceptable(
|
||||||
|
reason=_("qemu-img is not installed and the disk "
|
||||||
|
"format is not specified. Only RAW images "
|
||||||
|
"can be used if qemu-img is not installed."),
|
||||||
|
image_id=image_id)
|
||||||
|
return data
|
||||||
|
|
||||||
|
|
||||||
def fetch_verify_image(context, image_service, image_id, dest,
|
def fetch_verify_image(context, image_service, image_id, dest,
|
||||||
user_id=None, project_id=None, size=None,
|
user_id=None, project_id=None, size=None,
|
||||||
run_as_root=True):
|
run_as_root=True):
|
||||||
fetch(context, image_service, image_id, dest,
|
fetch(context, image_service, image_id, dest,
|
||||||
None, None)
|
None, None)
|
||||||
|
image_meta = image_service.show(context, image_id)
|
||||||
|
|
||||||
with fileutils.remove_path_on_error(dest):
|
with fileutils.remove_path_on_error(dest):
|
||||||
data = qemu_img_info(dest, run_as_root=run_as_root)
|
has_meta = False if not image_meta else True
|
||||||
|
try:
|
||||||
|
format_raw = True if image_meta['disk_format'] == 'raw' else False
|
||||||
|
except TypeError:
|
||||||
|
format_raw = False
|
||||||
|
data = get_qemu_data(image_id, has_meta, format_raw,
|
||||||
|
dest, run_as_root)
|
||||||
|
# We can only really do verification of the image if we have
|
||||||
|
# qemu data to use
|
||||||
|
if data is not None:
|
||||||
fmt = data.file_format
|
fmt = data.file_format
|
||||||
if fmt is None:
|
if fmt is None:
|
||||||
raise exception.ImageUnacceptable(
|
raise exception.ImageUnacceptable(
|
||||||
@ -249,7 +291,8 @@ def fetch_verify_image(context, image_service, image_id, dest,
|
|||||||
params = {'image_size': data.virtual_size, 'volume_size': size}
|
params = {'image_size': data.virtual_size, 'volume_size': size}
|
||||||
reason = _("Size is %(image_size)dGB and doesn't fit in a "
|
reason = _("Size is %(image_size)dGB and doesn't fit in a "
|
||||||
"volume of size %(volume_size)dGB.") % params
|
"volume of size %(volume_size)dGB.") % params
|
||||||
raise exception.ImageUnacceptable(image_id=image_id, reason=reason)
|
raise exception.ImageUnacceptable(image_id=image_id,
|
||||||
|
reason=reason)
|
||||||
|
|
||||||
|
|
||||||
def fetch_to_vhd(context, image_service,
|
def fetch_to_vhd(context, image_service,
|
||||||
@ -280,31 +323,15 @@ def fetch_to_volume_format(context, image_service,
|
|||||||
# Unfortunately it seems that you can't pipe to 'qemu-img convert' because
|
# Unfortunately it seems that you can't pipe to 'qemu-img convert' because
|
||||||
# it seeks. Maybe we can think of something for a future version.
|
# it seeks. Maybe we can think of something for a future version.
|
||||||
with temporary_file() as tmp:
|
with temporary_file() as tmp:
|
||||||
# We may be on a system that doesn't have qemu-img installed. That
|
has_meta = False if not image_meta else True
|
||||||
# is ok if we are working with a RAW image. This logic checks to see
|
|
||||||
# if qemu-img is installed. If not we make sure the image is RAW and
|
|
||||||
# throw an exception if not. Otherwise we stop before needing
|
|
||||||
# qemu-img. Systems with qemu-img will always progress through the
|
|
||||||
# whole function.
|
|
||||||
try:
|
try:
|
||||||
# Use the empty tmp file to make sure qemu_img_info works.
|
format_raw = True if image_meta['disk_format'] == 'raw' else False
|
||||||
qemu_img_info(tmp, run_as_root=run_as_root)
|
except TypeError:
|
||||||
except processutils.ProcessExecutionError:
|
format_raw = False
|
||||||
|
data = get_qemu_data(image_id, has_meta, format_raw,
|
||||||
|
tmp, run_as_root)
|
||||||
|
if data is None:
|
||||||
qemu_img = False
|
qemu_img = False
|
||||||
if image_meta:
|
|
||||||
if image_meta['disk_format'] != 'raw':
|
|
||||||
raise exception.ImageUnacceptable(
|
|
||||||
reason=_("qemu-img is not installed and image is of "
|
|
||||||
"type %s. Only RAW images can be used if "
|
|
||||||
"qemu-img is not installed.") %
|
|
||||||
image_meta['disk_format'],
|
|
||||||
image_id=image_id)
|
|
||||||
else:
|
|
||||||
raise exception.ImageUnacceptable(
|
|
||||||
reason=_("qemu-img is not installed and the disk "
|
|
||||||
"format is not specified. Only RAW images "
|
|
||||||
"can be used if qemu-img is not installed."),
|
|
||||||
image_id=image_id)
|
|
||||||
|
|
||||||
tmp_images = TemporaryImages.for_image_service(image_service)
|
tmp_images = TemporaryImages.for_image_service(image_service)
|
||||||
tmp_image = tmp_images.get(context, image_id)
|
tmp_image = tmp_images.get(context, image_id)
|
||||||
|
@ -792,7 +792,7 @@ class TestFetchToVolumeFormat(test.TestCase):
|
|||||||
blocksize)
|
blocksize)
|
||||||
|
|
||||||
self.assertIsNone(output)
|
self.assertIsNone(output)
|
||||||
image_service.show.assert_called_once_with(ctxt, image_id)
|
self.assertEqual(2, image_service.show.call_count)
|
||||||
self.assertEqual(2, mock_temp.call_count)
|
self.assertEqual(2, mock_temp.call_count)
|
||||||
mock_info.assert_has_calls([
|
mock_info.assert_has_calls([
|
||||||
mock.call(tmp, run_as_root=True),
|
mock.call(tmp, run_as_root=True),
|
||||||
@ -1243,6 +1243,86 @@ class TestFetchToVolumeFormat(test.TestCase):
|
|||||||
mock_convert.assert_called_once_with(tmp, dest, volume_format,
|
mock_convert.assert_called_once_with(tmp, dest, volume_format,
|
||||||
run_as_root=run_as_root)
|
run_as_root=run_as_root)
|
||||||
|
|
||||||
|
@mock.patch('cinder.image.image_utils.fetch')
|
||||||
|
@mock.patch('cinder.image.image_utils.qemu_img_info',
|
||||||
|
side_effect=processutils.ProcessExecutionError)
|
||||||
|
@mock.patch('cinder.image.image_utils.temporary_file')
|
||||||
|
@mock.patch('cinder.image.image_utils.CONF')
|
||||||
|
def test_no_qemu_img_fetch_verify_image(self, mock_conf,
|
||||||
|
mock_temp, mock_info,
|
||||||
|
mock_fetch):
|
||||||
|
ctxt = mock.sentinel.context
|
||||||
|
image_service = mock.Mock(temp_images=None)
|
||||||
|
image_id = mock.sentinel.image_id
|
||||||
|
dest = mock.sentinel.dest
|
||||||
|
ctxt.user_id = user_id = mock.sentinel.user_id
|
||||||
|
project_id = mock.sentinel.project_id
|
||||||
|
size = 4321
|
||||||
|
run_as_root = mock.sentinel.run_as_root
|
||||||
|
|
||||||
|
image_service.show.return_value = {'disk_format': 'raw',
|
||||||
|
'size': 41126400}
|
||||||
|
|
||||||
|
image_utils.fetch_verify_image(
|
||||||
|
ctxt, image_service, image_id, dest,
|
||||||
|
user_id=user_id, project_id=project_id, size=size,
|
||||||
|
run_as_root=run_as_root)
|
||||||
|
|
||||||
|
image_service.show.assert_called_once_with(ctxt, image_id)
|
||||||
|
mock_info.assert_called_once_with(dest, run_as_root=run_as_root)
|
||||||
|
mock_fetch.assert_called_once_with(ctxt, image_service, image_id,
|
||||||
|
dest, None, None)
|
||||||
|
|
||||||
|
@mock.patch('cinder.image.image_utils.qemu_img_info',
|
||||||
|
side_effect=processutils.ProcessExecutionError)
|
||||||
|
@mock.patch('cinder.image.image_utils.temporary_file')
|
||||||
|
@mock.patch('cinder.image.image_utils.CONF')
|
||||||
|
def test_get_qemu_data_returns_none(self, mock_conf, mock_temp, mock_info):
|
||||||
|
image_id = mock.sentinel.image_id
|
||||||
|
dest = mock.sentinel.dest
|
||||||
|
run_as_root = mock.sentinel.run_as_root
|
||||||
|
disk_format_raw = True
|
||||||
|
has_meta = True
|
||||||
|
|
||||||
|
output = image_utils.get_qemu_data(image_id, has_meta,
|
||||||
|
disk_format_raw, dest,
|
||||||
|
run_as_root=run_as_root)
|
||||||
|
|
||||||
|
self.assertIsNone(output)
|
||||||
|
|
||||||
|
@mock.patch('cinder.image.image_utils.qemu_img_info',
|
||||||
|
side_effect=processutils.ProcessExecutionError)
|
||||||
|
@mock.patch('cinder.image.image_utils.temporary_file')
|
||||||
|
@mock.patch('cinder.image.image_utils.CONF')
|
||||||
|
def test_get_qemu_data_with_image_meta_exception(self, mock_conf,
|
||||||
|
mock_temp, mock_info):
|
||||||
|
image_id = mock.sentinel.image_id
|
||||||
|
dest = mock.sentinel.dest
|
||||||
|
run_as_root = mock.sentinel.run_as_root
|
||||||
|
disk_format_raw = False
|
||||||
|
has_meta = True
|
||||||
|
self.assertRaises(
|
||||||
|
exception.ImageUnacceptable,
|
||||||
|
image_utils.get_qemu_data, image_id, has_meta, disk_format_raw,
|
||||||
|
dest, run_as_root=run_as_root)
|
||||||
|
|
||||||
|
@mock.patch('cinder.image.image_utils.qemu_img_info',
|
||||||
|
side_effect=processutils.ProcessExecutionError)
|
||||||
|
@mock.patch('cinder.image.image_utils.temporary_file')
|
||||||
|
@mock.patch('cinder.image.image_utils.CONF')
|
||||||
|
def test_get_qemu_data_without_image_meta_except(self, mock_conf,
|
||||||
|
mock_temp, mock_info):
|
||||||
|
image_id = mock.sentinel.image_id
|
||||||
|
dest = mock.sentinel.dest
|
||||||
|
run_as_root = mock.sentinel.run_as_root
|
||||||
|
|
||||||
|
disk_format_raw = False
|
||||||
|
has_meta = False
|
||||||
|
self.assertRaises(
|
||||||
|
exception.ImageUnacceptable,
|
||||||
|
image_utils.get_qemu_data, image_id, has_meta, disk_format_raw,
|
||||||
|
dest, run_as_root=run_as_root)
|
||||||
|
|
||||||
|
|
||||||
class TestXenserverUtils(test.TestCase):
|
class TestXenserverUtils(test.TestCase):
|
||||||
def test_is_xenserver_format(self):
|
def test_is_xenserver_format(self):
|
||||||
|
Loading…
Reference in New Issue
Block a user