Stop allocating double space for raw images

There are two possibilities when ironic conductor will be converting
instance image to raw format:

1. when iscsi deploy is used and [DEFAULT]force_raw_images config option
   is set to True;
2. when direct deploy is used, [agent]image_download_source config
   option is set to "http" and [DEFAULT]force_raw_images and
   [agent]stream_raw_images config options are set to True.

If one of the above conditions is met, ironic tries to allocate the disk
space needed for conversion even if the image is already raw, basically
doubling the disk space requirements for each image. This change adds a
check that skips the unnecessary conductor disk space allocation attempt
if the image is already raw.

Story: 2007830
Task: 40112
Change-Id: I730f9ec89ff1598d42777e4a503f46316eb85cad
This commit is contained in:
Vladyslav Drok 2020-06-17 23:32:27 +02:00
parent 1c0a9c8c28
commit fbebcc3201
4 changed files with 61 additions and 18 deletions

View File

@ -414,9 +414,8 @@ def fetch(context, image_href, path, force_raw=False):
image_to_raw(image_href, path, "%s.part" % path) image_to_raw(image_href, path, "%s.part" % path)
def image_to_raw(image_href, path, path_tmp): def force_raw_get_source_format(image_href, path):
with fileutils.remove_path_on_error(path_tmp): data = disk_utils.qemu_img_info(path)
data = disk_utils.qemu_img_info(path_tmp)
fmt = data.file_format fmt = data.file_format
if fmt is None: if fmt is None:
@ -431,6 +430,21 @@ def image_to_raw(image_href, path, path_tmp):
reason=_("fmt=%(fmt)s backed by: %(backing_file)s") % reason=_("fmt=%(fmt)s backed by: %(backing_file)s") %
{'fmt': fmt, 'backing_file': backing_file}) {'fmt': fmt, 'backing_file': backing_file})
return fmt
def force_raw_will_convert(image_href, path_tmp):
with fileutils.remove_path_on_error(path_tmp):
fmt = force_raw_get_source_format(image_href, path_tmp)
if fmt != "raw":
return True
return False
def image_to_raw(image_href, path, path_tmp):
with fileutils.remove_path_on_error(path_tmp):
fmt = force_raw_get_source_format(image_href, path_tmp)
if fmt != "raw": if fmt != "raw":
staged = "%s.converted" % path staged = "%s.converted" % path
LOG.debug("%(image)s was %(format)s, converting to raw", LOG.debug("%(image)s was %(format)s, converting to raw",

View File

@ -313,6 +313,7 @@ def _fetch(context, image_href, path, force_raw=False):
# Notes(yjiang5): If glance can provide the virtual size information, # Notes(yjiang5): If glance can provide the virtual size information,
# then we can firstly clean cache and then invoke images.fetch(). # then we can firstly clean cache and then invoke images.fetch().
if force_raw: if force_raw:
if images.force_raw_will_convert(image_href, path_tmp):
required_space = images.converted_size(path_tmp) required_space = images.converted_size(path_tmp)
directory = os.path.dirname(path_tmp) directory = os.path.dirname(path_tmp)
_clean_up_caches(directory, required_space) _clean_up_caches(directory, required_space)

View File

@ -733,8 +733,12 @@ class TestFetchCleanup(base.TestCase):
@mock.patch.object(images, 'converted_size', autospec=True) @mock.patch.object(images, 'converted_size', autospec=True)
@mock.patch.object(images, 'fetch', autospec=True) @mock.patch.object(images, 'fetch', autospec=True)
@mock.patch.object(images, 'image_to_raw', autospec=True) @mock.patch.object(images, 'image_to_raw', autospec=True)
@mock.patch.object(images, 'force_raw_will_convert', autospec=True,
return_value=True)
@mock.patch.object(image_cache, '_clean_up_caches', autospec=True) @mock.patch.object(image_cache, '_clean_up_caches', autospec=True)
def test__fetch(self, mock_clean, mock_raw, mock_fetch, mock_size): def test__fetch(
self, mock_clean, mock_will_convert, mock_raw, mock_fetch,
mock_size):
mock_size.return_value = 100 mock_size.return_value = 100
image_cache._fetch('fake', 'fake-uuid', '/foo/bar', force_raw=True) image_cache._fetch('fake', 'fake-uuid', '/foo/bar', force_raw=True)
mock_fetch.assert_called_once_with('fake', 'fake-uuid', mock_fetch.assert_called_once_with('fake', 'fake-uuid',
@ -742,3 +746,22 @@ class TestFetchCleanup(base.TestCase):
mock_clean.assert_called_once_with('/foo', 100) mock_clean.assert_called_once_with('/foo', 100)
mock_raw.assert_called_once_with('fake-uuid', '/foo/bar', mock_raw.assert_called_once_with('fake-uuid', '/foo/bar',
'/foo/bar.part') '/foo/bar.part')
mock_will_convert.assert_called_once_with('fake-uuid', '/foo/bar.part')
@mock.patch.object(images, 'converted_size', autospec=True)
@mock.patch.object(images, 'fetch', autospec=True)
@mock.patch.object(images, 'image_to_raw', autospec=True)
@mock.patch.object(images, 'force_raw_will_convert', autospec=True,
return_value=False)
@mock.patch.object(image_cache, '_clean_up_caches', autospec=True)
def test__fetch_already_raw(
self, mock_clean, mock_will_convert, mock_raw, mock_fetch,
mock_size):
image_cache._fetch('fake', 'fake-uuid', '/foo/bar', force_raw=True)
mock_fetch.assert_called_once_with('fake', 'fake-uuid',
'/foo/bar.part', force_raw=False)
mock_clean.assert_not_called()
mock_size.assert_not_called()
mock_raw.assert_called_once_with('fake-uuid', '/foo/bar',
'/foo/bar.part')
mock_will_convert.assert_called_once_with('fake-uuid', '/foo/bar.part')

View File

@ -0,0 +1,5 @@
---
fixes:
- |
Ironic now does not try to allocate the space needed for instance image
conversion to raw format if it is already raw.