NFS drivers don't honor vm size with volume from an image
The bug has been encountered with the NFS generic driver (cinder.volume.nfs.NfsDriver) and the NetApp NFS driver (cinder.volume.drivers.netapp.nfs.NetAppDirectCmodeNfsDriver). I believe that drivers based on distributed filesystem such as GlusterFS, nexenta and scality are also impacted however I didn't test it those backends. However since most of them already inherit from the RemoteFsDriver class, this should be fine. Change-Id: I14575da69a2c99c7cbcece27b40a171153371ee3 Fixes: bug #1183459
This commit is contained in:
parent
b782597725
commit
824a3b4521
@ -191,6 +191,12 @@ def convert_image(source, dest, out_format):
|
|||||||
utils.execute(*cmd, run_as_root=True)
|
utils.execute(*cmd, run_as_root=True)
|
||||||
|
|
||||||
|
|
||||||
|
def resize_image(source, size):
|
||||||
|
"""Changes the virtual size of the image."""
|
||||||
|
cmd = ('qemu-img', 'resize', source, '%sG' % size)
|
||||||
|
utils.execute(*cmd, run_as_root=False)
|
||||||
|
|
||||||
|
|
||||||
def fetch(context, image_service, image_id, path, _user_id, _project_id):
|
def fetch(context, image_service, image_id, path, _user_id, _project_id):
|
||||||
# TODO(vish): Improve context handling and add owner and auth data
|
# TODO(vish): Improve context handling and add owner and auth data
|
||||||
# when it is added to glance. Right now there is no
|
# when it is added to glance. Right now there is no
|
||||||
|
45
cinder/tests/test_image_utils.py
Normal file
45
cinder/tests/test_image_utils.py
Normal file
@ -0,0 +1,45 @@
|
|||||||
|
# vim: tabstop=4 shiftwidth=4 softtabstop=4
|
||||||
|
|
||||||
|
# Copyright (c) 2013 eNovance , Inc.
|
||||||
|
# All Rights Reserved.
|
||||||
|
#
|
||||||
|
# Licensed under the Apache License, Version 2.0 (the "License"); you may
|
||||||
|
# not use this file except in compliance with the License. You may obtain
|
||||||
|
# a copy of the License at
|
||||||
|
#
|
||||||
|
# http://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
#
|
||||||
|
# Unless required by applicable law or agreed to in writing, software
|
||||||
|
# distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
|
||||||
|
# WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
|
||||||
|
# License for the specific language governing permissions and limitations
|
||||||
|
# under the License.
|
||||||
|
"""Unit tests for image utils."""
|
||||||
|
|
||||||
|
from cinder.image import image_utils
|
||||||
|
from cinder import test
|
||||||
|
from cinder import utils
|
||||||
|
import mox
|
||||||
|
|
||||||
|
|
||||||
|
class TestUtils(test.TestCase):
|
||||||
|
def setUp(self):
|
||||||
|
super(TestUtils, self).setUp()
|
||||||
|
self._mox = mox.Mox()
|
||||||
|
self.addCleanup(self._mox.UnsetStubs)
|
||||||
|
|
||||||
|
def test_resize_image(self):
|
||||||
|
mox = self._mox
|
||||||
|
mox.StubOutWithMock(utils, 'execute')
|
||||||
|
|
||||||
|
TEST_IMG_SOURCE = 'boobar.img'
|
||||||
|
TEST_IMG_SIZE_IN_GB = 1
|
||||||
|
|
||||||
|
utils.execute('qemu-img', 'resize', TEST_IMG_SOURCE,
|
||||||
|
'%sG' % TEST_IMG_SIZE_IN_GB, run_as_root=False)
|
||||||
|
|
||||||
|
mox.ReplayAll()
|
||||||
|
|
||||||
|
image_utils.resize_image(TEST_IMG_SOURCE, TEST_IMG_SIZE_IN_GB)
|
||||||
|
|
||||||
|
mox.VerifyAll()
|
@ -33,6 +33,8 @@ from cinder.exception import ProcessExecutionError
|
|||||||
from cinder import test
|
from cinder import test
|
||||||
from cinder import units
|
from cinder import units
|
||||||
|
|
||||||
|
from cinder.image import image_utils
|
||||||
|
|
||||||
from cinder.volume import configuration as conf
|
from cinder.volume import configuration as conf
|
||||||
from cinder.volume.drivers import nfs
|
from cinder.volume.drivers import nfs
|
||||||
|
|
||||||
@ -167,6 +169,37 @@ class NfsDriverTestCase(test.TestCase):
|
|||||||
|
|
||||||
mox.VerifyAll()
|
mox.VerifyAll()
|
||||||
|
|
||||||
|
def test_copy_image_to_volume(self):
|
||||||
|
"""resize_image common case usage."""
|
||||||
|
mox = self._mox
|
||||||
|
drv = self._driver
|
||||||
|
|
||||||
|
TEST_IMG_SOURCE = 'foo.img'
|
||||||
|
|
||||||
|
volume = {'size': self.TEST_SIZE_IN_GB, 'name': TEST_IMG_SOURCE}
|
||||||
|
|
||||||
|
def fake_local_path(volume):
|
||||||
|
return volume['name']
|
||||||
|
|
||||||
|
self.stubs.Set(drv, 'local_path', fake_local_path)
|
||||||
|
|
||||||
|
mox.StubOutWithMock(image_utils, 'fetch_to_raw')
|
||||||
|
image_utils.fetch_to_raw(None, None, None, TEST_IMG_SOURCE)
|
||||||
|
|
||||||
|
mox.StubOutWithMock(image_utils, 'resize_image')
|
||||||
|
image_utils.resize_image(TEST_IMG_SOURCE, self.TEST_SIZE_IN_GB)
|
||||||
|
|
||||||
|
mox.StubOutWithMock(image_utils, 'qemu_img_info')
|
||||||
|
data = mox_lib.MockAnything()
|
||||||
|
data.virtual_size = 1024 ** 3
|
||||||
|
image_utils.qemu_img_info(TEST_IMG_SOURCE).AndReturn(data)
|
||||||
|
|
||||||
|
mox.ReplayAll()
|
||||||
|
|
||||||
|
drv.copy_image_to_volume(None, volume, None, None)
|
||||||
|
|
||||||
|
mox.VerifyAll()
|
||||||
|
|
||||||
def test_mount_nfs_should_suppress_already_mounted_error(self):
|
def test_mount_nfs_should_suppress_already_mounted_error(self):
|
||||||
"""_mount_nfs should suppress already mounted error if ensure=True
|
"""_mount_nfs should suppress already mounted error if ensure=True
|
||||||
"""
|
"""
|
||||||
|
@ -122,6 +122,23 @@ class RemoteFsDriver(driver.VolumeDriver):
|
|||||||
image_id,
|
image_id,
|
||||||
self.local_path(volume))
|
self.local_path(volume))
|
||||||
|
|
||||||
|
# NOTE (leseb): Set the virtual size of the image
|
||||||
|
# the raw conversion overwrote the destination file
|
||||||
|
# (which had the correct size)
|
||||||
|
# with the fetched glance image size,
|
||||||
|
# thus the initial 'size' parameter is not honored
|
||||||
|
# this sets the size to the one asked in the first place by the user
|
||||||
|
# and then verify the final virtual size
|
||||||
|
image_utils.resize_image(self.local_path(volume), volume['size'])
|
||||||
|
|
||||||
|
data = image_utils.qemu_img_info(self.local_path(volume))
|
||||||
|
virt_size = data.virtual_size / units.GiB
|
||||||
|
if virt_size != volume['size']:
|
||||||
|
raise exception.ImageUnacceptable(
|
||||||
|
image_id=image_id,
|
||||||
|
reason=(_("Expected volume size was %d") % volume['size'])
|
||||||
|
+ (_(" but size is now %d") % virt_size))
|
||||||
|
|
||||||
def copy_volume_to_image(self, context, volume, image_service, image_meta):
|
def copy_volume_to_image(self, context, volume, image_service, image_meta):
|
||||||
"""Copy the volume to the specified image."""
|
"""Copy the volume to the specified image."""
|
||||||
image_utils.upload_volume(context,
|
image_utils.upload_volume(context,
|
||||||
|
Loading…
Reference in New Issue
Block a user