baremetal: volume driver refactoring and tests

Refactor the LibVirtVolumeDriver to test each step of
attaching and detaching a volume. And add tests.

Closes-Bug: 1240470
Change-Id: I51906bec6ba5d6af71997df277bb368cff81b3b3
This commit is contained in:
Arata Notsu
2013-10-09 20:48:26 +09:00
parent 97425310a9
commit 41aa9a8f56
2 changed files with 114 additions and 21 deletions

View File

@@ -17,6 +17,7 @@
from oslo.config import cfg from oslo.config import cfg
from nova import exception
from nova import test from nova import test
from nova.virt.baremetal import volume_driver from nova.virt.baremetal import volume_driver
from nova.virt import fake from nova.virt import fake
@@ -184,8 +185,10 @@ class BareMetalLibVirtVolumeDriverTestCase(test.TestCase):
} }
self.connection_info = {'driver_volume_type': 'fake'} self.connection_info = {'driver_volume_type': 'fake'}
self.mount_point = '/dev/vdc' self.mount_point = '/dev/vdc'
self.mount_device = 'vdc'
self.source_path = '/dev/sdx' self.source_path = '/dev/sdx'
self.instance = {'name': 'instance-00000001'} self.instance = {'uuid': '12345678-1234-1234-1234-123467890123456',
'name': 'instance-00000001'}
self.fixed_ips = [{'address': '10.2.3.4'}, self.fixed_ips = [{'address': '10.2.3.4'},
{'address': '172.16.17.18'}, {'address': '172.16.17.18'},
] ]
@@ -205,18 +208,44 @@ class BareMetalLibVirtVolumeDriverTestCase(test.TestCase):
self.connection_info, self.connection_info,
self.disk_info) self.disk_info)
def test_attach_volume(self): def test_volume_driver_method_ok(self):
self.mox.StubOutWithMock(volume_driver, '_get_fixed_ips') fake_driver = self.driver.volume_drivers['fake']
self.mox.StubOutWithMock(fake_driver, 'connect_volume')
fake_driver.connect_volume(self.connection_info, self.disk_info)
self.mox.ReplayAll()
self.driver._volume_driver_method('connect_volume',
self.connection_info,
self.disk_info)
def test_volume_driver_method_driver_type_not_found(self):
self.connection_info['driver_volume_type'] = 'qwerty'
self.assertRaises(exception.VolumeDriverNotFound,
self.driver._volume_driver_method,
'connect_volume',
self.connection_info,
self.disk_info)
def test_connect_volume(self):
self.mox.StubOutWithMock(self.driver, '_volume_driver_method') self.mox.StubOutWithMock(self.driver, '_volume_driver_method')
self.driver._volume_driver_method('connect_volume',
self.connection_info,
self.disk_info)
self.mox.ReplayAll()
self.driver._connect_volume(self.connection_info, self.disk_info)
def test_disconnect_volume(self):
self.mox.StubOutWithMock(self.driver, '_volume_driver_method')
self.driver._volume_driver_method('disconnect_volume',
self.connection_info,
self.mount_device)
self.mox.ReplayAll()
self.driver._disconnect_volume(self.connection_info, self.mount_device)
def test_publish_iscsi(self):
self.mox.StubOutWithMock(volume_driver, '_get_iqn') self.mox.StubOutWithMock(volume_driver, '_get_iqn')
self.mox.StubOutWithMock(volume_driver, '_get_next_tid') self.mox.StubOutWithMock(volume_driver, '_get_next_tid')
self.mox.StubOutWithMock(volume_driver, '_create_iscsi_export_tgtadm') self.mox.StubOutWithMock(volume_driver, '_create_iscsi_export_tgtadm')
self.mox.StubOutWithMock(volume_driver, '_allow_iscsi_tgtadm') self.mox.StubOutWithMock(volume_driver, '_allow_iscsi_tgtadm')
volume_driver._get_fixed_ips(self.instance).AndReturn(self.fixed_ips)
self.driver._volume_driver_method('connect_volume',
self.connection_info,
self.disk_info).\
AndReturn(FakeConf(self.source_path))
volume_driver._get_iqn(self.instance['name'], self.mount_point).\ volume_driver._get_iqn(self.instance['name'], self.mount_point).\
AndReturn(self.iqn) AndReturn(self.iqn)
volume_driver._get_next_tid().AndReturn(self.tid) volume_driver._get_next_tid().AndReturn(self.tid)
@@ -228,6 +257,57 @@ class BareMetalLibVirtVolumeDriverTestCase(test.TestCase):
volume_driver._allow_iscsi_tgtadm(self.tid, volume_driver._allow_iscsi_tgtadm(self.tid,
self.fixed_ips[1]['address']) self.fixed_ips[1]['address'])
self.mox.ReplayAll() self.mox.ReplayAll()
self.driver._publish_iscsi(self.instance,
self.mount_point,
self.fixed_ips,
self.source_path)
def test_depublish_iscsi_ok(self):
self.mox.StubOutWithMock(volume_driver, '_get_iqn')
self.mox.StubOutWithMock(volume_driver, '_find_tid')
self.mox.StubOutWithMock(volume_driver, '_delete_iscsi_export_tgtadm')
volume_driver._get_iqn(self.instance['name'], self.mount_point).\
AndReturn(self.iqn)
volume_driver._find_tid(self.iqn).AndReturn(self.tid)
volume_driver._delete_iscsi_export_tgtadm(self.tid)
self.mox.ReplayAll()
self.driver._depublish_iscsi(self.instance, self.mount_point)
def test_depublish_iscsi_do_nothing_if_tid_is_not_found(self):
self.mox.StubOutWithMock(volume_driver, '_get_iqn')
self.mox.StubOutWithMock(volume_driver, '_find_tid')
volume_driver._get_iqn(self.instance['name'], self.mount_point).\
AndReturn(self.iqn)
volume_driver._find_tid(self.iqn).AndReturn(None)
self.mox.ReplayAll()
self.driver._depublish_iscsi(self.instance, self.mount_point)
def test_attach_volume(self):
self.mox.StubOutWithMock(volume_driver, '_get_fixed_ips')
self.mox.StubOutWithMock(self.driver, '_connect_volume')
self.mox.StubOutWithMock(self.driver, '_publish_iscsi')
volume_driver._get_fixed_ips(self.instance).AndReturn(self.fixed_ips)
self.driver._connect_volume(self.connection_info, self.disk_info).\
AndReturn(FakeConf(self.source_path))
self.driver._publish_iscsi(self.instance, self.mount_point,
self.fixed_ips, self.source_path)
self.mox.ReplayAll()
self.driver.attach_volume(self.connection_info, self.driver.attach_volume(self.connection_info,
self.instance, self.instance,
self.mount_point) self.mount_point)
def test_detach_volume(self):
self.mox.StubOutWithMock(volume_driver, '_get_iqn')
self.mox.StubOutWithMock(volume_driver, '_find_tid')
self.mox.StubOutWithMock(volume_driver, '_delete_iscsi_export_tgtadm')
self.mox.StubOutWithMock(self.driver, '_disconnect_volume')
volume_driver._get_iqn(self.instance['name'], self.mount_point).\
AndReturn(self.iqn)
volume_driver._find_tid(self.iqn).AndReturn(self.tid)
volume_driver._delete_iscsi_export_tgtadm(self.tid)
self.driver._disconnect_volume(self.connection_info,
self.mount_device)
self.mox.ReplayAll()
self.driver.detach_volume(self.connection_info,
self.instance,
self.mount_point)

View File

@@ -192,8 +192,8 @@ class VolumeDriver(object):
if not self._initiator: if not self._initiator:
self._initiator = libvirt_utils.get_iscsi_initiator() self._initiator = libvirt_utils.get_iscsi_initiator()
if not self._initiator: if not self._initiator:
LOG.warn(_('Could not determine iscsi initiator name ' LOG.warn(_('Could not determine iscsi initiator name'),
'for instance %s') % instance) instance=instance)
return { return {
'ip': CONF.my_ip, 'ip': CONF.my_ip,
'initiator': self._initiator, 'initiator': self._initiator,
@@ -240,10 +240,16 @@ class LibvirtVolumeDriver(VolumeDriver):
'bus': 'baremetal', 'bus': 'baremetal',
'type': 'baremetal', 'type': 'baremetal',
} }
conf = self._volume_driver_method('connect_volume', conf = self._connect_volume(connection_info, disk_info)
self._publish_iscsi(instance, mountpoint, fixed_ips,
conf.source_path)
def _connect_volume(self, connection_info, disk_info):
return self._volume_driver_method('connect_volume',
connection_info, connection_info,
disk_info) disk_info)
device_path = conf.source_path
def _publish_iscsi(self, instance, mountpoint, fixed_ips, device_path):
iqn = _get_iqn(instance['name'], mountpoint) iqn = _get_iqn(instance['name'], mountpoint)
tid = _get_next_tid() tid = _get_next_tid()
_create_iscsi_export_tgtadm(device_path, tid, iqn) _create_iscsi_export_tgtadm(device_path, tid, iqn)
@@ -264,16 +270,23 @@ class LibvirtVolumeDriver(VolumeDriver):
def detach_volume(self, connection_info, instance, mountpoint): def detach_volume(self, connection_info, instance, mountpoint):
mount_device = mountpoint.rpartition("/")[2] mount_device = mountpoint.rpartition("/")[2]
try: try:
iqn = _get_iqn(instance['name'], mountpoint) self._depublish_iscsi(instance, mountpoint)
tid = _find_tid(iqn)
if tid is not None:
_delete_iscsi_export_tgtadm(tid)
else:
LOG.warn(_('detach volume could not find tid for %s') % iqn)
finally: finally:
self._volume_driver_method('disconnect_volume', self._disconnect_volume(connection_info, mount_device)
connection_info,
mount_device) def _disconnect_volume(self, connection_info, disk_dev):
return self._volume_driver_method('disconnect_volume',
connection_info,
disk_dev)
def _depublish_iscsi(self, instance, mountpoint):
iqn = _get_iqn(instance['name'], mountpoint)
tid = _find_tid(iqn)
if tid is not None:
_delete_iscsi_export_tgtadm(tid)
else:
LOG.warn(_('detach volume could not find tid for %s'), iqn,
instance=instance)
def get_all_block_devices(self): def get_all_block_devices(self):
""" """