Merge "Add Pure Storage NVMe-RoCE driver"

This commit is contained in:
Zuul 2022-07-29 20:10:36 +00:00 committed by Gerrit Code Review
commit c12c690271
5 changed files with 691 additions and 20 deletions

View File

@ -39,6 +39,7 @@ from cinder.volume import volume_utils
def fake_retry(exceptions, interval=1, retries=3, backoff_rate=2): def fake_retry(exceptions, interval=1, retries=3, backoff_rate=2):
def _decorator(f): def _decorator(f):
return f return f
return _decorator return _decorator
@ -54,6 +55,7 @@ DRIVER_PATH = "cinder.volume.drivers.pure"
BASE_DRIVER_OBJ = DRIVER_PATH + ".PureBaseVolumeDriver" BASE_DRIVER_OBJ = DRIVER_PATH + ".PureBaseVolumeDriver"
ISCSI_DRIVER_OBJ = DRIVER_PATH + ".PureISCSIDriver" ISCSI_DRIVER_OBJ = DRIVER_PATH + ".PureISCSIDriver"
FC_DRIVER_OBJ = DRIVER_PATH + ".PureFCDriver" FC_DRIVER_OBJ = DRIVER_PATH + ".PureFCDriver"
NVME_DRIVER_OBJ = DRIVER_PATH + ".PureNVMEDriver"
ARRAY_OBJ = DRIVER_PATH + ".FlashArray" ARRAY_OBJ = DRIVER_PATH + ".FlashArray"
UNMANAGED_SUFFIX = "-unmanaged" UNMANAGED_SUFFIX = "-unmanaged"
@ -78,7 +80,21 @@ PRIMARY_MANAGEMENT_IP = GET_ARRAY_PRIMARY["array_name"]
API_TOKEN = "12345678-abcd-1234-abcd-1234567890ab" API_TOKEN = "12345678-abcd-1234-abcd-1234567890ab"
VOLUME_BACKEND_NAME = "Pure_iSCSI" VOLUME_BACKEND_NAME = "Pure_iSCSI"
ISCSI_PORT_NAMES = ["ct0.eth2", "ct0.eth3", "ct1.eth2", "ct1.eth3"] ISCSI_PORT_NAMES = ["ct0.eth2", "ct0.eth3", "ct1.eth2", "ct1.eth3"]
NVME_PORT_NAMES = ["ct0.eth8", "ct0.eth9", "ct1.eth8", "ct1.eth9"]
FC_PORT_NAMES = ["ct0.fc2", "ct0.fc3", "ct1.fc2", "ct1.fc3"] FC_PORT_NAMES = ["ct0.fc2", "ct0.fc3", "ct1.fc2", "ct1.fc3"]
# These two IP blocks should use the same prefix (see NVME_CIDR_FILTERED to
# make sure changes make sense). Our arrays now have 4 IPv4 + 4 IPv6 ports.
NVME_IPS = ["10.0.0." + str(i + 1) for i in range(len(NVME_PORT_NAMES))]
NVME_IPS += ["[2001:db8::" + str(i + 1) + "]"
for i in range(len(NVME_PORT_NAMES))]
NVME_CIDR = "0.0.0.0/0"
NVME_CIDR_V6 = "::/0"
NVME_PORT = 4420
# Designed to filter out only one of the AC NVMe IPs, leaving the rest in
NVME_CIDR_FILTERED = "10.0.0.0/29"
# Include several IP / networks: 10.0.0.2, 10.0.0.3, 10.0.0.6, 10.0.0.7
NVME_CIDRS_FILTERED = ["10.0.0.2", "10.0.0.3", "2001:db8::1:2/127"]
# These two IP blocks should use the same prefix (see ISCSI_CIDR_FILTERED to # These two IP blocks should use the same prefix (see ISCSI_CIDR_FILTERED to
# make sure changes make sense). Our arrays now have 4 IPv4 + 4 IPv6 ports. # make sure changes make sense). Our arrays now have 4 IPv4 + 4 IPv6 ports.
ISCSI_IPS = ["10.0.0." + str(i + 1) for i in range(len(ISCSI_PORT_NAMES))] ISCSI_IPS = ["10.0.0." + str(i + 1) for i in range(len(ISCSI_PORT_NAMES))]
@ -102,17 +118,24 @@ PURE_HOST_NAME = pure.PureBaseVolumeDriver._generate_purity_host_name(HOSTNAME)
PURE_HOST = { PURE_HOST = {
"name": PURE_HOST_NAME, "name": PURE_HOST_NAME,
"hgroup": None, "hgroup": None,
"nqn": [],
"iqn": [], "iqn": [],
"wwn": [], "wwn": [],
} }
INITIATOR_NQN = (
"nqn.2014-08.org.nvmexpress:uuid:6953a373-c3f7-4ea8-ae77-105c393012ff"
)
INITIATOR_IQN = "iqn.1993-08.org.debian:01:222" INITIATOR_IQN = "iqn.1993-08.org.debian:01:222"
INITIATOR_WWN = "5001500150015081abc" INITIATOR_WWN = "5001500150015081abc"
NVME_CONNECTOR = {"nqn": INITIATOR_NQN, "host": HOSTNAME}
ISCSI_CONNECTOR = {"initiator": INITIATOR_IQN, "host": HOSTNAME} ISCSI_CONNECTOR = {"initiator": INITIATOR_IQN, "host": HOSTNAME}
FC_CONNECTOR = {"wwpns": {INITIATOR_WWN}, "host": HOSTNAME} FC_CONNECTOR = {"wwpns": {INITIATOR_WWN}, "host": HOSTNAME}
TARGET_NQN = "nqn.2010-06.com.purestorage:flasharray.12345abc"
TARGET_IQN = "iqn.2010-06.com.purestorage:flasharray.12345abc" TARGET_IQN = "iqn.2010-06.com.purestorage:flasharray.12345abc"
AC_TARGET_IQN = "iqn.2018-06.com.purestorage:flasharray.67890def" AC_TARGET_IQN = "iqn.2018-06.com.purestorage:flasharray.67890def"
TARGET_WWN = "21000024ff59fe94" TARGET_WWN = "21000024ff59fe94"
TARGET_PORT = "3260" TARGET_PORT = "3260"
TARGET_ROCE_PORT = "4420"
INITIATOR_TARGET_MAP = { INITIATOR_TARGET_MAP = {
# _build_initiator_target_map() calls list(set()) on the list, # _build_initiator_target_map() calls list(set()) on the list,
# we must also call list(set()) to get the exact same order # we must also call list(set()) to get the exact same order
@ -137,24 +160,34 @@ AC_DEVICE_MAPPING = {
} }
# We now have IPv6 in addition to IPv4 on each interface # We now have IPv6 in addition to IPv4 on each interface
NVME_PORTS = [{"name": name,
"nqn": TARGET_NQN,
"iqn": None,
"portal": ip + ":" + TARGET_ROCE_PORT,
"wwn": None,
} for name, ip in zip(NVME_PORT_NAMES * 2, NVME_IPS)]
ISCSI_PORTS = [{"name": name, ISCSI_PORTS = [{"name": name,
"iqn": TARGET_IQN, "iqn": TARGET_IQN,
"portal": ip + ":" + TARGET_PORT, "portal": ip + ":" + TARGET_PORT,
"nqn": None,
"wwn": None, "wwn": None,
} for name, ip in zip(ISCSI_PORT_NAMES * 2, ISCSI_IPS)] } for name, ip in zip(ISCSI_PORT_NAMES * 2, ISCSI_IPS)]
AC_ISCSI_PORTS = [{"name": name, AC_ISCSI_PORTS = [{"name": name,
"iqn": AC_TARGET_IQN, "iqn": AC_TARGET_IQN,
"portal": ip + ":" + TARGET_PORT, "portal": ip + ":" + TARGET_PORT,
"nqn": None,
"wwn": None, "wwn": None,
} for name, ip in zip(ISCSI_PORT_NAMES * 2, AC_ISCSI_IPS)] } for name, ip in zip(ISCSI_PORT_NAMES * 2, AC_ISCSI_IPS)]
FC_PORTS = [{"name": name, FC_PORTS = [{"name": name,
"iqn": None, "iqn": None,
"nqn": None,
"portal": None, "portal": None,
"wwn": wwn, "wwn": wwn,
"nqn": None, "nqn": None,
} for name, wwn in zip(FC_PORT_NAMES, FC_WWNS)] } for name, wwn in zip(FC_PORT_NAMES, FC_WWNS)]
AC_FC_PORTS = [{"name": name, AC_FC_PORTS = [{"name": name,
"iqn": None, "iqn": None,
"nqn": None,
"portal": None, "portal": None,
"wwn": wwn, "wwn": wwn,
"nqn": None, "nqn": None,
@ -162,9 +195,11 @@ AC_FC_PORTS = [{"name": name,
NON_ISCSI_PORT = { NON_ISCSI_PORT = {
"name": "ct0.fc1", "name": "ct0.fc1",
"iqn": None, "iqn": None,
"nqn": None,
"portal": None, "portal": None,
"wwn": "5001500150015081", "wwn": "5001500150015081",
} }
NVME_PORTS_WITH = NVME_PORTS + [NON_ISCSI_PORT]
PORTS_WITH = ISCSI_PORTS + [NON_ISCSI_PORT] PORTS_WITH = ISCSI_PORTS + [NON_ISCSI_PORT]
PORTS_WITHOUT = [NON_ISCSI_PORT] PORTS_WITHOUT = [NON_ISCSI_PORT]
TOTAL_CAPACITY = 50.0 TOTAL_CAPACITY = 50.0
@ -281,6 +316,31 @@ ISCSI_CONNECTION_INFO_AC_FILTERED_LIST = {
}, },
} }
NVME_CONNECTION_INFO = {
"driver_volume_type": "nvmeof",
"data": {
"target_nqn": TARGET_NQN,
"discard": True,
"portals": [(NVME_IPS[0], NVME_PORT, "rdma"),
(NVME_IPS[1], NVME_PORT, "rdma"),
(NVME_IPS[2], NVME_PORT, "rdma"),
(NVME_IPS[3], NVME_PORT, "rdma")],
"volume_nguid": "0009714b5cb916324a9374c470002b2c8",
},
}
NVME_CONNECTION_INFO_V6 = {
"driver_volume_type": "nvmeof",
"data": {
"target_nqn": TARGET_NQN,
"discard": True,
"portals": [(NVME_IPS[4].strip("[]"), NVME_PORT, "rdma"),
(NVME_IPS[5].strip("[]"), NVME_PORT, "rdma"),
(NVME_IPS[6].strip("[]"), NVME_PORT, "rdma"),
(NVME_IPS[7].strip("[]"), NVME_PORT, "rdma")],
"volume_nguid": "0009714b5cb916324a9374c470002b2c8",
},
}
FC_CONNECTION_INFO = { FC_CONNECTION_INFO = {
"driver_volume_type": "fibre_channel", "driver_volume_type": "fibre_channel",
"data": { "data": {
@ -573,6 +633,9 @@ class PureDriverTestCase(test.TestCase):
self.mock_config.driver_ssl_cert_path = None self.mock_config.driver_ssl_cert_path = None
self.mock_config.pure_iscsi_cidr = ISCSI_CIDR self.mock_config.pure_iscsi_cidr = ISCSI_CIDR
self.mock_config.pure_iscsi_cidr_list = None self.mock_config.pure_iscsi_cidr_list = None
self.mock_config.pure_nvme_cidr = NVME_CIDR
self.mock_config.pure_nvme_cidr_list = None
self.mock_config.pure_nvme_transport = "roce"
self.array = mock.Mock() self.array = mock.Mock()
self.array.get.return_value = GET_ARRAY_PRIMARY self.array.get.return_value = GET_ARRAY_PRIMARY
self.array.array_name = GET_ARRAY_PRIMARY["array_name"] self.array.array_name = GET_ARRAY_PRIMARY["array_name"]
@ -802,6 +865,7 @@ class PureBaseVolumeDriverTestCase(PureBaseSharedDriverTestCase):
mock_target.get.return_value = GET_ARRAY_PRIMARY mock_target.get.return_value = GET_ARRAY_PRIMARY
self.purestorage_module.FlashArray.return_value = mock_target self.purestorage_module.FlashArray.return_value = mock_target
self.driver._storage_protocol = 'iSCSI'
self.driver.parse_replication_configs() self.driver.parse_replication_configs()
self.assertEqual(1, len(self.driver._replication_target_arrays)) self.assertEqual(1, len(self.driver._replication_target_arrays))
self.assertEqual(mock_target, self.assertEqual(mock_target,
@ -838,6 +902,7 @@ class PureBaseVolumeDriverTestCase(PureBaseSharedDriverTestCase):
mock_target.get.return_value = GET_ARRAY_PRIMARY mock_target.get.return_value = GET_ARRAY_PRIMARY
self.purestorage_module.FlashArray.return_value = mock_target self.purestorage_module.FlashArray.return_value = mock_target
self.driver._storage_protocol = 'iSCSI'
self.driver.parse_replication_configs() self.driver.parse_replication_configs()
self.assertEqual(1, len(self.driver._replication_target_arrays)) self.assertEqual(1, len(self.driver._replication_target_arrays))
self.assertEqual(mock_target, self.assertEqual(mock_target,
@ -863,6 +928,7 @@ class PureBaseVolumeDriverTestCase(PureBaseSharedDriverTestCase):
self.array.get.return_value = GET_ARRAY_PRIMARY self.array.get.return_value = GET_ARRAY_PRIMARY
self.purestorage_module.FlashArray.side_effect = [self.array, self.purestorage_module.FlashArray.side_effect = [self.array,
self.async_array2] self.async_array2]
self.driver._storage_protocol = 'iSCSI'
self.driver.do_setup(None) self.driver.do_setup(None)
self.assertEqual(self.array, self.driver._array) self.assertEqual(self.array, self.driver._array)
self.assertEqual(1, len(self.driver._replication_target_arrays)) self.assertEqual(1, len(self.driver._replication_target_arrays))
@ -899,6 +965,8 @@ class PureBaseVolumeDriverTestCase(PureBaseSharedDriverTestCase):
self.array.get.return_value = GET_ARRAY_PRIMARY self.array.get.return_value = GET_ARRAY_PRIMARY
self.purestorage_module.FlashArray.side_effect = [self.array, self.purestorage_module.FlashArray.side_effect = [self.array,
mock_sync_target] mock_sync_target]
self.driver.configuration.pure_nvme_transport = "roce"
self.driver._storage_protocol = 'iSCSI'
self.driver.do_setup(None) self.driver.do_setup(None)
self.assertEqual(self.array, self.driver._array) self.assertEqual(self.array, self.driver._array)
@ -910,6 +978,37 @@ class PureBaseVolumeDriverTestCase(PureBaseSharedDriverTestCase):
mock.call(self.array, [mock_sync_target], 'cinder-pod') mock.call(self.array, [mock_sync_target], 'cinder-pod')
]) ])
@mock.patch(BASE_DRIVER_OBJ + '._setup_replicated_pods')
@mock.patch(BASE_DRIVER_OBJ + '._generate_replication_retention')
@mock.patch(BASE_DRIVER_OBJ + '._setup_replicated_pgroups')
def test_do_setup_replicated_sync_rep_bad_driver(
self,
mock_setup_repl_pgroups,
mock_generate_replication_retention,
mock_setup_pods):
retention = mock.MagicMock()
mock_generate_replication_retention.return_value = retention
self._setup_mocks_for_replication()
self.mock_config.safe_get.return_value = [
{
"backend_id": "foo",
"managed_backend_name": None,
"san_ip": "1.2.3.4",
"api_token": "abc123",
"type": "sync",
}
]
mock_sync_target = mock.MagicMock()
mock_sync_target.get.return_value = GET_ARRAY_SECONDARY
self.array.get.return_value = GET_ARRAY_PRIMARY
self.driver._storage_protocol = 'NVMe-RoCE'
self.purestorage_module.FlashArray.side_effect = [self.array,
mock_sync_target]
self.assertRaises(pure.PureDriverException,
self.driver.do_setup,
None)
def test_update_provider_info_update_all(self): def test_update_provider_info_update_all(self):
test_vols = [ test_vols = [
self.new_fake_vol(spec={'id': fake.VOLUME_ID}, self.new_fake_vol(spec={'id': fake.VOLUME_ID},
@ -3276,6 +3375,7 @@ class PureISCSIDriverTestCase(PureBaseSharedDriverTestCase):
self.mock_config.use_chap_auth = False self.mock_config.use_chap_auth = False
self.driver = pure.PureISCSIDriver(configuration=self.mock_config) self.driver = pure.PureISCSIDriver(configuration=self.mock_config)
self.driver._array = self.array self.driver._array = self.array
self.driver._storage_protocol = 'iSCSI'
self.mock_utils = mock.Mock() self.mock_utils = mock.Mock()
self.driver.driver_utils = self.mock_utils self.driver.driver_utils = self.mock_utils
@ -3490,6 +3590,7 @@ class PureISCSIDriverTestCase(PureBaseSharedDriverTestCase):
mock_get_chap_creds, mock_get_chap_creds,
mock_get_wwn): mock_get_wwn):
vol, vol_name = self.new_fake_vol() vol, vol_name = self.new_fake_vol()
self.maxDiff = None
auth_type = "CHAP" auth_type = "CHAP"
chap_username = ISCSI_CONNECTOR["host"] chap_username = ISCSI_CONNECTOR["host"]
chap_password = "password" chap_password = "password"
@ -3774,6 +3875,7 @@ class PureFCDriverTestCase(PureBaseSharedDriverTestCase):
def setUp(self): def setUp(self):
super(PureFCDriverTestCase, self).setUp() super(PureFCDriverTestCase, self).setUp()
self.driver = pure.PureFCDriver(configuration=self.mock_config) self.driver = pure.PureFCDriver(configuration=self.mock_config)
self.driver._storage_protocol = "FC"
self.driver._array = self.array self.driver._array = self.array
self.driver._lookup_service = mock.Mock() self.driver._lookup_service = mock.Mock()
@ -4308,3 +4410,341 @@ class PureVolumeGroupsTestCase(PureBaseSharedDriverTestCase):
group_snapshot, group_snapshot,
snapshots snapshots
) )
class PureNVMEDriverTestCase(PureBaseSharedDriverTestCase):
def setUp(self):
super(PureNVMEDriverTestCase, self).setUp()
self.driver = pure.PureNVMEDriver(configuration=self.mock_config)
self.driver._array = self.array
self.driver._storage_protocol = 'NVMe-RoCE'
self.mock_utils = mock.Mock()
self.driver.transport_type = "rdma"
self.driver.driver_utils = self.mock_utils
def test_get_host(self):
good_host = PURE_HOST.copy()
good_host.update(nqn=["another-wrong-nqn", INITIATOR_NQN])
bad_host = {'name': "bad-host", "nqn": ["wrong-nqn"]}
self.array.list_hosts.return_value = [bad_host]
real_result = self.driver._get_host(self.array, NVME_CONNECTOR)
self.assertEqual([], real_result)
self.array.list_hosts.return_value.append(good_host)
real_result = self.driver._get_host(self.array, NVME_CONNECTOR)
self.assertEqual([good_host], real_result)
self.assert_error_propagates(
[self.array.list_hosts],
self.driver._get_host,
self.array,
NVME_CONNECTOR,
)
def test_get_nguid(self):
vol = {'created': '2019-01-28T14:16:54Z',
'name': 'volume-fdc9892f-5af0-47c8-9d4a-5167ac29dc98-cinder',
'serial': '9714B5CB91634C470002B2C8',
'size': 3221225472,
'source': 'volume-a366b1ba-ec27-4ca3-9051-c301b75bc778-cinder'}
self.array.get_volume.return_value = vol
returned_nguid = self.driver._get_nguid(vol['name'])
expected_nguid = '009714b5cb91634c24a937470002b2c8'
self.assertEqual(expected_nguid, returned_nguid)
@mock.patch(NVME_DRIVER_OBJ + "._get_nguid")
@mock.patch(NVME_DRIVER_OBJ + "._get_wwn")
@mock.patch(NVME_DRIVER_OBJ + "._connect")
@mock.patch(NVME_DRIVER_OBJ + "._get_target_nvme_ports")
def test_initialize_connection(
self, mock_get_nvme_ports, mock_connection, mock_get_wwn,
mock_get_nguid
):
vol, vol_name = self.new_fake_vol()
mock_get_nvme_ports.return_value = NVME_PORTS
mock_get_wwn.return_value = "3624a93709714b5cb91634c470002b2c8"
mock_get_nguid.return_value = "0009714b5cb916324a9374c470002b2c8"
lun = 1
connection = {
"vol": vol_name,
"lun": lun,
}
mock_connection.return_value = connection
result = deepcopy(NVME_CONNECTION_INFO)
real_result = self.driver.initialize_connection(vol, NVME_CONNECTOR)
self.maxDiff = None
self.assertDictEqual(result, real_result)
mock_get_nvme_ports.assert_called_with(self.array)
mock_connection.assert_called_with(
self.array, vol_name, NVME_CONNECTOR
)
self.assert_error_propagates(
[mock_get_nvme_ports, mock_connection],
self.driver.initialize_connection,
vol,
NVME_CONNECTOR,
)
@mock.patch(NVME_DRIVER_OBJ + "._get_nguid")
@mock.patch(NVME_DRIVER_OBJ + "._get_wwn")
@mock.patch(NVME_DRIVER_OBJ + "._connect")
@mock.patch(NVME_DRIVER_OBJ + "._get_target_nvme_ports")
def test_initialize_connection_ipv6(
self, mock_get_nvme_ports, mock_connection, mock_get_wwn,
mock_get_nguid
):
vol, vol_name = self.new_fake_vol()
mock_get_nvme_ports.return_value = NVME_PORTS
mock_get_wwn.return_value = "3624a93709714b5cb91634c470002b2c8"
mock_get_nguid.return_value = "0009714b5cb916324a9374c470002b2c8"
lun = 1
connection = {
"vol": vol_name,
"lun": lun,
}
mock_connection.return_value = connection
self.mock_config.pure_nvme_cidr = NVME_CIDR_V6
result = deepcopy(NVME_CONNECTION_INFO_V6)
real_result = self.driver.initialize_connection(vol, NVME_CONNECTOR)
self.maxDiff = None
self.assertDictEqual(result, real_result)
mock_get_nvme_ports.assert_called_with(self.array)
mock_connection.assert_called_with(
self.array, vol_name, NVME_CONNECTOR
)
self.assert_error_propagates(
[mock_get_nvme_ports, mock_connection],
self.driver.initialize_connection,
vol,
NVME_CONNECTOR,
)
@mock.patch(NVME_DRIVER_OBJ + "._get_nguid")
@mock.patch(NVME_DRIVER_OBJ + "._get_wwn")
@mock.patch(NVME_DRIVER_OBJ + "._connect")
@mock.patch(NVME_DRIVER_OBJ + "._get_target_nvme_ports")
def test_initialize_connection_multipath(
self, mock_get_nvme_ports, mock_connection, mock_get_wwn,
mock_get_nguid
):
self.driver.configuration.pure_nvme_transport = "roce"
vol, vol_name = self.new_fake_vol()
mock_get_nvme_ports.return_value = NVME_PORTS
mock_get_wwn.return_value = "3624a93709714b5cb91634c470002b2c8"
mock_get_nguid.return_value = "0009714b5cb916324a9374c470002b2c8"
lun = 1
connection = {
"vol": vol_name,
"lun": lun,
}
mock_connection.return_value = connection
multipath_connector = deepcopy(NVME_CONNECTOR)
multipath_connector["multipath"] = True
result = deepcopy(NVME_CONNECTION_INFO)
real_result = self.driver.initialize_connection(
vol, multipath_connector
)
self.assertDictEqual(result, real_result)
mock_get_nvme_ports.assert_called_with(self.array)
mock_connection.assert_called_with(
self.array, vol_name, multipath_connector
)
multipath_connector["multipath"] = False
self.driver.initialize_connection(vol, multipath_connector)
def test_get_target_nvme_ports(self):
self.array.list_ports.return_value = NVME_PORTS
ret = self.driver._get_target_nvme_ports(self.array)
self.assertEqual(NVME_PORTS, ret)
def test_get_target_nvme_ports_with_nvme_and_fc(self):
self.array.list_ports.return_value = NVME_PORTS_WITH
ret = self.driver._get_target_nvme_ports(self.array)
self.assertEqual(NVME_PORTS, ret)
def test_get_target_nvme_ports_with_no_ports(self):
# Should raise an exception if there are no ports
self.array.list_ports.return_value = []
self.assertRaises(
pure.PureDriverException,
self.driver._get_target_nvme_ports,
self.array,
)
def test_get_target_nvme_ports_with_only_fc_ports(self):
# Should raise an exception of there are no nvme ports
self.array.list_ports.return_value = PORTS_WITHOUT
self.assertRaises(
pure.PureDriverException,
self.driver._get_target_nvme_ports,
self.array,
)
@mock.patch(NVME_DRIVER_OBJ + "._get_host", autospec=True)
@mock.patch(NVME_DRIVER_OBJ + "._generate_purity_host_name", spec=True)
def test_connect(self, mock_generate, mock_host):
vol, vol_name = self.new_fake_vol()
result = {"vol": vol_name, "lun": 1}
# Branch where host already exists
mock_host.return_value = [PURE_HOST]
self.array.connect_host.return_value = {"vol": vol_name, "lun": 1}
real_result = self.driver._connect(
self.array, vol_name, NVME_CONNECTOR
)
self.assertEqual(result, real_result)
mock_host.assert_called_with(
self.driver, self.array, NVME_CONNECTOR, remote=False
)
self.assertFalse(mock_generate.called)
self.assertFalse(self.array.create_host.called)
self.array.connect_host.assert_called_with(PURE_HOST_NAME, vol_name)
# Branch where new host is created
mock_host.return_value = []
mock_generate.return_value = PURE_HOST_NAME
real_result = self.driver._connect(
self.array, vol_name, NVME_CONNECTOR
)
mock_host.assert_called_with(
self.driver, self.array, NVME_CONNECTOR, remote=False
)
mock_generate.assert_called_with(HOSTNAME)
self.array.create_host.assert_called_with(
PURE_HOST_NAME, nqnlist=[INITIATOR_NQN]
)
self.assertFalse(self.array.set_host.called)
self.assertEqual(result, real_result)
mock_generate.reset_mock()
self.array.reset_mock()
self.assert_error_propagates(
[
mock_host,
mock_generate,
self.array.connect_host,
self.array.create_host,
],
self.driver._connect,
self.array,
vol_name,
NVME_CONNECTOR,
)
self.mock_config.safe_get.return_value = "oracle-vm-server"
# Branch where personality is set
self.driver._connect(self.array, vol_name, NVME_CONNECTOR)
self.assertDictEqual(result, real_result)
self.array.set_host.assert_called_with(
PURE_HOST_NAME, personality="oracle-vm-server"
)
@mock.patch(NVME_DRIVER_OBJ + "._get_host", autospec=True)
def test_connect_already_connected(self, mock_host):
vol, vol_name = self.new_fake_vol()
mock_host.return_value = [PURE_HOST]
expected = {"host": PURE_HOST_NAME, "lun": 1}
self.array.list_volume_private_connections.return_value = [
expected,
{"host": "extra", "lun": 2},
]
self.array.connect_host.side_effect = (
self.purestorage_module.PureHTTPError(
code=http.client.BAD_REQUEST, text="Connection already exists"
)
)
actual = self.driver._connect(self.array, vol_name, NVME_CONNECTOR)
self.assertEqual(expected, actual)
self.assertTrue(self.array.connect_host.called)
self.assertTrue(bool(self.array.list_volume_private_connections))
@mock.patch(NVME_DRIVER_OBJ + "._get_host", autospec=True)
def test_connect_already_connected_list_hosts_empty(self, mock_host):
vol, vol_name = self.new_fake_vol()
mock_host.return_value = [PURE_HOST]
self.array.list_volume_private_connections.return_value = {}
self.array.connect_host.side_effect = (
self.purestorage_module.PureHTTPError(
code=http.client.BAD_REQUEST, text="Connection already exists"
)
)
self.assertRaises(
pure.PureDriverException,
self.driver._connect,
self.array,
vol_name,
NVME_CONNECTOR,
)
self.assertTrue(self.array.connect_host.called)
self.assertTrue(bool(self.array.list_volume_private_connections))
@mock.patch(NVME_DRIVER_OBJ + "._get_host", autospec=True)
def test_connect_already_connected_list_hosts_exception(self, mock_host):
vol, vol_name = self.new_fake_vol()
mock_host.return_value = [PURE_HOST]
self.array.list_volume_private_connections.side_effect = (
self.purestorage_module.PureHTTPError(
code=http.client.BAD_REQUEST, text=""
)
)
self.array.connect_host.side_effect = (
self.purestorage_module.PureHTTPError(
code=http.client.BAD_REQUEST, text="Connection already exists"
)
)
self.assertRaises(
self.purestorage_module.PureHTTPError,
self.driver._connect,
self.array,
vol_name,
NVME_CONNECTOR,
)
self.assertTrue(self.array.connect_host.called)
self.assertTrue(bool(self.array.list_volume_private_connections))
@mock.patch(NVME_DRIVER_OBJ + "._get_host", autospec=True)
def test_connect_nqn_already_in_use(self, mock_host):
vol, vol_name = self.new_fake_vol()
mock_host.return_value = []
self.array.create_host.side_effect = (
self.purestorage_module.PureHTTPError(
code=http.client.BAD_REQUEST,
text="The specified NQN is already in use.",
)
)
# Because we mocked out retry make sure we are raising the right
# exception to allow for retries to happen.
self.assertRaises(
pure.PureRetryableException,
self.driver._connect,
self.array,
vol_name,
NVME_CONNECTOR,
)
@mock.patch(NVME_DRIVER_OBJ + "._get_host", autospec=True)
def test_connect_create_host_already_exists(self, mock_host):
vol, vol_name = self.new_fake_vol()
mock_host.return_value = []
self.array.create_host.side_effect = (
self.purestorage_module.PureHTTPError(
code=http.client.BAD_REQUEST, text="Host already exists."
)
)
# Because we mocked out retry make sure we are raising the right
# exception to allow for retries to happen.
self.assertRaises(
pure.PureRetryableException,
self.driver._connect,
self.array,
vol_name,
NVME_CONNECTOR,
)

View File

@ -96,6 +96,20 @@ PURE_OPTS = [
"targets hosts are allowed to connect to. It supports " "targets hosts are allowed to connect to. It supports "
"IPv4 and IPv6 subnets. This parameter supersedes " "IPv4 and IPv6 subnets. This parameter supersedes "
"pure_iscsi_cidr."), "pure_iscsi_cidr."),
cfg.StrOpt("pure_nvme_cidr", default="0.0.0.0/0",
help="CIDR of FlashArray NVMe targets hosts are allowed to "
"connect to. Default will allow connection to any "
"IPv4 address. This parameter now supports IPv6 subnets. "
"Ignored when pure_nvme_cidr_list is set."),
cfg.ListOpt("pure_nvme_cidr_list", default=None,
help="Comma-separated list of CIDR of FlashArray NVMe "
"targets hosts are allowed to connect to. It supports "
"IPv4 and IPv6 subnets. This parameter supersedes "
"pure_nvme_cidr."),
cfg.StrOpt("pure_nvme_transport", default="roce",
choices=['roce'],
help="The NVMe transport layer to be used by the NVMe driver. "
"This only supports RoCE at this time."),
cfg.BoolOpt("pure_eradicate_on_delete", cfg.BoolOpt("pure_eradicate_on_delete",
default=False, default=False,
help="When enabled, all Pure volumes, snapshots, and " help="When enabled, all Pure volumes, snapshots, and "
@ -139,6 +153,8 @@ MAX_VOL_LENGTH = 63
MAX_SNAP_LENGTH = 96 MAX_SNAP_LENGTH = 96
UNMANAGED_SUFFIX = '-unmanaged' UNMANAGED_SUFFIX = '-unmanaged'
NVME_PORT = 4420
REPL_SETTINGS_PROPAGATE_RETRY_INTERVAL = 5 # 5 seconds REPL_SETTINGS_PROPAGATE_RETRY_INTERVAL = 5 # 5 seconds
REPL_SETTINGS_PROPAGATE_MAX_RETRIES = 36 # 36 * 5 = 180 seconds REPL_SETTINGS_PROPAGATE_MAX_RETRIES = 36 # 36 * 5 = 180 seconds
@ -231,7 +247,8 @@ class PureBaseVolumeDriver(san.SanDriver):
additional_opts = cls._get_oslo_driver_opts( additional_opts = cls._get_oslo_driver_opts(
'san_ip', 'driver_ssl_cert_verify', 'driver_ssl_cert_path', 'san_ip', 'driver_ssl_cert_verify', 'driver_ssl_cert_path',
'use_chap_auth', 'replication_device', 'reserved_percentage', 'use_chap_auth', 'replication_device', 'reserved_percentage',
'max_over_subscription_ratio') 'max_over_subscription_ratio', 'pure_nvme_transport',
'pure_nvme_cidr_list', 'pure_nvme_cidr')
return PURE_OPTS + additional_opts return PURE_OPTS + additional_opts
def parse_replication_configs(self): def parse_replication_configs(self):
@ -265,6 +282,13 @@ class PureBaseVolumeDriver(san.SanDriver):
ssl_cert_path = replication_device.get("ssl_cert_path", None) ssl_cert_path = replication_device.get("ssl_cert_path", None)
repl_type = replication_device.get("type", repl_type = replication_device.get("type",
REPLICATION_TYPE_ASYNC) REPLICATION_TYPE_ASYNC)
if (
repl_type == REPLICATION_TYPE_SYNC
and "NVMe" in self._storage_protocol
):
msg = _('NVMe driver does not support synchronous '
'replication')
raise PureDriverException(reason=msg)
uniform = strutils.bool_from_string( uniform = strutils.bool_from_string(
replication_device.get("uniform", False)) replication_device.get("uniform", False))
@ -1796,6 +1820,9 @@ class PureBaseVolumeDriver(san.SanDriver):
@staticmethod @staticmethod
def _connect_host_to_vol(array, host_name, vol_name): def _connect_host_to_vol(array, host_name, vol_name):
connection = None connection = None
LOG.debug("Connecting volume %(vol)s to host %(host)s.",
{"vol": vol_name,
"host": host_name})
try: try:
connection = array.connect_host(host_name, vol_name) connection = array.connect_host(host_name, vol_name)
except purestorage.PureHTTPError as err: except purestorage.PureHTTPError as err:
@ -1834,7 +1861,7 @@ class PureBaseVolumeDriver(san.SanDriver):
""" """
qos = None qos = None
# TODO(patrickeast): Can remove this once new_type is a VolumeType OVO # TODO: Can remove this once new_type is a VolumeType OVO
new_type = volume_type.VolumeType.get_by_name_or_id(context, new_type = volume_type.VolumeType.get_by_name_or_id(context,
new_type['id']) new_type['id'])
previous_vol_replicated = volume.is_replicated() previous_vol_replicated = volume.is_replicated()
@ -2584,14 +2611,16 @@ class PureISCSIDriver(PureBaseVolumeDriver, san.SanISCSIDriver):
} }
if self.configuration.pure_iscsi_cidr_list: if self.configuration.pure_iscsi_cidr_list:
cidrs = self.configuration.pure_iscsi_cidr_list iscsi_cidrs = self.configuration.pure_iscsi_cidr_list
if self.configuration.pure_iscsi_cidr != "0.0.0.0/0": if self.configuration.pure_iscsi_cidr != "0.0.0.0/0":
LOG.warning("pure_iscsi_cidr was ignored as " LOG.warning("pure_iscsi_cidr was ignored as "
"pure_iscsi_cidr_list is set") "pure_iscsi_cidr_list is set")
else: else:
cidrs = [self.configuration.pure_iscsi_cidr] iscsi_cidrs = [self.configuration.pure_iscsi_cidr]
check_cidrs = [ipaddress.ip_network(item) for item in cidrs] check_iscsi_cidrs = [
ipaddress.ip_network(item) for item in iscsi_cidrs
]
target_luns = [] target_luns = []
target_iqns = [] target_iqns = []
@ -2606,10 +2635,10 @@ class PureISCSIDriver(PureBaseVolumeDriver, san.SanISCSIDriver):
# Check to ensure that the portal IP is in the iSCSI target # Check to ensure that the portal IP is in the iSCSI target
# CIDR before adding it # CIDR before adding it
target_portal = port["portal"] target_portal = port["portal"]
portal, p_sep, p_port = target_portal.rpartition(':') portal, p_port = target_portal.rsplit(':', 1)
portal = portal.strip('[]') portal = portal.strip('[]')
check_ip = ipaddress.ip_address(portal) check_ip = ipaddress.ip_address(portal)
for check_cidr in check_cidrs: for check_cidr in check_iscsi_cidrs:
if check_ip in check_cidr: if check_ip in check_cidr:
target_luns.append(target["connection"]["lun"]) target_luns.append(target["connection"]["lun"])
target_iqns.append(port["iqn"]) target_iqns.append(port["iqn"])
@ -2725,7 +2754,7 @@ class PureISCSIDriver(PureBaseVolumeDriver, san.SanISCSIDriver):
LOG.debug('Unable to set CHAP info: %s', err.text) LOG.debug('Unable to set CHAP info: %s', err.text)
raise PureRetryableException() raise PureRetryableException()
# TODO(patrickeast): Ensure that the host has the correct preferred # TODO: Ensure that the host has the correct preferred
# arrays configured for it. # arrays configured for it.
connection = self._connect_host_to_vol(array, connection = self._connect_host_to_vol(array,
@ -2798,7 +2827,7 @@ class PureFCDriver(PureBaseVolumeDriver, driver.FibreChannelDriver):
target_luns.append(connection["lun"]) target_luns.append(connection["lun"])
# Build the zoning map based on *all* wwns, this could be multiple # Build the zoning map based on *all* wwns, this could be multiple
# arrays connecting to the same host with a strected volume. # arrays connecting to the same host with a stretched volume.
init_targ_map = self._build_initiator_target_map(target_wwns, init_targ_map = self._build_initiator_target_map(target_wwns,
connector) connector)
@ -2850,7 +2879,7 @@ class PureFCDriver(PureBaseVolumeDriver, driver.FibreChannelDriver):
if personality: if personality:
self.set_personality(array, host_name, personality) self.set_personality(array, host_name, personality)
# TODO(patrickeast): Ensure that the host has the correct preferred # TODO: Ensure that the host has the correct preferred
# arrays configured for it. # arrays configured for it.
return self._connect_host_to_vol(array, host_name, vol_name) return self._connect_host_to_vol(array, host_name, vol_name)
@ -2924,3 +2953,184 @@ class PureFCDriver(PureBaseVolumeDriver, driver.FibreChannelDriver):
fczm_utils.remove_fc_zone(properties) fczm_utils.remove_fc_zone(properties)
return properties return properties
@interface.volumedriver
class PureNVMEDriver(PureBaseVolumeDriver, driver.BaseVD):
"""OpenStack Volume Driver to support Pure Storage FlashArray.
This version of the driver enables the use of NVMe over different
transport types for the underlying storage connectivity with the
FlashArray.
"""
VERSION = "15.0.nvme"
def __init__(self, *args, **kwargs):
execute = kwargs.pop("execute", utils.execute)
super(PureNVMEDriver, self).__init__(execute=execute,
*args, **kwargs)
if self.configuration.pure_nvme_transport == "roce":
self.transport_type = "rdma"
self._storage_protocol = constants.NVMEOF_ROCE
def _get_nguid(self, pure_vol_name):
"""Return the NGUID based on the volume's serial number
The NGUID is constructed from the volume serial number and
3 octet OUI
// octet 0: padding
// octets 1 - 7: first 7 octets of volume serial number
// octets 8 - 10: 3 octet OUI (24a937)
// octets 11 - 15: last 5 octets of volume serial number
"""
array = self._get_current_array()
volume_info = array.get_volume(pure_vol_name)
nguid = ("00" + volume_info['serial'][0:14] +
"24a937" + volume_info['serial'][-10:])
return nguid.lower()
def _get_host(self, array, connector, remote=False):
"""Return a list of dicts describing existing host objects or None."""
hosts = array.list_hosts(remote=remote)
matching_hosts = [host for host in hosts
if connector['nqn'] in host['nqn']]
return matching_hosts
@pure_driver_debug_trace
def initialize_connection(self, volume, connector):
"""Allow connection to connector and return connection info."""
pure_vol_name = self._get_vol_name(volume)
target_arrays = [self._get_current_array()]
if (
self._is_vol_in_pod(pure_vol_name)
and self._is_active_cluster_enabled
):
target_arrays += self._uniform_active_cluster_target_arrays
targets = []
for array in target_arrays:
connection = self._connect(array, pure_vol_name, connector)
target_ports = self._get_target_nvme_ports(array)
targets.append(
{
"connection": connection,
"ports": target_ports,
}
)
properties = self._build_connection_properties(targets)
properties["data"]["volume_nguid"] = self._get_nguid(pure_vol_name)
return properties
def _build_connection_properties(self, targets):
props = {
"driver_volume_type": "nvmeof",
"data": {
"discard": True,
},
}
if self.configuration.pure_nvme_cidr_list:
nvme_cidrs = self.configuration.pure_nvme_cidr_list
if self.configuration.pure_nvme_cidr != "0.0.0.0/0":
LOG.warning(
"pure_nvme_cidr was ignored as "
"pure_nvme_cidr_list is set"
)
else:
nvme_cidrs = [self.configuration.pure_nvme_cidr]
check_nvme_cidrs = [
ipaddress.ip_network(item) for item in nvme_cidrs
]
target_luns = []
target_nqns = []
target_portals = []
# Aggregate all targets together, we may end up with different
# namespaces for different target nqn/subsys sets (ie. it could
# be a unique namespace for each FlashArray)
for target in targets:
for port in target["ports"]:
# Check to ensure that the portal IP is in the NVMe target
# CIDR before adding it
target_portal = port["portal"]
if target_portal and port["nqn"]:
portal, p_port = target_portal.rsplit(':', 1)
portal = portal.strip("[]")
check_ip = ipaddress.ip_address(portal)
for check_cidr in check_nvme_cidrs:
if check_ip in check_cidr:
target_luns.append(target["connection"]["lun"])
target_nqns.append(port["nqn"])
target_portals.append(
(portal, NVME_PORT, self.transport_type)
)
LOG.debug(
"NVMe target portals that match CIDR range: '%s'", target_portals
)
# If we have multiple ports always report them.
if target_luns and target_nqns:
props["data"]["portals"] = target_portals
props["data"]["target_nqn"] = target_nqns[0]
else:
raise PureDriverException(
reason=_("No approrpiate nvme ports on target array.")
)
return props
def _get_target_nvme_ports(self, array):
"""Return list of nvme-enabled port descriptions."""
ports = array.list_ports()
nvme_ports = [port for port in ports if port["nqn"]]
if not nvme_ports:
raise PureDriverException(
reason=_("No nvme-enabled ports on target array.")
)
return nvme_ports
@utils.retry(PureRetryableException, retries=HOST_CREATE_MAX_RETRIES)
def _connect(self, array, vol_name, connector):
"""Connect the host and volume; return dict describing connection."""
nqn = connector["nqn"]
hosts = self._get_host(array, connector, remote=False)
host = hosts[0] if len(hosts) > 0 else None
if host:
host_name = host["name"]
LOG.info(
"Re-using existing purity host %(host_name)r",
{"host_name": host_name},
)
else:
personality = self.configuration.safe_get('pure_host_personality')
host_name = self._generate_purity_host_name(connector["host"])
LOG.info(
"Creating host object %(host_name)r with NQN:" " %(nqn)s.",
{"host_name": host_name, "nqn": connector["nqn"]},
)
try:
array.create_host(host_name, nqnlist=[nqn])
except purestorage.PureHTTPError as err:
if err.code == 400 and (
ERR_MSG_ALREADY_EXISTS in err.text
or ERR_MSG_ALREADY_IN_USE in err.text
):
# If someone created it before we could just retry, we will
# pick up the new host.
LOG.debug("Unable to create host: %s", err.text)
raise PureRetryableException()
if personality:
self.set_personality(array, host_name, personality)
# TODO: Ensure that the host has the correct preferred
# arrays configured for it.
return self._connect_host_to_vol(array, host_name, vol_name)

View File

@ -1,16 +1,19 @@
=================================================== =========================================================
Pure Storage iSCSI and Fibre Channel volume drivers Pure Storage iSCSI, Fibre Channel and NVMe volume drivers
=================================================== =========================================================
The Pure Storage FlashArray volume drivers for OpenStack Block Storage The Pure Storage FlashArray volume drivers for OpenStack Block Storage
interact with configured Pure Storage arrays and support various interact with configured Pure Storage arrays and support various
operations. operations.
Support for iSCSI storage protocol is available with the PureISCSIDriver Support for iSCSI storage protocol is available with the PureISCSIDriver
Volume Driver class, and Fibre Channel with PureFCDriver. Volume Driver class, Fibre Channel with the PureFCDriver and
NVMe-ROCE with the PureNVMEDriver.
All drivers are compatible with Purity FlashArrays that support the REST iSCSI and Fibre Channel drivers are compatible with Purity FlashArrays
API version 1.2, 1.3, 1.4, 1.5, 1.13, and 1.14 (Purity 4.0.0 and newer). that support the REST API version 1.6 and higher (Purity 4.7.0 and newer).
The NVMe driver is compatible with Purity FlashArrays
that support the REST API version 1.16 and higher (Purity 5.2.0 and newer).
Some features may require newer versions of Purity. Some features may require newer versions of Purity.
Limitations and known issues Limitations and known issues
@ -23,6 +26,8 @@ means you do not have the high-availability and non-disruptive upgrade
benefits provided by FlashArray. Multipathing must be used to take advantage benefits provided by FlashArray. Multipathing must be used to take advantage
of these benefits. of these benefits.
The NVMe driver does not support synchronous replication using ActiveCluster.
Supported operations Supported operations
~~~~~~~~~~~~~~~~~~~~ ~~~~~~~~~~~~~~~~~~~~
@ -46,7 +51,8 @@ Supported operations
* Create a thin provisioned volume. * Create a thin provisioned volume.
* Replicate volumes to remote Pure Storage array(s). * Replicate volumes to remote Pure Storage array(s) - synchronous replication
is not supported with the NVMe driver.
QoS support for the Pure Storage drivers include the ability to set the QoS support for the Pure Storage drivers include the ability to set the
following capabilities in the OpenStack Block Storage API following capabilities in the OpenStack Block Storage API
@ -152,9 +158,13 @@ Pure Storage FlashArray as back-end storage.
Replace the following variables accordingly: Replace the following variables accordingly:
PURE_VOLUME_DRIVER PURE_VOLUME_DRIVER
Use either ``cinder.volume.drivers.pure.PureISCSIDriver`` for iSCSI or Use ``cinder.volume.drivers.pure.PureISCSIDriver`` for iSCSI,
``cinder.volume.drivers.pure.PureFCDriver`` for Fibre Channel ``cinder.volume.drivers.pure.PureFCDriver`` for Fibre Channel
connectivity. or ``cinder.volume.drivers.pure.PureNVMEDriver`` for
NVME connectivity.
If using the NVME driver, specify the ``pure_nvme_transport`` value.
Currently only ``roce`` is supported.
IP_PURE_MGMT IP_PURE_MGMT
The IP address of the Pure Storage array's management interface or a The IP address of the Pure Storage array's management interface or a
@ -257,6 +267,10 @@ of the remote array.
The ``REPLICATION_TYPE`` value for the ``type`` key can be either ``sync`` or The ``REPLICATION_TYPE`` value for the ``type`` key can be either ``sync`` or
``async`` ``async``
.. note::
Synchronous replication is not supported by the NVMe driver.
If the ``type`` is ``sync`` volumes will be created in a stretched Pod. This If the ``type`` is ``sync`` volumes will be created in a stretched Pod. This
requires two arrays pre-configured with Active Cluster enabled. You can requires two arrays pre-configured with Active Cluster enabled. You can
optionally specify ``uniform`` as ``true`` or ``false``, this will instruct optionally specify ``uniform`` as ``true`` or ``false``, this will instruct

View File

@ -169,7 +169,7 @@ title=Open-E JovianDSS Storage Driver (iSCSI)
title=ProphetStor Flexvisor Driver (iSCSI, NFS) title=ProphetStor Flexvisor Driver (iSCSI, NFS)
[driver.pure] [driver.pure]
title=Pure Storage Driver (iSCSI, FC) title=Pure Storage Driver (iSCSI, FC, NVMe-RoCE)
[driver.qnap] [driver.qnap]
title=QNAP Storage Driver (iSCSI) title=QNAP Storage Driver (iSCSI)

View File

@ -0,0 +1,7 @@
---
features:
- |
Pure Storage adds a new driver to support NVMe-RoCE for the
FlashArray.
All features of the iSCSI and FC drivers are fully supported by this new
driver with the exception of synchronous replication.