[Unity] Add support of removing empty host

Add an option in Unity Cinder driver to customize the removing
of empty host. The option is named `remove_empty_host`. Its
default value is False. When it is set to True, the host will
be removed from Unity after the last LUN is detached from it.

Change-Id: I2d9fad2c977a61f5b26a6449a8e509911eff28e0
Closes-bug: #1768711
This commit is contained in:
Ryan Liang 2018-05-02 10:30:20 +08:00
parent c93371dc2a
commit f9a9aa5a25
8 changed files with 146 additions and 22 deletions

View File

@ -80,3 +80,7 @@ class UnexpectedLunDeletion(Exception):
class AdapterSetupError(Exception): class AdapterSetupError(Exception):
pass pass
class HostDeleteIsCalled(Exception):
pass

View File

@ -45,6 +45,7 @@ class MockConfig(object):
self.san_password = 'pass' self.san_password = 'pass'
self.driver_ssl_cert_verify = True self.driver_ssl_cert_verify = True
self.driver_ssl_cert_path = None self.driver_ssl_cert_path = None
self.remove_empty_host = False
def safe_get(self, name): def safe_get(self, name):
return getattr(self, name) return getattr(self, name)
@ -70,6 +71,7 @@ class MockDriver(object):
class MockClient(object): class MockClient(object):
def __init__(self): def __init__(self):
self._system = test_client.MockSystem() self._system = test_client.MockSystem()
self.host = '10.10.10.10' # fake unity IP
@staticmethod @staticmethod
def get_pools(): def get_pools():
@ -130,6 +132,15 @@ class MockClient(object):
def create_host(name): def create_host(name):
return test_client.MockResource(name=name) return test_client.MockResource(name=name)
@staticmethod
def create_host_wo_lock(name):
return test_client.MockResource(name=name)
@staticmethod
def delete_host_wo_lock(host):
if host.name == 'empty-host':
raise ex.HostDeleteIsCalled()
@staticmethod @staticmethod
def attach(host, lun_or_snap): def attach(host, lun_or_snap):
return 10 return 10
@ -467,6 +478,16 @@ class CommonAdapterTest(test.TestCase):
self.assertRaises(ex.DetachIsCalled, f) self.assertRaises(ex.DetachIsCalled, f)
def test_terminate_connection_remove_empty_host(self):
self.adapter.remove_empty_host = True
def f():
connector = {'host': 'empty-host'}
vol = MockOSResource(provider_location='id^lun_45', id='id_45')
self.adapter.terminate_connection(vol, connector)
self.assertRaises(ex.HostDeleteIsCalled, f)
def test_manage_existing_by_name(self): def test_manage_existing_by_name(self):
ref = {'source-id': 12} ref = {'source-id': 12}
volume = MockOSResource(name='lun1') volume = MockOSResource(name='lun1')
@ -808,6 +829,20 @@ class FCAdapterTest(test.TestCase):
target_wwn = ['100000051e55a100', '100000051e55a121'] target_wwn = ['100000051e55a100', '100000051e55a121']
self.assertListEqual(target_wwn, data['target_wwn']) self.assertListEqual(target_wwn, data['target_wwn'])
def test_terminate_connection_remove_empty_host_return_data(self):
self.adapter.remove_empty_host = True
connector = {'host': 'empty-host-return-data', 'wwpns': 'abcdefg'}
volume = MockOSResource(provider_location='id^lun_41', id='id_41')
ret = self.adapter.terminate_connection(volume, connector)
self.assertEqual('fibre_channel', ret['driver_volume_type'])
data = ret['data']
target_map = {
'200000051e55a100': ('100000051e55a100', '100000051e55a121'),
'200000051e55a121': ('100000051e55a100', '100000051e55a121')}
self.assertDictEqual(target_map, data['initiator_target_map'])
target_wwn = ['100000051e55a100', '100000051e55a121']
self.assertListEqual(target_wwn, data['target_wwn'])
def test_validate_ports_whitelist_none(self): def test_validate_ports_whitelist_none(self):
ports = self.adapter.validate_ports(None) ports = self.adapter.validate_ports(None)
self.assertEqual(set(('spa_iom_0_fc0', 'spa_iom_0_fc1')), set(ports)) self.assertEqual(set(('spa_iom_0_fc0', 'spa_iom_0_fc1')), set(ports))

View File

@ -63,6 +63,8 @@ class MockResource(object):
raise ex.UnityResourceNotFoundError() raise ex.UnityResourceNotFoundError()
elif self.get_id() == 'snap_in_use': elif self.get_id() == 'snap_in_use':
raise ex.UnityDeleteAttachedSnapError() raise ex.UnityDeleteAttachedSnapError()
elif self.name == 'empty_host':
raise ex.HostDeleteIsCalled()
@property @property
def pool(self): def pool(self):
@ -545,3 +547,15 @@ class ClientTest(unittest.TestCase):
def test_restore_snapshot(self): def test_restore_snapshot(self):
back_snap = self.client.restore_snapshot('snap1') back_snap = self.client.restore_snapshot('snap1')
self.assertEqual("internal_snap", back_snap.name) self.assertEqual("internal_snap", back_snap.name)
def test_delete_host_wo_lock(self):
host = MockResource(name='empty-host')
self.client.host_cache['empty-host'] = host
self.assertRaises(ex.HostDeleteIsCalled,
self.client.delete_host_wo_lock(host))
def test_delete_host_wo_lock_remove_from_cache(self):
host = MockResource(name='empty-host-in-cache')
self.client.host_cache['empty-host-in-cache'] = host
self.client.delete_host_wo_lock(host)
self.assertNotIn(host.name, self.client.host_cache)

View File

@ -139,6 +139,8 @@ class CommonAdapter(object):
self.storage_pools_map = None self.storage_pools_map = None
self._client = None self._client = None
self.allowed_ports = None self.allowed_ports = None
self.remove_empty_host = False
self.to_lock_host = False
def do_setup(self, driver, conf): def do_setup(self, driver, conf):
self.driver = driver self.driver = driver
@ -166,6 +168,9 @@ class CommonAdapter(object):
self.allowed_ports = self.validate_ports(self.config.unity_io_ports) self.allowed_ports = self.validate_ports(self.config.unity_io_ports)
self.remove_empty_host = self.config.remove_empty_host
self.to_lock_host = self.remove_empty_host
group_name = (self.config.config_group if self.config.config_group group_name = (self.config.config_group if self.config.config_group
else 'DEFAULT') else 'DEFAULT')
folder_name = '%(group)s.%(sys_name)s' % { folder_name = '%(group)s.%(sys_name)s' % {
@ -291,11 +296,25 @@ class CommonAdapter(object):
else: else:
self.client.delete_lun(lun_id) self.client.delete_lun(lun_id)
def _create_host_and_attach(self, host_name, lun_or_snap):
@utils.lock_if(self.to_lock_host, '{lock_name}')
def _lock_helper(lock_name):
if not self.to_lock_host:
host = self.client.create_host(host_name)
else:
# Use the lock in the decorator
host = self.client.create_host_wo_lock(host_name)
hlu = self.client.attach(host, lun_or_snap)
return host, hlu
return _lock_helper('{unity}-{host}'.format(unity=self.client.host,
host=host_name))
def _initialize_connection(self, lun_or_snap, connector, vol_id): def _initialize_connection(self, lun_or_snap, connector, vol_id):
host = self.client.create_host(connector['host']) host, hlu = self._create_host_and_attach(connector['host'],
lun_or_snap)
self.client.update_host_initiators( self.client.update_host_initiators(
host, self.get_connector_uids(connector)) host, self.get_connector_uids(connector))
hlu = self.client.attach(host, lun_or_snap)
data = self.get_connection_info(hlu, host, connector) data = self.get_connection_info(hlu, host, connector)
data['target_discovered'] = True data['target_discovered'] = True
if vol_id is not None: if vol_id is not None:
@ -311,13 +330,44 @@ class CommonAdapter(object):
lun = self.client.get_lun(lun_id=self.get_lun_id(volume)) lun = self.client.get_lun(lun_id=self.get_lun_id(volume))
return self._initialize_connection(lun, connector, volume.id) return self._initialize_connection(lun, connector, volume.id)
@staticmethod
def filter_targets_by_host(host):
# No target info for iSCSI driver
return []
def _detach_and_delete_host(self, host_name, lun_or_snap):
@utils.lock_if(self.to_lock_host, '{lock_name}')
def _lock_helper(lock_name):
# Only get the host from cache here
host = self.client.create_host_wo_lock(host_name)
self.client.detach(host, lun_or_snap)
host.update() # need update to get the latest `host_luns`
targets = self.filter_targets_by_host(host)
if self.remove_empty_host and not host.host_luns:
self.client.delete_host_wo_lock(host)
return targets
return _lock_helper('{unity}-{host}'.format(unity=self.client.host,
host=host_name))
@staticmethod
def get_terminate_connection_info(connector, targets):
# No return data from terminate_connection for iSCSI driver
return {}
def _terminate_connection(self, lun_or_snap, connector): def _terminate_connection(self, lun_or_snap, connector):
is_force_detach = connector is None is_force_detach = connector is None
data = {}
if is_force_detach: if is_force_detach:
self.client.detach_all(lun_or_snap) self.client.detach_all(lun_or_snap)
else: else:
host = self.client.create_host(connector['host']) targets = self._detach_and_delete_host(connector['host'],
self.client.detach(host, lun_or_snap) lun_or_snap)
data = self.get_terminate_connection_info(connector, targets)
return {
'driver_volume_type': self.driver_volume_type,
'data': data,
}
@cinder_utils.trace @cinder_utils.trace
def terminate_connection(self, volume, connector): def terminate_connection(self, volume, connector):
@ -742,24 +792,19 @@ class FCAdapter(CommonAdapter):
data['target_lun'] = hlu data['target_lun'] = hlu
return data return data
def _terminate_connection(self, lun_or_snap, connector): def filter_targets_by_host(self, host):
if self.auto_zone_enabled and not host.host_luns:
return self.client.get_fc_target_info(
host=host, logged_in_only=True,
allowed_ports=self.allowed_ports)
return []
def get_terminate_connection_info(self, connector, targets):
# For FC, terminate_connection needs to return data to zone manager # For FC, terminate_connection needs to return data to zone manager
# which would clean the zone based on the data. # which would clean the zone based on the data.
super(FCAdapter, self)._terminate_connection(lun_or_snap, connector) if targets:
return self._get_fc_zone_info(connector['wwpns'], targets)
ret = None return {}
if self.auto_zone_enabled:
ret = {
'driver_volume_type': self.driver_volume_type,
'data': {}
}
host = self.client.create_host(connector['host'])
if not host.host_luns:
targets = self.client.get_fc_target_info(
logged_in_only=True, allowed_ports=self.allowed_ports)
ret['data'] = self._get_fc_zone_info(connector['wwpns'],
targets)
return ret
def _get_fc_zone_info(self, initiator_wwns, target_wwns): def _get_fc_zone_info(self, initiator_wwns, target_wwns):
mapping = self.lookup_service.get_device_mapping_from_network( mapping = self.lookup_service.get_device_mapping_from_network(

View File

@ -188,6 +188,9 @@ class UnityClient(object):
@coordination.synchronized('{self.host}-{name}') @coordination.synchronized('{self.host}-{name}')
def create_host(self, name): def create_host(self, name):
return self.create_host_wo_lock(name)
def create_host_wo_lock(self, name):
"""Provides existing host if exists else create one.""" """Provides existing host if exists else create one."""
if name not in self.host_cache: if name not in self.host_cache:
try: try:
@ -202,6 +205,10 @@ class UnityClient(object):
host = self.host_cache[name] host = self.host_cache[name]
return host return host
def delete_host_wo_lock(self, host):
host.delete()
del self.host_cache[host.name]
def update_host_initiators(self, host, uids): def update_host_initiators(self, host, uids):
"""Updates host with the supplied uids.""" """Updates host with the supplied uids."""
host_initiators_ids = self.get_host_initiator_ids(host) host_initiators_ids = self.get_host_initiator_ids(host)

View File

@ -37,7 +37,11 @@ UNITY_OPTS = [
cfg.ListOpt('unity_io_ports', cfg.ListOpt('unity_io_ports',
default=None, default=None,
help='A comma-separated list of iSCSI or FC ports to be used. ' help='A comma-separated list of iSCSI or FC ports to be used. '
'Each port can be Unix-style glob expressions.')] 'Each port can be Unix-style glob expressions.'),
cfg.BoolOpt('remove_empty_host',
default=False,
help='To remove the host from Unity when the last LUN is '
'detached from it. By default, it is False.')]
CONF.register_opts(UNITY_OPTS, group=configuration.SHARED_CONF_GROUP) CONF.register_opts(UNITY_OPTS, group=configuration.SHARED_CONF_GROUP)
@ -53,9 +57,10 @@ class UnityDriver(driver.ManageableVD,
2.0.0 - Add thin clone support 2.0.0 - Add thin clone support
3.0.0 - Add IPv6 support 3.0.0 - Add IPv6 support
3.1.0 - Support revert to snapshot API 3.1.0 - Support revert to snapshot API
4.0.0 - Support remove empty host
""" """
VERSION = '03.01.00' VERSION = '04.00.00'
VENDOR = 'Dell EMC' VENDOR = 'Dell EMC'
# ThirdPartySystems wiki page # ThirdPartySystems wiki page
CI_WIKI_NAME = "EMC_UNITY_CI" CI_WIKI_NAME = "EMC_UNITY_CI"

View File

@ -23,6 +23,7 @@ from oslo_utils import fnmatch
from oslo_utils import units from oslo_utils import units
import six import six
from cinder import coordination
from cinder import exception from cinder import exception
from cinder.i18n import _ from cinder.i18n import _
from cinder.volume import utils as vol_utils from cinder.volume import utils as vol_utils
@ -295,3 +296,10 @@ def match_any(full, patterns):
def is_before_4_1(ver): def is_before_4_1(ver):
return version.LooseVersion(ver) < version.LooseVersion('4.1') return version.LooseVersion(ver) < version.LooseVersion('4.1')
def lock_if(condition, lock_name):
if condition:
return coordination.synchronized(lock_name)
else:
return functools.partial

View File

@ -0,0 +1,6 @@
---
features:
- |
Dell EMC Unity Driver: Adds support for removing empty host. The new option
named `remove_empty_host` could be configured as `True` to notify Unity
driver to remove the host after the last LUN is detached from it.