Merge "[Unity] Add support of removing empty host"
This commit is contained in:
commit
d8c97b6f9c
@ -80,3 +80,7 @@ class UnexpectedLunDeletion(Exception):
|
|||||||
|
|
||||||
class AdapterSetupError(Exception):
|
class AdapterSetupError(Exception):
|
||||||
pass
|
pass
|
||||||
|
|
||||||
|
|
||||||
|
class HostDeleteIsCalled(Exception):
|
||||||
|
pass
|
||||||
|
@ -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))
|
||||||
|
@ -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)
|
||||||
|
@ -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(
|
||||||
|
@ -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)
|
||||||
|
@ -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"
|
||||||
|
@ -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
|
||||||
|
@ -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.
|
Loading…
x
Reference in New Issue
Block a user