diff --git a/ChangeLog.md b/ChangeLog.md index 4118982..e251f79 100644 --- a/ChangeLog.md +++ b/ChangeLog.md @@ -1,5 +1,10 @@ # PyPowerStore Change Log +## Version 2.1.0.0 - released on 29/09/23 +- Added support for cloning, refreshing, and restoring filesystem. +- Added support for creating and deleting NAS server. +- Added support for discovered appliances. + ## Version 2.0.0.0 - released on 30/06/23 - Added configuration operations for storage containers. - Added support for manual appliance selection to volume. diff --git a/ProgrammersGuideExamples/discovered_appliances_example.py b/ProgrammersGuideExamples/discovered_appliances_example.py new file mode 100644 index 0000000..f4a6bed --- /dev/null +++ b/ProgrammersGuideExamples/discovered_appliances_example.py @@ -0,0 +1,18 @@ +# -*- coding: utf-8 -*- +# Copyright: (c) 2023, Dell Technologies + +""" Discovered Appliance operations""" + +from PyPowerStore import powerstore_conn + +CONN = powerstore_conn.PowerStoreConn(username="", + password="", + server_ip="", + verify=False, + application_type="", + timeout=180.0) +print(CONN) + +# Get Discovered appliances list +discovered_appliances_list = CONN.config_mgmt.get_discovered_appliances() +print(discovered_appliances_list) diff --git a/ProgrammersGuideExamples/nas_server_examples.py b/ProgrammersGuideExamples/nas_server_examples.py index 02bb51a..46123ac 100644 --- a/ProgrammersGuideExamples/nas_server_examples.py +++ b/ProgrammersGuideExamples/nas_server_examples.py @@ -18,6 +18,18 @@ 'protection_policy_id': 'samplepolicyid' } +CREATE_PARAMS = { + "name": "test server", + "current_unix_directory_service": "LDAP", + "default_unix_user": "test-user", + "is_username_translation_enabled": True, + "is_auto_user_mapping_enabled": True, +} + +# create nasserver +NAS_SERVER = CONN.provisioning.create_nasserver(CREATE_PARAMS) +print(NAS_SERVER) + # Get nasserver list NAS_SERVERS = CONN.provisioning.get_nas_servers(all_pages=True) print(NAS_SERVERS) @@ -34,3 +46,7 @@ MODIFY_NAS = CONN.provisioning.modify_nasserver(NAS_SERVER[0]['id'], MODIFY_PARAMS) print(MODIFY_NAS) + +# delete nasserver +DELETE_NAS = CONN.provisioning.delete_nasserver(NAS_SERVER[0]['id']) +print(DELETE_NAS) diff --git a/PyPowerStore/__init__.py b/PyPowerStore/__init__.py index 92b47b1..7ec4390 100644 --- a/PyPowerStore/__init__.py +++ b/PyPowerStore/__init__.py @@ -2,6 +2,6 @@ """__init__.py.""" __title__ = 'PyPowerStore' -__version__ = '2.0.0.0' +__version__ = '2.1.0.0' __author__ = 'Dell Technologies or its subsidiaries' __copyright__ = 'Copyright 2019 Dell Technologies' diff --git a/PyPowerStore/configuration.py b/PyPowerStore/configuration.py index 50640e5..117f4bb 100644 --- a/PyPowerStore/configuration.py +++ b/PyPowerStore/configuration.py @@ -1028,7 +1028,10 @@ def get_appliances(self, filter_dict=None, all_pages=False): "Getting all appliances with filter: '%s' and all_pages: '%s'" % (filter_dict, all_pages)) querystring = helpers.prepare_querystring( - constants.SELECT_ID_NAME_AND_MODEL, filter_dict) + constants.APPLIANCE_DETAILS_QUERY, filter_dict) + if helpers.is_foot_hill_prime_or_higher(): + querystring = helpers.prepare_querystring( + constants.APPLIANCE_DETAILS_FHP_QUERY, filter_dict) LOG.info("Querystring: '%s'" % querystring) return self.config_client.request( constants.GET, @@ -1046,13 +1049,18 @@ def get_appliance_details(self, appliance_id): LOG.info("Getting appliance details by ID: '%s'" % appliance_id) querystring = constants.APPLIANCE_DETAILS_QUERY - if helpers.is_foot_hill_or_higher(): + if helpers.is_foot_hill_prime_or_higher(): + querystring = constants.APPLIANCE_DETAILS_FHP_QUERY + elif helpers.is_foot_hill_or_higher(): querystring = { 'select': 'id,name,service_tag,express_service_code,model,' 'drive_failure_tolerance_level,nodes,' - 'ip_pool_addresses,veth_ports,maintenance_windows,' - 'fc_ports,sas_ports,eth_ports,software_installed,' - 'virtual_volumes,hardware,volumes' + 'ip_pool_addresses(id,name),veth_ports(id,name),' + 'maintenance_windows,fc_ports(id,name),' + 'sas_ports(id,name),eth_ports(id,name),' + 'software_installed(id,release_version),' + 'virtual_volumes(id,name),hardware(id,name),' + 'volumes(id,name)' } return self.config_client.request( constants.GET, @@ -1071,13 +1079,18 @@ def get_appliance_by_name(self, appliance_name): """ LOG.info("Getting appliance details by name: '%s'" % appliance_name) querystring = constants.APPLIANCE_DETAILS_QUERY - if helpers.is_foot_hill_or_higher(): + if helpers.is_foot_hill_prime_or_higher(): + querystring = constants.APPLIANCE_DETAILS_FHP_QUERY + elif helpers.is_foot_hill_or_higher(): querystring = { 'select': 'id,name,service_tag,express_service_code,model,' 'drive_failure_tolerance_level,nodes,' - 'ip_pool_addresses,veth_ports,maintenance_windows,' - 'fc_ports,sas_ports,eth_ports,software_installed,' - 'virtual_volumes,hardware,volumes' + 'ip_pool_addresses(id,name),veth_ports(id,name),' + 'maintenance_windows,fc_ports(id,name),' + 'sas_ports(id,name),eth_ports(id,name),' + 'software_installed(id,release_version),' + 'virtual_volumes(id,name),hardware(id,name),' + 'volumes(id,name)' } return self.config_client.request( @@ -1086,6 +1099,48 @@ def get_appliance_by_name(self, appliance_name): querystring=helpers.prepare_querystring( querystring, name=constants.EQUALS + appliance_name)) + + # Discovered Appliances methods + def get_discovered_appliances(self, filter_dict=None, all_pages=False): + """ + Get discovered appliances. + Parameters: + - filter_dict (dict): (optional) Filter details + - all_pages (bool): (optional) Indicates whether to return all + appliances or not + + Returns: + - List of discovered appliances (list)[dict] + """ + LOG.info("Getting discovered appliances with filter: '%s' and " + "all_pages: '%s'",filter_dict, all_pages) + + if all_pages: + raise ValueError("Pagination is not supported for discovered " + "appliances.") + + querystring = constants.DISCOVERED_APPLIANCE_DETAILS_QUERY.copy() + + if not filter_dict: + querystring = helpers.prepare_querystring(querystring) + + LOG.info("Querystring without filters dict: '%s'", querystring) + return self.config_client.request( + constants.GET, + constants.GET_DISCOVERED_APPLIANCE_LIST_URL.format(self.server_ip), + querystring=querystring, + all_pages=False + ) + resp = self.config_client.request( + constants.GET, + constants.GET_DISCOVERED_APPLIANCE_LIST_URL.format(self.server_ip), + querystring=querystring, all_pages=False) + filterable_keys = ['id', 'link_local_address', 'service_name', 'state', + 'is_unified_capable', 'mode', 'is_local', + 'management_service_ready', 'build_version', + 'node_count', 'express_service_code'] + return helpers.filtered_details(filterable_keys, filter_dict, + resp, 'discovered_appliances') # Appliance operations end # Certificate operations start diff --git a/PyPowerStore/provisioning.py b/PyPowerStore/provisioning.py index 61c86a7..0e68f5f 100644 --- a/PyPowerStore/provisioning.py +++ b/PyPowerStore/provisioning.py @@ -406,6 +406,64 @@ def _prepare_restore_volume_payload(self, snap_id_to_restore_from=None, return refresh_volume_dict + def clone_filesystem(self, filesystem_id, advance_parameters): + """Clone a filesystem. + + :param filesystem_id: The filesystem ID + :type filesystem_id: str + :param advance_parameters: Advance attributes + :type advance_parameters: str + :return: Unique identifier of the new instance created if success else raise exception + :rtype: dict + """ + LOG.info("Cloning the filesystem: '%s'" % filesystem_id) + payload = dict() + + if advance_parameters: + for key, value in advance_parameters.items(): + if key in constants.FILESYSTEM_PRIME and \ + not helpers.is_foot_hill_prime_or_higher(): + raise Exception( key + " is supported for PowerStore" \ + " version 3.0.0.0 and above.") + payload[key] = value + return self.client.request( + constants.POST, constants.CLONE_FILESYSTEM_URL.format( + self.server_ip, filesystem_id), + payload) + + def restore_filesystem(self, snapshot_id, backup_snap_name=None): + """Restore a filesystem. + + :param snapshot_id: Unique identifier of the file system snapshot. + :type snapshot_id: str + :param backup_snap_name: Name of the backup snap to be created before the restore operation occurs. If no name is specified, no backup copy will be made. + :type backup_snap_name: str + :return: Unique identifier of the backup snapshot set or None if backup_snap_name is None + if success else raise exception + :rtype: dict + """ + LOG.info("Restoring the filesystem from snapshot: '%s'" % snapshot_id) + payload = {} + if backup_snap_name is not None: + payload['copy_name'] = backup_snap_name + return self.client.request( + constants.POST, constants.RESTORE_FILESYSTEM_URL.format( + self.server_ip, snapshot_id), + payload) + + def refresh_filesystem(self, snapshot_id): + """Refresh a filesystem. + + :param snapshot_id: Unique identifier of the file system snapshot. + :type snapshot_id: str + :return: None if success else raise exception. + :rtype: None + """ + LOG.info("Refreshing the filesystem from snapshot: '%s'" % snapshot_id) + return self.client.request( + constants.POST, constants.REFRESH_FILESYSTEM_URL.format( + self.server_ip, snapshot_id)) + def add_protection_policy_for_volume(self, volume_id, protection_policy_id): """Add protection policy for volume. @@ -1508,7 +1566,7 @@ def get_cluster_list(self): constants.GET_CLUSTER.format (self.server_ip), payload=None, querystring=constants. - SELECT_ID_AND_NAME) + CLUSTER_DETAILS_QUERY) def get_host_volume_mapping(self, volume_id): """Get Host volume mapping details. @@ -1596,6 +1654,25 @@ def get_nas_server_by_name(self, nas_server_name): ) ) + def create_nasserver(self, payload): + """Create a NAS Server. + + :param payload: The payload to create the NAS Server + :type payload: dict + :return: NAS server ID on success else raise exception + :rtype: dict + """ + LOG.info("Creating NAS server: '%s'" % payload.get('name')) + if 'protection-policy' in payload and \ + not helpers.is_foot_hill_prime_or_higher(): + raise Exception("Protection policy is supported for PowerStore" \ + " version 3.0.0.0 and above.") + return self.client.request( + constants.POST, + constants.CREATE_NAS_SERVER_URL.format(self.server_ip), + payload=payload) + + def modify_nasserver(self, nasserver_id, modify_parameters): """Modify NAS Server attributes. @@ -1622,6 +1699,19 @@ def modify_nasserver(self, nasserver_id, modify_parameters): raise ValueError("Nothing to modify") + def delete_nasserver(self, nasserver_id): + """Delete a NAS Server. + + :param nasserver_id: The ID of the NAS Server to delete + :type nasserver_id: str + :return: None on success else raise exception + :rtype: None + """ + LOG.info("Deleting NAS server: '%s'" % nasserver_id) + return self.client.request( + constants.DELETE, + constants.DELETE_NAS_SERVER_URL.format(self.server_ip, nasserver_id)) + # NAS Server methods end # File System Methods diff --git a/PyPowerStore/tests/unit_tests/base_test.py b/PyPowerStore/tests/unit_tests/base_test.py index ff6c3da..43b6cb2 100644 --- a/PyPowerStore/tests/unit_tests/base_test.py +++ b/PyPowerStore/tests/unit_tests/base_test.py @@ -21,6 +21,7 @@ from PyPowerStore.tests.unit_tests.data.storage_container_data import StorageContainerData from PyPowerStore.tests.unit_tests.data.storage_container_destination_data import StorageContainerDestinationData from PyPowerStore.tests.unit_tests.data.replication_group_data import ReplicationGroupData +from PyPowerStore.tests.unit_tests.data.discovered_appliances import DiscoveredApplianceData from unittest import mock class TestBase(TestCase): @@ -45,6 +46,7 @@ def setUp(self): self.vcenter_data = VcenterData() self.virtual_volume_data = VirtualVolumeData() self.file_system_data = FileSystemData() + self.discovered_appliance_data = DiscoveredApplianceData() self.conf = PowerStoreConfig() self.mock_client = mock.patch('PyPowerStore.provisioning.Client', new=MockClient) diff --git a/PyPowerStore/tests/unit_tests/data/discovered_appliances.py b/PyPowerStore/tests/unit_tests/data/discovered_appliances.py new file mode 100644 index 0000000..1cdc422 --- /dev/null +++ b/PyPowerStore/tests/unit_tests/data/discovered_appliances.py @@ -0,0 +1,28 @@ +class DiscoveredApplianceData: + discovered_appliance_id = '51d0dc86-f0e8-2fdb-81g0-5cd1hgfhhd1e' + discovered_appliance_list = [ + { + "id": "51d0dc86-f0e8-2fdb-81g0-5cd1hgfhhd1e", + "link_local_address": "xx.xx.xx.xx", + "service_name": "appliance", + "service_tag": "appliance-tag", + "state": "Unconfigured", + "mode": "Unified", + "model": "appliance-model", + "express_service_code": "service-code", + "is_local": True, + "management_service_ready": True, + "software_version_compatibility": "Same", + "build_version": "build-version", + "build_id": "build-id", + "power_score": 0, + "node_count": 1, + "is_unified_capable": True, + "drive_failure_tolerance_level_and_availability": [ + { + "level": "Single", + "availability": "Available" + } + ] + } + ] diff --git a/PyPowerStore/tests/unit_tests/entity/discovered_appliances.py b/PyPowerStore/tests/unit_tests/entity/discovered_appliances.py new file mode 100644 index 0000000..64014be --- /dev/null +++ b/PyPowerStore/tests/unit_tests/entity/discovered_appliances.py @@ -0,0 +1,24 @@ +from PyPowerStore.tests.unit_tests.entity.base_abstract import Entity +from PyPowerStore.tests.unit_tests.data.discovered_appliances import DiscoveredApplianceData + + +class DiscoveredApplianceResponse(Entity): + + def __init__(self, method, url, **kwargs): + self.method = method + self.url = url + self.kwargs = kwargs + self.discovered_appliance_data = DiscoveredApplianceData() + self.status_code = 200 + + def get_api_name(self): + if self.method == 'GET': + return self.get_discovered_appliances + + def execute_api(self, api_name): + status_code, response = api_name() + return status_code, response + + def get_discovered_appliances(self): + return self.status_code,\ + self.discovered_appliance_data.discovered_appliance_list diff --git a/PyPowerStore/tests/unit_tests/entity/file_system.py b/PyPowerStore/tests/unit_tests/entity/file_system.py index fa91a82..58aa994 100644 --- a/PyPowerStore/tests/unit_tests/entity/file_system.py +++ b/PyPowerStore/tests/unit_tests/entity/file_system.py @@ -29,6 +29,10 @@ def get_api_name(self): elif self.method == 'POST': if self.url.endswith('/snapshot'): return self.create_filesystem_snapshot + elif self.url.endswith('/refresh'): + return self.refresh_filesystem + elif self.url.endswith('/restore'): + return self.restore_filesystem return self.create_filesystem elif self.method == 'PATCH': return self.modify_fs @@ -57,6 +61,12 @@ def get_snapshots_filesystem(self): def modify_fs(self): return 204, None + def restore_filesystem(self): + return 204, None + + def refresh_filesystem(self): + return 204, None + def delete_fs(self): if self.url.endswith('/file_system/{0}'.format( self.data.invalid_fs_id)): diff --git a/PyPowerStore/tests/unit_tests/entity/nas_server.py b/PyPowerStore/tests/unit_tests/entity/nas_server.py index c025aad..5c69da5 100644 --- a/PyPowerStore/tests/unit_tests/entity/nas_server.py +++ b/PyPowerStore/tests/unit_tests/entity/nas_server.py @@ -24,6 +24,10 @@ def get_api_name(self): return self.get_nasserver_detail elif self.method == 'PATCH': return self.modify_nas + elif self.method == 'POST': + return self.create_nasserver + elif self.method == 'DELETE': + return self.delete_nasserver def execute_api(self, api_name): status_code, response = api_name() @@ -48,3 +52,9 @@ def modify_nas(self): self.data.nas_id_not_exist)): return 404, self.data.nas_error[404] return 204, None + + def create_nasserver(self): + return 201, self.data.nas_id1 + + def delete_nasserver(self): + return 204, None diff --git a/PyPowerStore/tests/unit_tests/myrequests.py b/PyPowerStore/tests/unit_tests/myrequests.py index ab9ed88..ae2c52a 100644 --- a/PyPowerStore/tests/unit_tests/myrequests.py +++ b/PyPowerStore/tests/unit_tests/myrequests.py @@ -43,6 +43,7 @@ from PyPowerStore.tests.unit_tests.entity.storage_container import StorageContainerResponse from PyPowerStore.tests.unit_tests.entity.storage_container_destination import StorageContainerDestinationResponse from PyPowerStore.tests.unit_tests.entity.replication_group import ReplicationGroupResponse +from PyPowerStore.tests.unit_tests.entity.discovered_appliances import DiscoveredApplianceResponse import json # map the entity class name with the url resource name @@ -91,7 +92,8 @@ 'ldap_account': LDAPAccountResponse, 'storage_container': StorageContainerResponse, 'storage_container_destination': StorageContainerDestinationResponse, - 'replication_group': ReplicationGroupResponse + 'replication_group': ReplicationGroupResponse, + 'discovered_appliance': DiscoveredApplianceResponse } diff --git a/PyPowerStore/tests/unit_tests/test_discovered_appliances.py b/PyPowerStore/tests/unit_tests/test_discovered_appliances.py new file mode 100644 index 0000000..bc47e1f --- /dev/null +++ b/PyPowerStore/tests/unit_tests/test_discovered_appliances.py @@ -0,0 +1,14 @@ +from PyPowerStore.tests.unit_tests.base_test import TestBase +from PyPowerStore.utils.exception import PowerStoreException + + +class TestDiscoveredAppliances(TestBase): + def test_get_discovered_appliances(self): + discovered_appliance_list = self.configuration.get_discovered_appliances() + self.assertListEqual(discovered_appliance_list, self.discovered_appliance_data.discovered_appliance_list) + + def test_get_discovered_appliances_all_pages(self): + all_pages = True + callable_func = lambda: self.configuration.get_discovered_appliances(None, all_pages) + # Check if the correct exception is raised + self.assertRaises(ValueError, callable_func) diff --git a/PyPowerStore/tests/unit_tests/test_file_system.py b/PyPowerStore/tests/unit_tests/test_file_system.py index 0cd1ab3..4b1531b 100644 --- a/PyPowerStore/tests/unit_tests/test_file_system.py +++ b/PyPowerStore/tests/unit_tests/test_file_system.py @@ -113,6 +113,20 @@ def test_delete_filesystem(self): resp = self.provisioning.delete_filesystem(self.file_system_data.fs_id1) self.assertIsNone(resp) + def test_clone_filesystem(self): + resp = \ + self.provisioning.clone_filesystem(self.file_system_data.fs_id1, + advance_parameters={'name': self.file_system_data.fs_name2}) + self.assertEqual(resp, self.file_system_data.create_filesystem) + + def test_restore_filesystem(self): + resp = self.provisioning.restore_filesystem(self.file_system_data.fs_snap_id) + self.assertIsNone(resp) + + def test_refresh_filesystem(self): + resp = self.provisioning.refresh_filesystem(self.file_system_data.fs_snap_id) + self.assertIsNone(resp) + def test_delete_invalid_filesystem(self): self.assertRaisesRegex( PowerStoreException, diff --git a/PyPowerStore/tests/unit_tests/test_nas_server.py b/PyPowerStore/tests/unit_tests/test_nas_server.py index 58922cf..7c62181 100644 --- a/PyPowerStore/tests/unit_tests/test_nas_server.py +++ b/PyPowerStore/tests/unit_tests/test_nas_server.py @@ -82,3 +82,12 @@ def test_modify_nasserver_with_empty_param(self): self.assertRaises( ValueError, self.provisioning.modify_nasserver, self.data.nas_id1, {}) + + def test_create_nas_server(self): + payload = {"name": "nas1", "default_unix_user": "user1", "default_windows_user": "user2"} + nas_id = self.provisioning.create_nasserver(payload) + self.assertEqual(nas_id, self.data.nas_id1) + + def test_delete_nas_server(self): + resp = self.provisioning.delete_nasserver(self.data.nas_name1) + self.assertIsNone(resp) diff --git a/PyPowerStore/utils/constants.py b/PyPowerStore/utils/constants.py index e36f0ba..a0fb406 100644 --- a/PyPowerStore/utils/constants.py +++ b/PyPowerStore/utils/constants.py @@ -20,6 +20,9 @@ # max number of items limit in a response MAX_LIMIT = 2000 +# Foot Hill Prime Plus Version +FOOTHILL_PRIME_PLUS_VERSION = '3.2.0.0' + # Query params # Volume Query @@ -341,9 +344,29 @@ # Appliance details APPLIANCE_DETAILS_QUERY = { 'select': 'id,name,service_tag,express_service_code,model,nodes,' - 'veth_ports,maintenance_windows,fc_ports,sas_ports,eth_ports,' - 'software_installed,virtual_volumes,hardware,volumes,' - 'ip_pool_addresses' + 'veth_ports(id,name),maintenance_windows,fc_ports(id,name),' + 'sas_ports(id,name),eth_ports(id,name),' + 'software_installed(id,release_version),' + 'virtual_volumes(id,name),hardware(id,name),volumes(id,name),' + 'ip_pool_addresses(id,name)' +} +APPLIANCE_DETAILS_FHP_QUERY = { + 'select': 'id,name,service_tag,express_service_code,model,nodes,' + 'veth_ports(id,name),maintenance_windows,fc_ports(id,name),' + 'sas_ports(id,name),eth_ports(id,name),eth_be_ports(id,name),' + 'software_installed(id,release_version),virtual_volumes(id,name)' + ',hardware(id,name),volumes(id,name),ip_pool_addresses(id,name),' + 'node_count,drive_failure_tolerance_level,' + 'drive_failure_tolerance_level_l10n' +} + +# Discovered appliance details +DISCOVERED_APPLIANCE_DETAILS_QUERY = { + 'select': 'id,link_local_address,service_name,service_tag,state,mode,' + 'model,express_service_code,is_local,management_service_ready,' + 'software_version_compatibility,build_version,build_id,' + 'power_score,node_count,is_unified_capable,' + 'drive_failure_tolerance_level_and_availability(level,availability)' } # Remote System @@ -578,6 +601,8 @@ GET_NAS_SERVER_DETAILS_URL = 'https://{0}/api/rest/nas_server/{1}' GET_NAS_SERVER_DETAILS_BY_NAME_URL = GET_NAS_SERVER_LIST_URL MODIFY_NAS_SERVER_URL = GET_NAS_SERVER_DETAILS_URL +CREATE_NAS_SERVER_URL = GET_NAS_SERVER_LIST_URL +DELETE_NAS_SERVER_URL = GET_NAS_SERVER_DETAILS_URL # NFS Export endpoints GET_NFS_EXPORT_LIST_URL = 'https://{0}/api/rest/nfs_export' @@ -614,6 +639,10 @@ CREATE_FILESYSTEM_URL = GET_FILESYSTEM_DETAILS_BY_NAME_URL DELETE_FILESYSTEM_URL = GET_FILESYSTEM_DETAILS_URL +CLONE_FILESYSTEM_URL = 'https://{0}/api/rest/file_system/{1}/clone' +REFRESH_FILESYSTEM_URL = 'https://{0}/api/rest/file_system/{1}/refresh' +RESTORE_FILESYSTEM_URL = 'https://{0}/api/rest/file_system/{1}/restore' + MODIFY_FILESYSTEM_URL = GET_FILESYSTEM_DETAILS_URL CREATE_FILESYSTEM_SNAPSHOT_URL = 'https://{0}/api/rest/file_system/{1}/' \ 'snapshot' @@ -687,6 +716,9 @@ GET_APPLIANCE_LIST_URL = 'https://{0}/api/rest/appliance' GET_APPLIANCE_DETAILS_URL = 'https://{0}/api/rest/appliance/{1}' +# Discovered Appliance endpoints +GET_DISCOVERED_APPLIANCE_LIST_URL = 'https://{0}/api/rest/discovered_appliance' + # Certificate endpoints GET_CERTIFICATE_LIST_URL = 'https://{0}/api/rest/x509_certificate' GET_CERTIFICATE_DETAILS_URL = 'https://{0}/api/rest/x509_certificate/{1}' diff --git a/PyPowerStore/utils/helpers.py b/PyPowerStore/utils/helpers.py index 3546b9b..de03fa1 100644 --- a/PyPowerStore/utils/helpers.py +++ b/PyPowerStore/utils/helpers.py @@ -46,7 +46,6 @@ def get_logger(module_name, enable_log=False): LOG.disabled = True return LOG - def is_foot_hill_or_higher(): """Returns true if the array version is foot hill or higher. @@ -120,8 +119,12 @@ def filtered_details(filterable_keys, filter_dict, resource_list, temp_dict['id'] = resource['id'] # check if resource has 'name' parameter or not. if resource_name not in ["CHAP config", "service config", - "security config", "remote_support_contact", "ldap_domain"]: + "security config", + "remote_support_contact", + "ldap_domain", "discovered_appliances"]: temp_dict['name'] = resource['name'] + elif resource_name == "discovered_appliances": + temp_dict.update(resource) response.append(temp_dict) return response diff --git a/docs/conf.py b/docs/conf.py index 525fa7e..052f1a4 100644 --- a/docs/conf.py +++ b/docs/conf.py @@ -22,7 +22,7 @@ author = 'Dell' # The full version, including alpha/beta/rc tags -release = '2.0.0' +release = '2.1.0' # -- General configuration --------------------------------------------------- diff --git a/setup.py b/setup.py index 3309b16..2c77af2 100644 --- a/setup.py +++ b/setup.py @@ -10,7 +10,7 @@ setup(name='PyPowerStore', - version='2.0.0.0', + version='2.1.0.0', description='Python Library for Dell PowerStore', author='Ansible Team at Dell', author_email='ansible.team@dell.com',