a31e641d73
Change-Id: Iab3df26827e725eacf28302c76e45d29a579b983
367 lines
14 KiB
Python
367 lines
14 KiB
Python
# (c) Copyright 2014 Brocade Communications Systems Inc.
|
|
# All Rights Reserved.
|
|
#
|
|
# Copyright 2014 OpenStack Foundation
|
|
#
|
|
# Licensed under the Apache License, Version 2.0 (the "License"); you may
|
|
# not use this file except in compliance with the License. You may obtain
|
|
# a copy of the License at
|
|
#
|
|
# http://www.apache.org/licenses/LICENSE-2.0
|
|
#
|
|
# Unless required by applicable law or agreed to in writing, software
|
|
# distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
|
|
# WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
|
|
# License for the specific language governing permissions and limitations
|
|
# under the License.
|
|
#
|
|
"""
|
|
ZoneManager is responsible to manage access control using FC zoning
|
|
when zoning mode is set as 'fabric'.
|
|
ZoneManager provides interfaces to add connection and remove connection
|
|
for given initiator and target list associated with a FC volume attach and
|
|
detach operation.
|
|
|
|
**Related Flags**
|
|
|
|
:zone_driver: Used by:class:`ZoneManager`.
|
|
Defaults to
|
|
`cinder.zonemanager.drivers.brocade.brcd_fc_zone_driver.BrcdFCZoneDriver`
|
|
:zoning_policy: Used by: class: 'ZoneManager'. Defaults to 'none'
|
|
|
|
"""
|
|
|
|
from oslo_config import cfg
|
|
from oslo_log import log as logging
|
|
from oslo_utils import importutils
|
|
|
|
from cinder import exception
|
|
from cinder.i18n import _
|
|
from cinder.volume import configuration as config
|
|
from cinder.zonemanager import fc_common
|
|
import cinder.zonemanager.fczm_constants as zone_constant
|
|
|
|
|
|
LOG = logging.getLogger(__name__)
|
|
|
|
zone_manager_opts = [
|
|
cfg.StrOpt('zone_driver',
|
|
default='cinder.zonemanager.drivers.brocade.brcd_fc_zone_driver'
|
|
'.BrcdFCZoneDriver',
|
|
help='FC Zone Driver responsible for zone management'),
|
|
cfg.StrOpt('zoning_policy',
|
|
default='initiator-target',
|
|
help='Zoning policy configured by user; valid values include '
|
|
'"initiator-target" or "initiator"'),
|
|
cfg.StrOpt('fc_fabric_names',
|
|
help='Comma separated list of Fibre Channel fabric names.'
|
|
' This list of names is used to retrieve other SAN credentials'
|
|
' for connecting to each SAN fabric'),
|
|
cfg.StrOpt('fc_san_lookup_service',
|
|
default='cinder.zonemanager.drivers.brocade'
|
|
'.brcd_fc_san_lookup_service.BrcdFCSanLookupService',
|
|
help='FC SAN Lookup Service'),
|
|
cfg.BoolOpt('enable_unsupported_driver',
|
|
default=False,
|
|
help="Set this to True when you want to allow an unsupported "
|
|
"zone manager driver to start. Drivers that haven't "
|
|
"maintained a working CI system and testing are marked "
|
|
"as unsupported until CI is working again. This also "
|
|
"marks a driver as deprecated and may be removed in the "
|
|
"next release."),
|
|
|
|
]
|
|
|
|
CONF = cfg.CONF
|
|
CONF.register_opts(zone_manager_opts, group='fc-zone-manager')
|
|
|
|
|
|
class ZoneManager(fc_common.FCCommon):
|
|
|
|
"""Manages Connection control during attach/detach.
|
|
|
|
Version History:
|
|
1.0 - Initial version
|
|
1.0.1 - Added __new__ for singleton
|
|
1.0.2 - Added friendly zone name
|
|
|
|
"""
|
|
|
|
VERSION = "1.0.2"
|
|
driver = None
|
|
_initialized = False
|
|
fabric_names = []
|
|
|
|
def __new__(class_, *args, **kwargs):
|
|
if not hasattr(class_, "_instance"):
|
|
class_._instance = object.__new__(class_)
|
|
return class_._instance
|
|
|
|
def __init__(self, **kwargs):
|
|
"""Load the driver from the one specified in args, or from flags."""
|
|
super(ZoneManager, self).__init__(**kwargs)
|
|
|
|
self.configuration = config.Configuration(zone_manager_opts,
|
|
'fc-zone-manager')
|
|
self.set_initialized(False)
|
|
self._build_driver()
|
|
|
|
def _build_driver(self):
|
|
zone_driver = self.configuration.zone_driver
|
|
LOG.debug("Zone driver from config: %(driver)s",
|
|
{'driver': zone_driver})
|
|
|
|
zm_config = config.Configuration(zone_manager_opts, 'fc-zone-manager')
|
|
# Initialize vendor specific implementation of FCZoneDriver
|
|
self.driver = importutils.import_object(
|
|
zone_driver,
|
|
configuration=zm_config)
|
|
|
|
if not self.driver.supported:
|
|
self._log_unsupported_driver_warning()
|
|
|
|
if not self.configuration.enable_unsupported_driver:
|
|
LOG.error("Unsupported drivers are disabled."
|
|
" You can re-enable by adding "
|
|
"enable_unsupported_driver=True to the "
|
|
"fc-zone-manager section in cinder.conf",
|
|
resource={'type': 'zone_manager',
|
|
'id': self.__class__.__name__})
|
|
return
|
|
|
|
self.set_initialized(True)
|
|
|
|
@property
|
|
def initialized(self):
|
|
return self._initialized
|
|
|
|
def set_initialized(self, value=True):
|
|
self._initialized = value
|
|
|
|
def _require_initialized(self):
|
|
"""Verifies that the zone manager has been properly initialized."""
|
|
if not self.initialized:
|
|
LOG.error("Fibre Channel Zone Manager is not initialized.")
|
|
raise exception.ZoneManagerNotInitialized()
|
|
else:
|
|
self._log_unsupported_driver_warning()
|
|
|
|
def _log_unsupported_driver_warning(self):
|
|
"""Annoy the log about unsupported fczm drivers."""
|
|
if not self.driver.supported:
|
|
LOG.warning("Zone Manager driver (%(driver_name)s %(version)s)"
|
|
" is currently unsupported and may be removed in "
|
|
"the next release of OpenStack. Use at your own "
|
|
"risk.",
|
|
{'driver_name': self.driver.__class__.__name__,
|
|
'version': self.driver.get_version()},
|
|
resource={'type': 'zone_manager',
|
|
'id': self.driver.__class__.__name__})
|
|
|
|
def get_zoning_state_ref_count(self, initiator_wwn, target_wwn):
|
|
"""Zone management state check.
|
|
|
|
Performs state check for given I-T pair to return the current count of
|
|
active attach for the pair.
|
|
"""
|
|
# TODO(sk): ref count state management
|
|
count = 0
|
|
# check the state for I-T pair
|
|
return count
|
|
|
|
def add_connection(self, conn_info):
|
|
"""Add connection control.
|
|
|
|
Adds connection control for the given initiator target map.
|
|
initiator_target_map - each initiator WWN mapped to a list of one
|
|
or more target WWN:
|
|
|
|
.. code-block:: python
|
|
|
|
e.g.:
|
|
{
|
|
'10008c7cff523b01': ['20240002ac000a50', '20240002ac000a40']
|
|
}
|
|
"""
|
|
connected_fabric = None
|
|
host_name = None
|
|
storage_system = None
|
|
|
|
try:
|
|
# Make sure the driver is loaded and we are initialized
|
|
self._log_unsupported_driver_warning()
|
|
self._require_initialized()
|
|
except exception.ZoneManagerNotInitialized:
|
|
LOG.error("Cannot add Fibre Channel Zone because the "
|
|
"Zone Manager is not initialized properly.",
|
|
resource={'type': 'zone_manager',
|
|
'id': self.__class__.__name__})
|
|
return
|
|
|
|
try:
|
|
initiator_target_map = (
|
|
conn_info[zone_constant.DATA][zone_constant.IT_MAP])
|
|
|
|
if zone_constant.HOST in conn_info[zone_constant.DATA]:
|
|
host_name = conn_info[
|
|
zone_constant.DATA][
|
|
zone_constant.HOST].replace(" ", "_")
|
|
|
|
if zone_constant.STORAGE in conn_info[zone_constant.DATA]:
|
|
storage_system = (
|
|
conn_info[
|
|
zone_constant.DATA][
|
|
zone_constant.STORAGE].replace(" ", "_"))
|
|
|
|
for initiator in initiator_target_map.keys():
|
|
target_list = initiator_target_map[initiator]
|
|
LOG.debug("Target list : %(targets)s",
|
|
{'targets': target_list})
|
|
|
|
# get SAN context for the target list
|
|
fabric_map = self.get_san_context(target_list)
|
|
LOG.debug("Fabric map after context lookup: %(fabricmap)s",
|
|
{'fabricmap': fabric_map})
|
|
# iterate over each SAN and apply connection control
|
|
for fabric in fabric_map.keys():
|
|
connected_fabric = fabric
|
|
t_list = fabric_map[fabric]
|
|
# get valid I-T map to add connection control
|
|
i_t_map = {initiator: t_list}
|
|
valid_i_t_map = self.get_valid_initiator_target_map(
|
|
i_t_map, True)
|
|
LOG.info("Final filtered map for fabric: %(i_t_map)s",
|
|
{'i_t_map': valid_i_t_map})
|
|
|
|
# Call driver to add connection control
|
|
self.driver.add_connection(fabric, valid_i_t_map,
|
|
host_name, storage_system)
|
|
|
|
LOG.info("Add connection: finished iterating "
|
|
"over all target list")
|
|
except Exception as e:
|
|
msg = _("Failed adding connection for fabric=%(fabric)s: "
|
|
"Error: %(err)s") % {'fabric': connected_fabric,
|
|
'err': str(e)}
|
|
LOG.error(msg)
|
|
raise exception.ZoneManagerException(reason=msg)
|
|
|
|
def delete_connection(self, conn_info):
|
|
"""Delete connection.
|
|
|
|
Updates/deletes connection control for the given initiator target map.
|
|
initiator_target_map - each initiator WWN mapped to a list of one
|
|
or more target WWN:
|
|
|
|
.. code-block:: python
|
|
|
|
e.g.:
|
|
{
|
|
'10008c7cff523b01': ['20240002ac000a50', '20240002ac000a40']
|
|
}
|
|
"""
|
|
connected_fabric = None
|
|
host_name = None
|
|
storage_system = None
|
|
|
|
try:
|
|
# Make sure the driver is loaded and we are initialized
|
|
self._log_unsupported_driver_warning()
|
|
self._require_initialized()
|
|
except exception.ZoneManagerNotInitialized:
|
|
LOG.error("Cannot delete fibre channel zone because the "
|
|
"Zone Manager is not initialized properly.",
|
|
resource={'type': 'zone_manager',
|
|
'id': self.__class__.__name__})
|
|
return
|
|
|
|
try:
|
|
initiator_target_map = (
|
|
conn_info[zone_constant.DATA][zone_constant.IT_MAP])
|
|
|
|
if zone_constant.HOST in conn_info[zone_constant.DATA]:
|
|
host_name = conn_info[zone_constant.DATA][zone_constant.HOST]
|
|
|
|
if zone_constant.STORAGE in conn_info[zone_constant.DATA]:
|
|
storage_system = (
|
|
conn_info[
|
|
zone_constant.DATA][
|
|
zone_constant.STORAGE].replace(" ", "_"))
|
|
|
|
for initiator in initiator_target_map.keys():
|
|
target_list = initiator_target_map[initiator]
|
|
LOG.info("Delete connection target list: %(targets)s",
|
|
{'targets': target_list})
|
|
|
|
# get SAN context for the target list
|
|
fabric_map = self.get_san_context(target_list)
|
|
LOG.debug("Delete connection fabric map from SAN "
|
|
"context: %(fabricmap)s", {'fabricmap': fabric_map})
|
|
|
|
# iterate over each SAN and apply connection control
|
|
for fabric in fabric_map.keys():
|
|
connected_fabric = fabric
|
|
t_list = fabric_map[fabric]
|
|
# get valid I-T map to add connection control
|
|
i_t_map = {initiator: t_list}
|
|
valid_i_t_map = self.get_valid_initiator_target_map(
|
|
i_t_map, False)
|
|
LOG.info("Final filtered map for delete connection: "
|
|
"%(i_t_map)s", {'i_t_map': valid_i_t_map})
|
|
|
|
# Call driver to delete connection control
|
|
if len(valid_i_t_map) > 0:
|
|
self.driver.delete_connection(fabric,
|
|
valid_i_t_map,
|
|
host_name,
|
|
storage_system)
|
|
|
|
LOG.debug("Delete connection - finished iterating over all"
|
|
" target list")
|
|
except Exception as e:
|
|
msg = _("Failed removing connection for fabric=%(fabric)s: "
|
|
"Error: %(err)s") % {'fabric': connected_fabric,
|
|
'err': str(e)}
|
|
LOG.error(msg)
|
|
raise exception.ZoneManagerException(reason=msg)
|
|
|
|
def get_san_context(self, target_wwn_list):
|
|
"""SAN lookup for end devices.
|
|
|
|
Look up each SAN configured and return a map of SAN (fabric IP)
|
|
to list of target WWNs visible to the fabric.
|
|
"""
|
|
fabric_map = self.driver.get_san_context(target_wwn_list)
|
|
LOG.debug("Got SAN context: %(fabricmap)s", {'fabricmap': fabric_map})
|
|
return fabric_map
|
|
|
|
def get_valid_initiator_target_map(self, initiator_target_map,
|
|
add_control):
|
|
"""Reference count check for end devices.
|
|
|
|
Looks up the reference count for each initiator-target pair from the
|
|
map and returns a filtered list based on the operation type
|
|
add_control - operation type can be true for add connection control
|
|
and false for remove connection control
|
|
"""
|
|
filtered_i_t_map = {}
|
|
for initiator in initiator_target_map.keys():
|
|
t_list = initiator_target_map[initiator]
|
|
for target in t_list:
|
|
count = self.get_zoning_state_ref_count(initiator, target)
|
|
if add_control:
|
|
if count > 0:
|
|
t_list.remove(target)
|
|
# update count = count + 1
|
|
else:
|
|
if count > 1:
|
|
t_list.remove(target)
|
|
# update count = count - 1
|
|
if t_list:
|
|
filtered_i_t_map[initiator] = t_list
|
|
else:
|
|
LOG.info("No targets to add or remove connection for "
|
|
"initiator: %(init_wwn)s",
|
|
{'init_wwn': initiator})
|
|
return filtered_i_t_map
|