list_quota_quotas() with v9.4 fails for quotas with no description
nikolisg7 opened this issue · 3 comments
Hello everyone!
OneFS 9.4 introduced a "description" field for quotas, which is as I understand it optional. Despite of that, running list_quota_quotas() against a 9.4.x.x cluster yields an error if there are quotas with no description. Here is the traceback:
In [9]: api_instance.list_quota_quotas()
---------------------------------------------------------------------------
ValueError Traceback (most recent call last)
Input In [9], in <cell line: 1>()
----> 1 api_instance.list_quota_quotas()
File ~/bin/anaconda3/envs/ansiblepython310/lib/python3.10/site-packages/isilon_sdk/v9_4_0/api/quota_api.py:1865, in QuotaApi.list_quota_quotas(self, **kwargs)
1863 return self.list_quota_quotas_with_http_info(**kwargs) # noqa: E501
1864 else:
-> 1865 (data) = self.list_quota_quotas_with_http_info(**kwargs) # noqa: E501
1866 return data
File ~/bin/anaconda3/envs/ansiblepython310/lib/python3.10/site-packages/isilon_sdk/v9_4_0/api/quota_api.py:1971, in QuotaApi.list_quota_quotas_with_http_info(self, **kwargs)
1968 # Authentication setting
1969 auth_settings = ['basicAuth'] # noqa: E501
-> 1971 return self.api_client.call_api(
1972 '/platform/15/quota/quotas', 'GET',
1973 path_params,
1974 query_params,
1975 header_params,
1976 body=body_params,
1977 post_params=form_params,
1978 files=local_var_files,
1979 response_type='QuotaQuotasExtended', # noqa: E501
1980 auth_settings=auth_settings,
1981 async_req=params.get('async_req'),
1982 _return_http_data_only=params.get('_return_http_data_only'),
1983 _preload_content=params.get('_preload_content', True),
1984 _request_timeout=params.get('_request_timeout'),
1985 collection_formats=collection_formats)
File ~/bin/anaconda3/envs/ansiblepython310/lib/python3.10/site-packages/isilon_sdk/v9_4_0/api_client.py:341, in ApiClient.call_api(self, resource_path, method, path_params, query_params, header_params, body, post_params, files, response_type, auth_settings, async_req, _return_http_data_only, collection_formats, _preload_content, _request_timeout)
304 """Makes the HTTP request (synchronous) and returns deserialized data.
305
306 To make an async request, set the async_req parameter.
(...)
338 then the method will return the response directly.
339 """
340 if not async_req:
--> 341 return self.__call_api(resource_path, method,
342 path_params, query_params, header_params,
343 body, post_params, files,
344 response_type, auth_settings,
345 _return_http_data_only, collection_formats,
346 _preload_content, _request_timeout)
347 else:
348 thread = self.pool.apply_async(self.__call_api, (resource_path,
349 method, path_params, query_params,
350 header_params, body,
(...)
354 collection_formats,
355 _preload_content, _request_timeout))
File ~/bin/anaconda3/envs/ansiblepython310/lib/python3.10/site-packages/isilon_sdk/v9_4_0/api_client.py:185, in ApiClient.__call_api(self, resource_path, method, path_params, query_params, header_params, body, post_params, files, response_type, auth_settings, _return_http_data_only, collection_formats, _preload_content, _request_timeout)
182 if _preload_content:
183 # deserialize response data
184 if response_type:
--> 185 return_data = self.deserialize(response_data, response_type)
186 else:
187 return_data = None
File ~/bin/anaconda3/envs/ansiblepython310/lib/python3.10/site-packages/isilon_sdk/v9_4_0/api_client.py:257, in ApiClient.deserialize(self, response, response_type)
254 except ValueError:
255 data = response.data
--> 257 return self.__deserialize(data, response_type)
File ~/bin/anaconda3/envs/ansiblepython310/lib/python3.10/site-packages/isilon_sdk/v9_4_0/api_client.py:296, in ApiClient.__deserialize(self, data, klass)
294 return self.__deserialize_datatime(data)
295 else:
--> 296 return self.__deserialize_model(data, klass)
File ~/bin/anaconda3/envs/ansiblepython310/lib/python3.10/site-packages/isilon_sdk/v9_4_0/api_client.py:671, in ApiClient.__deserialize_model(self, data, klass)
667 if (data is not None and
668 klass.attribute_map[attr] in data and
669 isinstance(data, (list, dict))):
670 value = data[klass.attribute_map[attr]]
--> 671 kwargs[attr] = self.__deserialize(value, attr_type)
673 instance = klass(**kwargs)
675 if hasattr(instance, 'get_real_child_model'):
File ~/bin/anaconda3/envs/ansiblepython310/lib/python3.10/site-packages/isilon_sdk/v9_4_0/api_client.py:273, in ApiClient.__deserialize(self, data, klass)
271 if klass.startswith('list['):
272 sub_kls = re.match('list\[(.*)\]', klass).group(1)
--> 273 return [self.__deserialize(sub_data, sub_kls)
274 for sub_data in data]
276 if klass.startswith('dict('):
277 sub_kls = re.match('dict\(([^,]*), (.*)\)', klass).group(2)
File ~/bin/anaconda3/envs/ansiblepython310/lib/python3.10/site-packages/isilon_sdk/v9_4_0/api_client.py:273, in <listcomp>(.0)
271 if klass.startswith('list['):
272 sub_kls = re.match('list\[(.*)\]', klass).group(1)
--> 273 return [self.__deserialize(sub_data, sub_kls)
274 for sub_data in data]
276 if klass.startswith('dict('):
277 sub_kls = re.match('dict\(([^,]*), (.*)\)', klass).group(2)
File ~/bin/anaconda3/envs/ansiblepython310/lib/python3.10/site-packages/isilon_sdk/v9_4_0/api_client.py:296, in ApiClient.__deserialize(self, data, klass)
294 return self.__deserialize_datatime(data)
295 else:
--> 296 return self.__deserialize_model(data, klass)
File ~/bin/anaconda3/envs/ansiblepython310/lib/python3.10/site-packages/isilon_sdk/v9_4_0/api_client.py:673, in ApiClient.__deserialize_model(self, data, klass)
670 value = data[klass.attribute_map[attr]]
671 kwargs[attr] = self.__deserialize(value, attr_type)
--> 673 instance = klass(**kwargs)
675 if hasattr(instance, 'get_real_child_model'):
676 klass_name = instance.get_real_child_model(data)
File ~/bin/anaconda3/envs/ansiblepython310/lib/python3.10/site-packages/isilon_sdk/v9_4_0/models/quota_quota_extended.py:100, in QuotaQuotaExtended.__init__(self, container, description, efficiency_ratio, enforced, id, include_snapshots, labels, linked, notifications, path, persona, ready, reduction_ratio, thresholds, thresholds_on, type, usage)
97 self.discriminator = None
99 self.container = container
--> 100 self.description = description
101 if efficiency_ratio is not None:
102 self.efficiency_ratio = efficiency_ratio
File ~/bin/anaconda3/envs/ansiblepython310/lib/python3.10/site-packages/isilon_sdk/v9_4_0/models/quota_quota_extended.py:169, in QuotaQuotaExtended.description(self, description)
161 """Sets the description of this QuotaQuotaExtended.
162
163 User settable free form text description of the quota. # noqa: E501
(...)
166 :type: str
167 """
168 if description is None:
--> 169 raise ValueError("Invalid value for `description`, must not be `None`") # noqa: E501
170 if description is not None and len(description) > 1024:
171 raise ValueError("Invalid value for `description`, length must be less than or equal to `1024`") # noqa: E501
ValueError: Invalid value for `description`, must not be `None`
We noticed that the API spec specifies the description entry in the response body for GET /platform/15/quota/quotas as required. Here is an excerpt from https://onefs_host:8080/platform/15/quota/quotas?describe, where the GET response body is documented:
"container": {
"required": true,
"type": "boolean",
"description": "If true, SMB shares using the quota directory see the quota thresholds as share size."
},
"description": {
"minLength": 0,
"required": true,
"type": "string",
"description": "User settable free form text description of the quota.",
"maxLength": 1024
},
There is no description field though in the response body, if none is defined for a particular quota.
We observed similar behavior with method get_quota_quota(), when called with an quota id that has no description.
My first assumption would be that this spec in the OneFS API results in the QuotaQuotaExtended.description to be mandatory too, then a response that doesn't include a "description" field at all causes the initialization of the QuotaQuotaExtended object to fail
Is this something that can be fixed within the Python SDK or would it require changes on side of the OneFS API?
Thanks!
Georgios
Using v9.5 and experiencing the same issue, would like very much for this to be addressed.
Please use latest isilon-sdk package v0.3.01
Here is the link for latest version : https://pypi.org/project/isilon-sdk/ , version : Isilon_SDK_v0.4.0.