NEC driver: Disallow access from the source node after live-migration.

NEC driver deletes access permission from nodes in the method
remove_export().
However, the method remove_export() is not called after live-migration.
After live-migration, access permission of the live-migration source
node is not deleted.

This patch adds the step to delete access permission from the node
in the method terminate_connection().
The terminate_connection() is called after live-migration and
access permission of the source node is deleted.

Change-Id: Id605eb6ff61cd1d382bf82173e6d65b986dda398
Closes-Bug: #1753375
This commit is contained in:
Shunei Shiono 2018-03-08 15:25:04 +09:00
parent e271cd549d
commit fcf09dedba
3 changed files with 99 additions and 2 deletions

View File

@ -877,6 +877,33 @@ class ExportTest(volume_helper.MStorageDSVDriver, test.TestCase):
self.assertEqual('127.0.0.1:3260', info['data']['target_portals'][0]) self.assertEqual('127.0.0.1:3260', info['data']['target_portals'][0])
self.assertEqual(88, info['data']['target_luns'][0]) self.assertEqual(88, info['data']['target_luns'][0])
@mock.patch('cinder.volume.drivers.nec.cli.MStorageISMCLI._execute',
patch_execute)
@mock.patch('cinder.volume.drivers.nec.cli.MStorageISMCLI.view_all',
patch_view_all)
def test_iscsi_terminate_connection(self):
self.vol.id = "46045673-41e7-44a7-9333-02f07feab04b"
connector = {'initiator': "iqn.1994-05.com.redhat:d1d8e8f23255",
'multipath': True}
ret = self._iscsi_terminate_connection(self.vol, connector)
self.assertIsNone(ret)
@mock.patch('cinder.volume.drivers.nec.cli.MStorageISMCLI._execute',
patch_execute)
@mock.patch('cinder.volume.drivers.nec.cli.MStorageISMCLI.view_all',
patch_view_all)
def test_iscsi_terminate_connection_negative(self):
self.vol.id = "46045673-41e7-44a7-9333-02f07feab04b"
connector = {'initiator': "iqn.1994-05.com.redhat:d1d8e8f23255",
'multipath': True}
with self.assertRaisesRegexp(exception.VolumeBackendAPIException,
r'Failed to unregister Logical Disk from'
r' Logical Disk Set \(iSM31064\)'):
mock_del = mock.Mock()
self._cli.delldsetld = mock_del
self._cli.delldsetld.return_value = False, 'iSM31064'
self._iscsi_terminate_connection(self.vol, connector)
@mock.patch('cinder.volume.drivers.nec.cli.MStorageISMCLI._execute', @mock.patch('cinder.volume.drivers.nec.cli.MStorageISMCLI._execute',
patch_execute) patch_execute)
@mock.patch('cinder.volume.drivers.nec.cli.MStorageISMCLI.view_all', @mock.patch('cinder.volume.drivers.nec.cli.MStorageISMCLI.view_all',
@ -915,6 +942,13 @@ class ExportTest(volume_helper.MStorageDSVDriver, test.TestCase):
self.assertEqual( self.assertEqual(
'2A00000991020012', '2A00000991020012',
info['data']['initiator_target_map']['10000090FAA0786B'][3]) info['data']['initiator_target_map']['10000090FAA0786B'][3])
with self.assertRaisesRegexp(exception.VolumeBackendAPIException,
r'Failed to unregister Logical Disk from'
r' Logical Disk Set \(iSM31064\)'):
mock_del = mock.Mock()
self._cli.delldsetld = mock_del
self._cli.delldsetld.return_value = False, 'iSM31064'
self._fc_terminate_connection(self.vol, connector)
@mock.patch('cinder.volume.drivers.nec.cli.MStorageISMCLI._execute', @mock.patch('cinder.volume.drivers.nec.cli.MStorageISMCLI._execute',
patch_execute) patch_execute)

View File

@ -153,7 +153,7 @@ def convert_to_id(value62):
class MStorageVolumeCommon(object): class MStorageVolumeCommon(object):
"""M-Series Storage volume common class.""" """M-Series Storage volume common class."""
VERSION = '1.10.1' VERSION = '1.10.2'
WIKI_NAME = 'NEC_Cinder_CI' WIKI_NAME = 'NEC_Cinder_CI'
def do_setup(self, context): def do_setup(self, context):

View File

@ -1180,7 +1180,50 @@ class MStorageDriver(volume_common.MStorageVolumeCommon):
def iscsi_terminate_connection(self, volume, connector): def iscsi_terminate_connection(self, volume, connector):
msgparm = ('Volume ID = %(id)s, Connector = %(connector)s' msgparm = ('Volume ID = %(id)s, Connector = %(connector)s'
% {'id': volume.id, 'connector': connector}) % {'id': volume.id, 'connector': connector})
try:
self._iscsi_terminate_connection(volume, connector)
LOG.info('Terminated iSCSI Connection (%s)', msgparm) LOG.info('Terminated iSCSI Connection (%s)', msgparm)
except exception.CinderException as e:
with excutils.save_and_reraise_exception():
LOG.warning('Failed to Terminate iSCSI Connection '
'(%(msgparm)s) (%(exception)s)',
{'msgparm': msgparm, 'exception': e})
def _iscsi_terminate_connection(self, volume, connector):
if self._properties['ldset_name'] != '':
LOG.debug('Ldset is specified. Access control setting '
'is not deleted automatically.')
return
if connector is None:
LOG.debug('Connector is not specified. Nothing to do.')
return
# delete unused access control setting.
xml = self._cli.view_all(self._properties['ismview_path'])
pools, lds, ldsets, used_ldns, hostports, max_ld_count = (
self.configs(xml))
ldname = self.get_ldname(
volume.id, self._properties['ld_name_format'])
if ldname not in lds:
LOG.debug('Logical Disk `%s` has unbound already.', ldname)
return
ldset = self._validate_iscsildset_exist(ldsets, connector)
retnum, errnum = self._cli.delldsetld(ldset['ldsetname'], ldname)
if retnum is not True:
if 'iSM31065' in errnum:
LOG.debug('LD `%(ld)s` already deleted '
'from LD Set `%(ldset)s`?',
{'ld': ldname, 'ldset': ldset['ldsetname']})
else:
msg = (_('Failed to unregister Logical Disk from '
'Logical Disk Set (%s)') % errnum)
raise exception.VolumeBackendAPIException(data=msg)
LOG.debug('LD `%(ld)s` deleted from LD Set `%(ldset)s`.',
{'ld': ldname, 'ldset': ldset['ldsetname']})
def iscsi_terminate_connection_snapshot(self, snapshot, connector, def iscsi_terminate_connection_snapshot(self, snapshot, connector,
**kwargs): **kwargs):
@ -1342,6 +1385,26 @@ class MStorageDriver(volume_common.MStorageVolumeCommon):
info['data'] = {'target_wwn': target_wwns, info['data'] = {'target_wwn': target_wwns,
'initiator_target_map': init_targ_map} 'initiator_target_map': init_targ_map}
if connector is not None and self._properties['ldset_name'] == '':
# delete LD from LD set.
ldname = self.get_ldname(
volume.id, self._properties['ld_name_format'])
if ldname not in lds:
LOG.debug('Logical Disk `%s` has unbound already.', ldname)
return info
ldset = self._validate_fcldset_exist(ldsets, connector)
retnum, errnum = self._cli.delldsetld(ldset['ldsetname'], ldname)
if retnum is not True:
if 'iSM31065' in errnum:
LOG.debug('LD `%(ld)s` already deleted '
'from LD Set `%(ldset)s`?',
{'ld': ldname, 'ldset': ldset['ldsetname']})
else:
msg = (_('Failed to unregister Logical Disk from '
'Logical Disk Set (%s)') % errnum)
raise exception.VolumeBackendAPIException(data=msg)
LOG.debug('_fc_terminate_connection' LOG.debug('_fc_terminate_connection'
'(Volume ID = %(id)s, connector = %(connector)s, ' '(Volume ID = %(id)s, connector = %(connector)s, '
'info = %(info)s) End.', 'info = %(info)s) End.',