Dell PowerStore and PowerFlex Manila drivers enhancement

Addressed unresolved comments for Dell PowerStore Manila Driver
and PowerFlex Manila Driver.
1. Updated Dell PowerFlex Driver configuration guide.
2. Improved code style.

Implements: blueprint dell-powerstore-manila-driver
Change-Id: I744279b77226a097f774d3ae0f9a77b6524db1d0
This commit is contained in:
Yian Zong 2023-09-01 04:26:31 +00:00
parent 0b70c0de11
commit bd471d5d9a
4 changed files with 44 additions and 62 deletions

View File

@ -141,13 +141,13 @@ creating share type:
.. code-block:: console .. code-block:: console
$ openstack share type create --extra_specs snapshot_support=True ${share_type_name} False $ openstack share type create --extra_specs snapshot_support=True powerflex False
Or you can update already existing share type with command: Or you can update already existing share type with command:
.. code-block:: console .. code-block:: console
$ openstack share type set --extra_specs snapshot_support=True ${share_type_name} $ openstack share type set --extra_specs snapshot_support=True powerflex
Known restrictions Known restrictions

View File

@ -355,6 +355,7 @@ class PowerFlexStorageConnection(driver.StorageConnection):
self.protection_domain, self.protection_domain,
self.storage_pool self.storage_pool
) )
total = free = used = provisioned = 0
statistic = self.manager.get_storage_pool_statistic(storage_pool_id) statistic = self.manager.get_storage_pool_statistic(storage_pool_id)
if statistic: if statistic:
total = statistic.get('maxCapacityInKb') // units.Mi total = statistic.get('maxCapacityInKb') // units.Mi

View File

@ -184,7 +184,7 @@ class StorageObjectManager(object):
"storage_pool_id": storage_pool_id, "storage_pool_id": storage_pool_id,
"nas_server_id": nas_server_id "nas_server_id": nas_server_id
} }
url = self.base_url + '/v1/file-systems' url = f'{self.base_url}/v1/file-systems'
res, response = self.execute_powerflex_post_request(url, params) res, response = self.execute_powerflex_post_request(url, params)
if res.status_code == 201: if res.status_code == 201:
return response["id"] return response["id"]
@ -202,7 +202,7 @@ class StorageObjectManager(object):
"path": "/" + str(name), "path": "/" + str(name),
"name": name "name": name
} }
url = self.base_url + '/v1/nfs-exports' url = f'{self.base_url}/v1/nfs-exports'
res, response = self.execute_powerflex_post_request(url, params) res, response = self.execute_powerflex_post_request(url, params)
if res.status_code == 201: if res.status_code == 201:
return response["id"] return response["id"]
@ -213,9 +213,7 @@ class StorageObjectManager(object):
:param filesystem_id: ID of the filesystem to delete :param filesystem_id: ID of the filesystem to delete
:return: True if deleted successfully :return: True if deleted successfully
""" """
url = self.base_url + \ url = f'{self.base_url}/v1/file-systems/{filesystem_id}'
'/v1/file-systems/' + \
filesystem_id
res = self.execute_powerflex_delete_request(url) res = self.execute_powerflex_delete_request(url)
return res.status_code == 204 return res.status_code == 204
@ -229,10 +227,7 @@ class StorageObjectManager(object):
params = { params = {
"name": name "name": name
} }
url = self.base_url + \ url = f'{self.base_url}/v1/file-systems/{filesystem_id}/snapshot'
'/v1/file-systems/' + \
filesystem_id + \
'/snapshot'
res, response = self.execute_powerflex_post_request(url, params) res, response = self.execute_powerflex_post_request(url, params)
return res.status_code == 201 return res.status_code == 201
@ -242,9 +237,7 @@ class StorageObjectManager(object):
:param nas_server: NAS server name :param nas_server: NAS server name
:return: ID of the NAS server if success :return: ID of the NAS server if success
""" """
url = self.base_url + \ url = f'{self.base_url}/v1/nas-servers?select=id&name=eq.{nas_server}'
'/v1/nas-servers?select=id&name=eq.' + \
nas_server
res, response = self.execute_powerflex_get_request(url) res, response = self.execute_powerflex_get_request(url)
if res.status_code == 200: if res.status_code == 200:
return response[0]['id'] return response[0]['id']
@ -255,7 +248,7 @@ class StorageObjectManager(object):
:param export_id: ID of the NFS export :param export_id: ID of the NFS export
:return: path of the NFS export if success :return: path of the NFS export if success
""" """
url = self.base_url + '/v1/nfs-exports/' + export_id + '?select=*' url = f'{self.base_url}/v1/nfs-exports/{export_id}?select=*'
res, response = self.execute_powerflex_get_request(url) res, response = self.execute_powerflex_get_request(url)
if res.status_code == 200: if res.status_code == 200:
return response["name"] return response["name"]
@ -266,9 +259,7 @@ class StorageObjectManager(object):
:param name: name of the filesystem :param name: name of the filesystem
:return: ID of the filesystem if success :return: ID of the filesystem if success
""" """
url = self.base_url + \ url = f'{self.base_url}/v1/file-systems?select=id&name=eq.{name}'
'/v1/file-systems?select=id&name=eq.' + \
name
res, response = self.execute_powerflex_get_request(url) res, response = self.execute_powerflex_get_request(url)
if res.status_code == 200: if res.status_code == 200:
return response[0]['id'] return response[0]['id']
@ -279,9 +270,7 @@ class StorageObjectManager(object):
:param name: name of the NFS export :param name: name of the NFS export
:return: id of the NFS export if success :return: id of the NFS export if success
""" """
url = self.base_url + \ url = f'{self.base_url}/v1/nfs-exports?select=id&name=eq.{name}'
'/v1/nfs-exports?select=id&name=eq.' + \
name
res, response = self.execute_powerflex_get_request(url) res, response = self.execute_powerflex_get_request(url)
if res.status_code == 200: if res.status_code == 200:
return response[0]['id'] return response[0]['id']
@ -297,8 +286,8 @@ class StorageObjectManager(object):
"protectionDomainName": protection_domain, "protectionDomainName": protection_domain,
"name": storage_pool "name": storage_pool
} }
url = self.host_url + \ url = (f'{self.host_url}/api/types/StoragePool/instances/'
'/api/types/StoragePool/instances/action/queryIdByKey' 'action/queryIdByKey')
res, response = self.execute_powerflex_post_request(url, params) res, response = self.execute_powerflex_post_request(url, params)
if res.status_code == 200: if res.status_code == 200:
return response return response
@ -315,9 +304,7 @@ class StorageObjectManager(object):
"read_only_hosts": list(ro_hosts), "read_only_hosts": list(ro_hosts),
"read_write_root_hosts": list(rw_hosts) "read_write_root_hosts": list(rw_hosts)
} }
url = self.base_url + \ url = f'{self.base_url}/v1/nfs-exports/{export_id}'
'/v1/nfs-exports/' + \
export_id
res = self.execute_powerflex_patch_request(url, params) res = self.execute_powerflex_patch_request(url, params)
return res.status_code == 204 return res.status_code == 204
@ -331,9 +318,7 @@ class StorageObjectManager(object):
params = { params = {
"size_total": new_size "size_total": new_size
} }
url = self.base_url + \ url = f'{self.base_url}/v1/file-systems/{export_id}'
'/v1/file-systems/' + \
export_id
res = self.execute_powerflex_patch_request(url, params) res = self.execute_powerflex_patch_request(url, params)
return res.status_code == 204 return res.status_code == 204
@ -343,9 +328,8 @@ class StorageObjectManager(object):
:param name: name of the export :param name: name of the export
:return: ID of the Filesystem which owns the export :return: ID of the Filesystem which owns the export
""" """
url = self.base_url + \ url = (f'{self.base_url}/v1/nfs-exports'
'/v1/nfs-exports?select=file_system_id&name=eq.' + \ f'?select=file_system_id&name=eq.{name}')
name
res, response = self.execute_powerflex_get_request(url) res, response = self.execute_powerflex_get_request(url)
if res.status_code == 200: if res.status_code == 200:
return response[0]['file_system_id'] return response[0]['file_system_id']
@ -356,9 +340,8 @@ class StorageObjectManager(object):
:param snapshot_name: Name of the snapshot :param snapshot_name: Name of the snapshot
:return: ID of the parent filesystem of the snapshot :return: ID of the parent filesystem of the snapshot
""" """
url = self.base_url + \ url = (f'{self.base_url}/v1/file-systems'
'/v1/file-systems?select=id&name=eq.' + \ f'?select=id&name=eq.{snapshot_name}')
snapshot_name
res, response = self.execute_powerflex_get_request(url) res, response = self.execute_powerflex_get_request(url)
if res.status_code == 200: if res.status_code == 200:
return response[0]['id'] return response[0]['id']
@ -369,9 +352,7 @@ class StorageObjectManager(object):
:param storage_pool_id: ID of the storage pool :param storage_pool_id: ID of the storage pool
:return: Spare capacity percentage of the storage pool :return: Spare capacity percentage of the storage pool
""" """
url = self.host_url + \ url = f'{self.host_url}/api/instances/StoragePool::{storage_pool_id}'
'/api/instances/StoragePool::' + \
storage_pool_id
res, response = self.execute_powerflex_get_request(url) res, response = self.execute_powerflex_get_request(url)
if res.status_code == 200: if res.status_code == 200:
return response['sparePercentage'] return response['sparePercentage']
@ -382,9 +363,8 @@ class StorageObjectManager(object):
:param storage_pool_id: ID of the storage pool :param storage_pool_id: ID of the storage pool
:return: Statistics of the storage pool :return: Statistics of the storage pool
""" """
url = self.host_url + \ url = (f'{self.host_url}/api/instances/StoragePool::{storage_pool_id}'
'/api/instances/StoragePool::' + \ '/relationships/Statistics')
storage_pool_id + '/relationships/Statistics'
res, response = self.execute_powerflex_get_request(url) res, response = self.execute_powerflex_get_request(url)
if res.status_code == 200: if res.status_code == 200:
statistics = { statistics = {
@ -401,9 +381,8 @@ class StorageObjectManager(object):
:param nas_server_id: ID of the NAS server :param nas_server_id: ID of the NAS server
:return: file interfaces of the NAS server :return: file interfaces of the NAS server
""" """
url = self.base_url + \ url = (f'{self.base_url}/v1/file-interfaces'
'/v1/file-interfaces?select=ip_address&nas_server_id=eq.' + \ f'?select=ip_address&nas_server_id=eq.{nas_server_id}')
nas_server_id
res, response = self.execute_powerflex_get_request(url) res, response = self.execute_powerflex_get_request(url)
if res.status_code == 200: if res.status_code == 200:
return [i['ip_address'] for i in response] return [i['ip_address'] for i in response]

View File

@ -181,12 +181,12 @@ class PowerStoreStorageConnection(driver.StorageConnection):
return locations return locations
def _create_share_NFS_CIFS(self, nas_server_id, filesystem_id, share_name, def _create_share_NFS_CIFS(self, nas_server_id, filesystem_id, share_name,
protocal): protocol):
LOG.debug(f"Get file interfaces of {nas_server_id}") LOG.debug(f"Get file interfaces of {nas_server_id}")
file_interfaces = self.client.get_nas_server_interfaces( file_interfaces = self.client.get_nas_server_interfaces(
nas_server_id) nas_server_id)
LOG.debug(f"Creating {protocal} export {share_name}") LOG.debug(f"Creating {protocol} export {share_name}")
if protocal == 'NFS': if protocol == 'NFS':
export_id = self.client.create_nfs_export(filesystem_id, export_id = self.client.create_nfs_export(filesystem_id,
share_name) share_name)
if not export_id: if not export_id:
@ -197,7 +197,7 @@ class PowerStoreStorageConnection(driver.StorageConnection):
LOG.error(message) LOG.error(message)
raise exception.ShareBackendException(msg=message) raise exception.ShareBackendException(msg=message)
locations = self._get_nfs_location(file_interfaces, share_name) locations = self._get_nfs_location(file_interfaces, share_name)
elif protocal == 'CIFS': elif protocol == 'CIFS':
export_id = self.client.create_smb_share(filesystem_id, export_id = self.client.create_smb_share(filesystem_id,
share_name) share_name)
if not export_id: if not export_id:
@ -243,8 +243,8 @@ class PowerStoreStorageConnection(driver.StorageConnection):
LOG.debug(f"Retrieving filesystem ID for filesystem {share['name']}") LOG.debug(f"Retrieving filesystem ID for filesystem {share['name']}")
filesystem_id = self.client.get_filesystem_id(share['name']) filesystem_id = self.client.get_filesystem_id(share['name'])
if not filesystem_id: if not filesystem_id:
LOG.warning(f'Filesystem with share name {share["name"]} \ LOG.warning(
is not found.') f'Filesystem with share name {share["name"]} is not found.')
else: else:
LOG.debug(f"Deleting filesystem ID {filesystem_id}") LOG.debug(f"Deleting filesystem ID {filesystem_id}")
share_deleted = self.client.delete_filesystem(filesystem_id) share_deleted = self.client.delete_filesystem(filesystem_id)
@ -293,11 +293,11 @@ class PowerStoreStorageConnection(driver.StorageConnection):
def update_access(self, context, share, access_rules, add_rules, def update_access(self, context, share, access_rules, add_rules,
delete_rules, share_server=None): delete_rules, share_server=None):
"""Is called to update share access.""" """Is called to update share access."""
protocal = share['share_proto'].upper() protocol = share['share_proto'].upper()
LOG.debug(f'Updating access to {protocal} share.') LOG.debug(f'Updating access to {protocol} share.')
if protocal == 'NFS': if protocol == 'NFS':
return self._update_nfs_access(share, access_rules) return self._update_nfs_access(share, access_rules)
elif protocal == 'CIFS': elif protocol == 'CIFS':
return self._update_cifs_access(share, access_rules) return self._update_cifs_access(share, access_rules)
def _update_nfs_access(self, share, access_rules): def _update_nfs_access(self, share, access_rules):
@ -350,8 +350,10 @@ class PowerStoreStorageConnection(driver.StorageConnection):
access_updates.update({rule['access_id']: {'state': 'error'}}) access_updates.update({rule['access_id']: {'state': 'error'}})
else: else:
prefix = self.ad_domain or \ prefix = (
self.ad_domain or
self.client.get_nas_server_smb_netbios(self.nas_server) self.client.get_nas_server_smb_netbios(self.nas_server)
)
if not prefix: if not prefix:
message = ( message = (
_('Failed to get daomain/netbios name of ' _('Failed to get daomain/netbios name of '
@ -386,16 +388,16 @@ class PowerStoreStorageConnection(driver.StorageConnection):
stats_dict['driver_version'] = VERSION stats_dict['driver_version'] = VERSION
stats_dict['storage_protocol'] = 'NFS_CIFS' stats_dict['storage_protocol'] = 'NFS_CIFS'
stats_dict['reserved_percentage'] = self.reserved_percentage stats_dict['reserved_percentage'] = self.reserved_percentage
stats_dict['reserved_snapshot_percentage'] = \ stats_dict['reserved_snapshot_percentage'] = (
self.reserved_snapshot_percentage self.reserved_snapshot_percentage)
stats_dict['reserved_share_extend_percentage'] = \ stats_dict['reserved_share_extend_percentage'] = (
self.reserved_share_extend_percentage self.reserved_share_extend_percentage)
stats_dict['max_over_subscription_ratio'] = \ stats_dict['max_over_subscription_ratio'] = (
self.max_over_subscription_ratio self.max_over_subscription_ratio)
cluster_id = self.client.get_cluster_id() cluster_id = self.client.get_cluster_id()
total, used = self.client.retreive_cluster_capacity_metrics(cluster_id) total, used = self.client.retreive_cluster_capacity_metrics(cluster_id)
if(total and used): if total and used:
free = total - used free = total - used
stats_dict['total_capacity_gb'] = total // units.Gi stats_dict['total_capacity_gb'] = total // units.Gi
stats_dict['free_capacity_gb'] = free // units.Gi stats_dict['free_capacity_gb'] = free // units.Gi