Fix hostname & initiator for volume drivers
This is a simple fix to make sure the iSCSI initiator is always returned and the host name is generally consistent between FC and iSCSI. Change-Id: I584b4ff4ea99b47a451e87c51c0c6ab4d0d28a0f Closes-Bug: 1658740
This commit is contained in:
parent
672bf31a06
commit
cec1c996e6
@ -181,17 +181,20 @@ class TestPowerVMDriver(test.TestCase):
|
|||||||
self.assertTrue(
|
self.assertTrue(
|
||||||
self.drv.session.get_event_listener.return_value.shutdown.called)
|
self.drv.session.get_event_listener.return_value.shutdown.called)
|
||||||
|
|
||||||
def test_get_volume_connector(self):
|
@mock.patch('nova_powervm.virt.powervm.volume.get_iscsi_initiator')
|
||||||
|
def test_get_volume_connector(self, mock_initiator):
|
||||||
"""Tests that a volume connector can be built."""
|
"""Tests that a volume connector can be built."""
|
||||||
|
mock_initiator.return_value = 'iscsi_initiator'
|
||||||
|
|
||||||
self.flags(volume_adapter='fibre_channel', group='powervm')
|
self.flags(volume_adapter='fibre_channel', group='powervm')
|
||||||
vol_connector = self.drv.get_volume_connector(mock.Mock())
|
vol_connector = self.drv.get_volume_connector(mock.Mock())
|
||||||
self.assertIsNotNone(vol_connector['wwpns'])
|
self.assertIsNotNone(vol_connector['wwpns'])
|
||||||
self.assertIsNotNone(vol_connector['host'])
|
self.assertIsNotNone(vol_connector['host'])
|
||||||
|
self.assertEqual('iscsi_initiator', vol_connector['initiator'])
|
||||||
|
|
||||||
self.flags(volume_adapter='iscsi', group='powervm')
|
self.flags(volume_adapter='iscsi', group='powervm')
|
||||||
vol_connector = self.drv.get_volume_connector(mock.Mock())
|
vol_connector = self.drv.get_volume_connector(mock.Mock())
|
||||||
self.assertIsNotNone(vol_connector['initiator'])
|
self.assertEqual('iscsi_initiator', vol_connector['initiator'])
|
||||||
|
|
||||||
def test_setup_disk_adapter(self):
|
def test_setup_disk_adapter(self):
|
||||||
# Ensure we can handle upper case option and we instantiate the class
|
# Ensure we can handle upper case option and we instantiate the class
|
||||||
|
@ -15,9 +15,12 @@
|
|||||||
# under the License.
|
# under the License.
|
||||||
|
|
||||||
import mock
|
import mock
|
||||||
|
from pypowervm.wrappers import virtual_io_server as pvm_vios
|
||||||
|
|
||||||
from nova import test
|
from nova import test
|
||||||
|
|
||||||
|
from nova_powervm.virt.powervm import volume
|
||||||
|
|
||||||
|
|
||||||
class TestVolumeAdapter(test.TestCase):
|
class TestVolumeAdapter(test.TestCase):
|
||||||
|
|
||||||
@ -32,3 +35,29 @@ class TestVolumeAdapter(test.TestCase):
|
|||||||
self.mock_inst_wrap = mock.MagicMock()
|
self.mock_inst_wrap = mock.MagicMock()
|
||||||
self.mock_inst_wrap.can_modify_io.return_value = (True, None)
|
self.mock_inst_wrap.can_modify_io.return_value = (True, None)
|
||||||
self.mock_get_inst_wrap.return_value = self.mock_inst_wrap
|
self.mock_get_inst_wrap.return_value = self.mock_inst_wrap
|
||||||
|
|
||||||
|
|
||||||
|
class TestInitMethods(test.TestCase):
|
||||||
|
|
||||||
|
@mock.patch('pypowervm.tasks.hdisk.discover_iscsi_initiator')
|
||||||
|
@mock.patch('pypowervm.tasks.partition.get_mgmt_partition')
|
||||||
|
def test_get_iscsi_initiator(self, mock_mgmt, mock_iscsi_init):
|
||||||
|
# Set up mocks and clear out data that may have been set by other
|
||||||
|
# tests
|
||||||
|
mock_adpt = mock.Mock()
|
||||||
|
mock_mgmt.return_value = mock.Mock(spec=pvm_vios.VIOS)
|
||||||
|
mock_iscsi_init.return_value = 'test_initiator'
|
||||||
|
|
||||||
|
self.assertEqual('test_initiator',
|
||||||
|
volume.get_iscsi_initiator(mock_adpt))
|
||||||
|
|
||||||
|
# Make sure it gets set properly in the backend
|
||||||
|
self.assertEqual('test_initiator', volume._ISCSI_INITIATOR)
|
||||||
|
self.assertTrue(volume._ISCSI_LOOKUP_COMPLETE)
|
||||||
|
mock_mgmt.assert_called_once_with(mock_adpt)
|
||||||
|
self.assertEqual(1, mock_mgmt.call_count)
|
||||||
|
|
||||||
|
# Invoke again, make sure it doesn't call down to the mgmt part again
|
||||||
|
self.assertEqual('test_initiator',
|
||||||
|
volume.get_iscsi_initiator(mock_adpt))
|
||||||
|
self.assertEqual(1, mock_mgmt.call_count)
|
||||||
|
@ -1129,11 +1129,9 @@ class PowerVMDriver(driver.ComputeDriver):
|
|||||||
# The host ID
|
# The host ID
|
||||||
connector = {'host': CONF.host}
|
connector = {'host': CONF.host}
|
||||||
|
|
||||||
# The WWPNs in case of FC connection.
|
# Get the contents from the volume driver
|
||||||
vol_drv = self._get_inst_vol_adpt(ctx.get_admin_context(),
|
vol_drv = self._get_inst_vol_adpt(ctx.get_admin_context(),
|
||||||
instance)
|
instance)
|
||||||
|
|
||||||
# The WWPNs in case of FC connection.
|
|
||||||
if vol_drv is not None:
|
if vol_drv is not None:
|
||||||
|
|
||||||
if CONF.powervm.volume_adapter.lower() == "fibre_channel":
|
if CONF.powervm.volume_adapter.lower() == "fibre_channel":
|
||||||
@ -1142,7 +1140,8 @@ class PowerVMDriver(driver.ComputeDriver):
|
|||||||
if wwpn_list is not None:
|
if wwpn_list is not None:
|
||||||
connector["wwpns"] = wwpn_list
|
connector["wwpns"] = wwpn_list
|
||||||
connector['host'] = vol_drv.host_name()
|
connector['host'] = vol_drv.host_name()
|
||||||
connector['initiator'] = vol_drv.host_name()
|
connector['initiator'] = vol_attach.get_iscsi_initiator(
|
||||||
|
self.adapter)
|
||||||
return connector
|
return connector
|
||||||
|
|
||||||
def migrate_disk_and_power_off(self, context, instance, dest,
|
def migrate_disk_and_power_off(self, context, instance, dest,
|
||||||
|
@ -14,6 +14,11 @@
|
|||||||
# License for the specific language governing permissions and limitations
|
# License for the specific language governing permissions and limitations
|
||||||
# under the License.
|
# under the License.
|
||||||
|
|
||||||
|
from oslo_concurrency import lockutils
|
||||||
|
from pypowervm.tasks import hdisk
|
||||||
|
from pypowervm.tasks import partition
|
||||||
|
from pypowervm.wrappers import virtual_io_server as pvm_vios
|
||||||
|
|
||||||
# Defines the various volume connectors that can be used.
|
# Defines the various volume connectors that can be used.
|
||||||
from nova_powervm import conf as cfg
|
from nova_powervm import conf as cfg
|
||||||
|
|
||||||
@ -26,3 +31,27 @@ FC_STRATEGY_MAPPING = {
|
|||||||
NETWORK_STRATEGY_MAPPING = {
|
NETWORK_STRATEGY_MAPPING = {
|
||||||
'iscsi': 'nova_powervm.virt.powervm.volume.iscsi.IscsiVolumeAdapter'
|
'iscsi': 'nova_powervm.virt.powervm.volume.iscsi.IscsiVolumeAdapter'
|
||||||
}
|
}
|
||||||
|
|
||||||
|
_ISCSI_INITIATOR = None
|
||||||
|
_ISCSI_LOOKUP_COMPLETE = False
|
||||||
|
|
||||||
|
|
||||||
|
@lockutils.synchronized("PowerVM_iSCSI_Initiator_Lookup")
|
||||||
|
def get_iscsi_initiator(adapter):
|
||||||
|
"""Gets the iSCSI initiator.
|
||||||
|
|
||||||
|
This is looked up once at process start up. Stored in memory thereafter.
|
||||||
|
|
||||||
|
:param adapter: The pypowervm adapter.
|
||||||
|
:return: The initiator name. If the NovaLink is not capable of supporting
|
||||||
|
iSCSI, None will be returned.
|
||||||
|
"""
|
||||||
|
global _ISCSI_INITIATOR, _ISCSI_LOOKUP_COMPLETE
|
||||||
|
if not _ISCSI_LOOKUP_COMPLETE:
|
||||||
|
mgmt_w = partition.get_mgmt_partition(adapter)
|
||||||
|
if isinstance(mgmt_w, pvm_vios.VIOS):
|
||||||
|
_ISCSI_INITIATOR = hdisk.discover_iscsi_initiator(
|
||||||
|
adapter, mgmt_w.uuid)
|
||||||
|
|
||||||
|
_ISCSI_LOOKUP_COMPLETE = True
|
||||||
|
return _ISCSI_INITIATOR
|
||||||
|
@ -26,7 +26,6 @@ from nova_powervm.virt.powervm.volume import driver as v_driver
|
|||||||
from nova_powervm.virt.powervm.volume import volume as volume
|
from nova_powervm.virt.powervm.volume import volume as volume
|
||||||
from pypowervm import const as pvm_const
|
from pypowervm import const as pvm_const
|
||||||
from pypowervm.tasks import hdisk
|
from pypowervm.tasks import hdisk
|
||||||
from pypowervm.tasks import partition
|
|
||||||
from pypowervm.utils import transaction as tx
|
from pypowervm.utils import transaction as tx
|
||||||
from pypowervm.wrappers import virtual_io_server as pvm_vios
|
from pypowervm.wrappers import virtual_io_server as pvm_vios
|
||||||
|
|
||||||
@ -52,19 +51,13 @@ class IscsiVolumeAdapter(volume.VscsiVolumeAdapter,
|
|||||||
super(IscsiVolumeAdapter, self).__init__(
|
super(IscsiVolumeAdapter, self).__init__(
|
||||||
adapter, host_uuid, instance, connection_info, stg_ftsk=stg_ftsk)
|
adapter, host_uuid, instance, connection_info, stg_ftsk=stg_ftsk)
|
||||||
|
|
||||||
# iSCSI volumes are assumed to be on the novalink partition
|
|
||||||
mgmt_part = partition.get_mgmt_partition(adapter)
|
|
||||||
vios_uuid = mgmt_part.uuid
|
|
||||||
self.initiator_name = hdisk.discover_iscsi_initiator(
|
|
||||||
adapter, vios_uuid)
|
|
||||||
|
|
||||||
@classmethod
|
@classmethod
|
||||||
def vol_type(cls):
|
def vol_type(cls):
|
||||||
"""The type of volume supported by this type."""
|
"""The type of volume supported by this type."""
|
||||||
return 'iscsi'
|
return 'iscsi'
|
||||||
|
|
||||||
def host_name(self):
|
def host_name(self):
|
||||||
return self.initiator_name
|
return CONF.host
|
||||||
|
|
||||||
@classmethod
|
@classmethod
|
||||||
def min_xags(cls):
|
def min_xags(cls):
|
||||||
|
Loading…
x
Reference in New Issue
Block a user