Fixed copy-on-write mode in GPFS NFS driver
IBM Spectrum Scale cinder driver (GPFS) support copy-on-write feature in all the configuration. Resolving the bug mentioned below will enable mmclone feature of the IBM Spectrum Scale filesystem to provide better performance while configured in GPFS NFS mode. Closes-Bug: #1947134 Closes-Bug: #1947123 Change-Id: I3e77c890c7abca85dab92500eae989b4dff9824d
This commit is contained in:
parent
bd63fa14b1
commit
dcc1916469
@ -153,6 +153,25 @@ class GPFSDriverTestCase(test.TestCase):
|
|||||||
self.assertRaises(exception.VolumeBackendAPIException,
|
self.assertRaises(exception.VolumeBackendAPIException,
|
||||||
self.driver._check_gpfs_state)
|
self.driver._check_gpfs_state)
|
||||||
|
|
||||||
|
@mock.patch('cinder.utils.execute')
|
||||||
|
def test_same_filesystem_ok(self, mock_exec):
|
||||||
|
# returns filesystem id in hex
|
||||||
|
mock_exec.return_value = ('ef0009600000002\nef0009600000002\n', '')
|
||||||
|
self.assertTrue(self.driver._same_filesystem('/path1', '/path2'))
|
||||||
|
|
||||||
|
@mock.patch('cinder.utils.execute')
|
||||||
|
def test_same_filesystem_not_ok(self, mock_exec):
|
||||||
|
# returns filesystem id in hex
|
||||||
|
mock_exec.return_value = ('ef0009600000002\n000000000000007\n', '')
|
||||||
|
self.assertFalse(self.driver._same_filesystem('/path1', '/path2'))
|
||||||
|
|
||||||
|
@mock.patch('cinder.utils.execute')
|
||||||
|
def test_same_filesystem_failed(self, mock_exec):
|
||||||
|
mock_exec.side_effect = processutils.ProcessExecutionError(
|
||||||
|
stdout='test', stderr='test')
|
||||||
|
self.assertRaises(exception.VolumeBackendAPIException,
|
||||||
|
self.driver._same_filesystem, '', '')
|
||||||
|
|
||||||
@mock.patch('cinder.utils.execute')
|
@mock.patch('cinder.utils.execute')
|
||||||
def test_get_fs_from_path_ok(self, mock_exec):
|
def test_get_fs_from_path_ok(self, mock_exec):
|
||||||
mock_exec.return_value = ('Filesystem 1K-blocks '
|
mock_exec.return_value = ('Filesystem 1K-blocks '
|
||||||
@ -559,12 +578,12 @@ class GPFSDriverTestCase(test.TestCase):
|
|||||||
conf.SHARED_CONF_GROUP)
|
conf.SHARED_CONF_GROUP)
|
||||||
|
|
||||||
# fail configuration.gpfs_images_share_mode == 'copy_on_write' and not
|
# fail configuration.gpfs_images_share_mode == 'copy_on_write' and not
|
||||||
# _same_filesystem(configuration.gpfs_mount_point_base,
|
# self._same_filesystem(configuration.gpfs_mount_point_base,
|
||||||
# configuration.gpfs_images_dir)
|
# configuration.gpfs_images_dir)
|
||||||
self.override_config('gpfs_images_share_mode', 'copy_on_write',
|
self.override_config('gpfs_images_share_mode', 'copy_on_write',
|
||||||
conf.SHARED_CONF_GROUP)
|
conf.SHARED_CONF_GROUP)
|
||||||
with mock.patch('cinder.volume.drivers.ibm.gpfs._same_filesystem',
|
with mock.patch('cinder.volume.drivers.ibm.gpfs.GPFSDriver.'
|
||||||
return_value=False):
|
'_same_filesystem', return_value=False):
|
||||||
self.assertRaises(exception.VolumeBackendAPIException,
|
self.assertRaises(exception.VolumeBackendAPIException,
|
||||||
self.driver.check_for_setup_error)
|
self.driver.check_for_setup_error)
|
||||||
|
|
||||||
@ -1281,21 +1300,22 @@ class GPFSDriverTestCase(test.TestCase):
|
|||||||
@mock.patch('cinder.image.image_utils.qemu_img_info')
|
@mock.patch('cinder.image.image_utils.qemu_img_info')
|
||||||
@mock.patch('cinder.volume.drivers.ibm.gpfs.GPFSDriver.'
|
@mock.patch('cinder.volume.drivers.ibm.gpfs.GPFSDriver.'
|
||||||
'_is_gpfs_parent_file')
|
'_is_gpfs_parent_file')
|
||||||
@mock.patch('cinder.volume.drivers.ibm.gpfs.GPFSDriver.local_path')
|
@mock.patch('cinder.volume.drivers.ibm.gpfs.GPFSDriver.'
|
||||||
|
'_get_volume_path')
|
||||||
@mock.patch('cinder.volume.drivers.ibm.gpfs.GPFSDriver._is_cloneable')
|
@mock.patch('cinder.volume.drivers.ibm.gpfs.GPFSDriver._is_cloneable')
|
||||||
@mock.patch('cinder.volume.drivers.ibm.gpfs.GPFSDriver.'
|
@mock.patch('cinder.volume.drivers.ibm.gpfs.GPFSDriver.'
|
||||||
'_verify_gpfs_path_state')
|
'_verify_gpfs_path_state')
|
||||||
def test_clone_image_format_raw_copy(self,
|
def test_clone_image_format_raw_copy(self,
|
||||||
mock_verify_gpfs_path_state,
|
mock_verify_gpfs_path_state,
|
||||||
mock_is_cloneable,
|
mock_is_cloneable,
|
||||||
mock_local_path,
|
mock_get_volume_path,
|
||||||
mock_is_gpfs_parent_file,
|
mock_is_gpfs_parent_file,
|
||||||
mock_qemu_img_info,
|
mock_qemu_img_info,
|
||||||
mock_copyfile,
|
mock_copyfile,
|
||||||
mock_set_rw_permission,
|
mock_set_rw_permission,
|
||||||
mock_resize_volume_file):
|
mock_resize_volume_file):
|
||||||
mock_is_cloneable.return_value = (True, 'test', self.images_dir)
|
mock_is_cloneable.return_value = (True, 'test', self.images_dir)
|
||||||
mock_local_path.return_value = self.volumes_path
|
mock_get_volume_path.return_value = self.volumes_path
|
||||||
mock_qemu_img_info.return_value = self._fake_qemu_raw_image_info('')
|
mock_qemu_img_info.return_value = self._fake_qemu_raw_image_info('')
|
||||||
volume = self._fake_volume()
|
volume = self._fake_volume()
|
||||||
org_value = self.driver.configuration.gpfs_images_share_mode
|
org_value = self.driver.configuration.gpfs_images_share_mode
|
||||||
@ -2255,7 +2275,11 @@ class GPFSNFSDriverTestCase(test.TestCase):
|
|||||||
@mock.patch('cinder.volume.volume_utils.is_group_a_cg_snapshot_type')
|
@mock.patch('cinder.volume.volume_utils.is_group_a_cg_snapshot_type')
|
||||||
@mock.patch('cinder.volume.drivers.ibm.gpfs.GPFSNFSDriver.'
|
@mock.patch('cinder.volume.drivers.ibm.gpfs.GPFSNFSDriver.'
|
||||||
'_get_mount_point_for_share')
|
'_get_mount_point_for_share')
|
||||||
def test_local_path(self, mock_mount_point,
|
@mock.patch('cinder.volume.drivers.ibm.gpfs.GPFSNFSDriver.'
|
||||||
|
'_find_share')
|
||||||
|
def test_local_path(self,
|
||||||
|
mock_find_share,
|
||||||
|
mock_mount_point,
|
||||||
mock_group_cg_snapshot_type,
|
mock_group_cg_snapshot_type,
|
||||||
mock_group):
|
mock_group):
|
||||||
mock_mount_point.return_value = self.TEST_MNT_POINT_BASE
|
mock_mount_point.return_value = self.TEST_MNT_POINT_BASE
|
||||||
@ -2263,7 +2287,7 @@ class GPFSNFSDriverTestCase(test.TestCase):
|
|||||||
volume = self._fake_volume()
|
volume = self._fake_volume()
|
||||||
group = self._fake_group()
|
group = self._fake_group()
|
||||||
mock_group.return_value = group
|
mock_group.return_value = group
|
||||||
volume['provider_location'] = self.TEST_MNT_POINT_BASE
|
mock_find_share.return_value = self.TEST_VOLUME_PATH
|
||||||
local_volume_path_in_cg = os.path.join(self.TEST_MNT_POINT_BASE,
|
local_volume_path_in_cg = os.path.join(self.TEST_MNT_POINT_BASE,
|
||||||
'consisgroup-' +
|
'consisgroup-' +
|
||||||
fake.CONSISTENCY_GROUP_ID,
|
fake.CONSISTENCY_GROUP_ID,
|
||||||
|
@ -134,11 +134,6 @@ def _different(difference_tuple):
|
|||||||
return False
|
return False
|
||||||
|
|
||||||
|
|
||||||
def _same_filesystem(path1, path2):
|
|
||||||
"""Return true if the two paths are in the same GPFS file system."""
|
|
||||||
return os.lstat(path1).st_dev == os.lstat(path2).st_dev
|
|
||||||
|
|
||||||
|
|
||||||
def _sizestr(size_in_g):
|
def _sizestr(size_in_g):
|
||||||
"""Convert the specified size into a string value."""
|
"""Convert the specified size into a string value."""
|
||||||
return '%sG' % size_in_g
|
return '%sG' % size_in_g
|
||||||
@ -204,6 +199,19 @@ class GPFSDriver(driver.CloneableImageVD,
|
|||||||
raise exception.VolumeBackendAPIException(
|
raise exception.VolumeBackendAPIException(
|
||||||
data=_('GPFS is not running, state: %s.') % gpfs_state)
|
data=_('GPFS is not running, state: %s.') % gpfs_state)
|
||||||
|
|
||||||
|
def _same_filesystem(self, path1, path2):
|
||||||
|
"""Return true if the two paths are in the same GPFS file system."""
|
||||||
|
try:
|
||||||
|
(out, err) = self.gpfs_execute('stat', '-f', '-c', '"%i"',
|
||||||
|
path1, path2)
|
||||||
|
lines = out.splitlines()
|
||||||
|
return lines[0] == lines[1]
|
||||||
|
except processutils.ProcessExecutionError as exc:
|
||||||
|
LOG.error('Failed to issue stat command on path '
|
||||||
|
'%(path1)s and path %(path2)s, error: %(error)s',
|
||||||
|
{'path1': path1, 'path2': path2, 'error': exc.stderr})
|
||||||
|
raise exception.VolumeBackendAPIException(data=exc.stderr)
|
||||||
|
|
||||||
def _get_filesystem_from_path(self, path):
|
def _get_filesystem_from_path(self, path):
|
||||||
"""Return filesystem for specified path."""
|
"""Return filesystem for specified path."""
|
||||||
try:
|
try:
|
||||||
@ -435,8 +443,8 @@ class GPFSDriver(driver.CloneableImageVD,
|
|||||||
raise exception.VolumeBackendAPIException(data=msg)
|
raise exception.VolumeBackendAPIException(data=msg)
|
||||||
|
|
||||||
if(self.configuration.gpfs_images_share_mode == 'copy_on_write' and
|
if(self.configuration.gpfs_images_share_mode == 'copy_on_write' and
|
||||||
not _same_filesystem(self.configuration.gpfs_mount_point_base,
|
not self._same_filesystem(self.configuration.gpfs_mount_point_base,
|
||||||
self.configuration.gpfs_images_dir)):
|
self.configuration.gpfs_images_dir)):
|
||||||
msg = (_('gpfs_images_share_mode is set to copy_on_write, but '
|
msg = (_('gpfs_images_share_mode is set to copy_on_write, but '
|
||||||
'%(vol)s and %(img)s belong to different file '
|
'%(vol)s and %(img)s belong to different file '
|
||||||
'systems.') %
|
'systems.') %
|
||||||
@ -907,13 +915,13 @@ class GPFSDriver(driver.CloneableImageVD,
|
|||||||
{'img': image_id, 'reas': reason})
|
{'img': image_id, 'reas': reason})
|
||||||
return (None, False)
|
return (None, False)
|
||||||
|
|
||||||
vol_path = self.local_path(volume)
|
|
||||||
|
|
||||||
data = image_utils.qemu_img_info(image_path)
|
data = image_utils.qemu_img_info(image_path)
|
||||||
|
|
||||||
# if image format is already raw either clone it or
|
# if image format is already raw either clone it or
|
||||||
# copy it depending on config file settings
|
# copy it depending on config file settings
|
||||||
|
# GPFS command (mmclone) needs to run on GPFS node on GPFS path
|
||||||
if data.file_format == 'raw':
|
if data.file_format == 'raw':
|
||||||
|
vol_path = self._get_volume_path(volume)
|
||||||
if (self.configuration.gpfs_images_share_mode ==
|
if (self.configuration.gpfs_images_share_mode ==
|
||||||
'copy_on_write'):
|
'copy_on_write'):
|
||||||
LOG.debug('Clone image to vol %s using mmclone.',
|
LOG.debug('Clone image to vol %s using mmclone.',
|
||||||
@ -929,7 +937,9 @@ class GPFSDriver(driver.CloneableImageVD,
|
|||||||
shutil.copyfile(image_path, vol_path)
|
shutil.copyfile(image_path, vol_path)
|
||||||
|
|
||||||
# if image is not raw convert it to raw into vol_path destination
|
# if image is not raw convert it to raw into vol_path destination
|
||||||
|
# Image conversion can be run locally on GPFS mount path
|
||||||
else:
|
else:
|
||||||
|
vol_path = self.local_path(volume)
|
||||||
LOG.debug('Clone image to vol %s using qemu convert.',
|
LOG.debug('Clone image to vol %s using qemu convert.',
|
||||||
volume['id'])
|
volume['id'])
|
||||||
image_utils.convert_image(image_path, vol_path, 'raw')
|
image_utils.convert_image(image_path, vol_path, 'raw')
|
||||||
@ -1587,7 +1597,7 @@ class GPFSNFSDriver(GPFSDriver, nfs.NfsDriver, san.SanDriver):
|
|||||||
|
|
||||||
def local_path(self, volume):
|
def local_path(self, volume):
|
||||||
"""Returns the local path for the specified volume."""
|
"""Returns the local path for the specified volume."""
|
||||||
remotefs_share = volume['provider_location']
|
remotefs_share = self._find_share(volume)
|
||||||
base_local_path = self._get_mount_point_for_share(remotefs_share)
|
base_local_path = self._get_mount_point_for_share(remotefs_share)
|
||||||
|
|
||||||
# Check if the volume is part of a consistency group and return
|
# Check if the volume is part of a consistency group and return
|
||||||
|
10
releasenotes/notes/bug-gpfs-fix-nfs-cow.yaml
Normal file
10
releasenotes/notes/bug-gpfs-fix-nfs-cow.yaml
Normal file
@ -0,0 +1,10 @@
|
|||||||
|
---
|
||||||
|
fixes:
|
||||||
|
- |
|
||||||
|
`Bug #1947134 <https://bugs.launchpad.net/cinder/+bug/1947134>`_: Fixed
|
||||||
|
the initialization of GPFS NFS driver when gpfs_images_share_mode
|
||||||
|
is set to copy_on_write by correcting _same_filesystem functionality.
|
||||||
|
- |
|
||||||
|
`Bug #1947123 <https://bugs.launchpad.net/cinder/+bug/1947123>`_: Fixed
|
||||||
|
the volume creation issue in GPFS NFS driver when gpfs_images_share_mode
|
||||||
|
is set to copy_on_write.
|
Loading…
Reference in New Issue
Block a user