mirror of
				https://github.com/ansible-collections/community.general.git
				synced 2025-10-25 13:34:01 -07:00 
			
		
		
		
	NSO ValueBuilder improvements. 4.5 leaf-list compatability. (#36583)
Fix issues in ValueBuilder used in nso_config and nso_verify so that it can handle leaf-list in NSO 4.5 and detect identityref types from unions. Fail gracefully if a type is not found.
This commit is contained in:
		
					parent
					
						
							
								b61b4bef11
							
						
					
				
			
			
				commit
				
					
						2789cc5c09
					
				
			
		
					 2 changed files with 119 additions and 14 deletions
				
			
		|  | @ -310,7 +310,9 @@ class ValueBuilder(object): | |||
|         if schema is None: | ||||
|             schema = self._get_schema(path) | ||||
| 
 | ||||
|         if self._is_leaf(schema): | ||||
|         if self._is_leaf_list(schema) and is_version(self._client, [(4, 5)]): | ||||
|             self._build_leaf_list(path, schema, value) | ||||
|         elif self._is_leaf(schema): | ||||
|             if self._is_empty_leaf(schema): | ||||
|                 exists = self._client.exists(path) | ||||
|                 if exists and value != [None]: | ||||
|  | @ -318,8 +320,16 @@ class ValueBuilder(object): | |||
|                 elif not exists and value == [None]: | ||||
|                     self._add_value(path, State.PRESENT, None) | ||||
|             else: | ||||
|                 value_type = self._get_type(parent, maybe_qname) | ||||
|                 if value_type == 'identityref': | ||||
|                 if maybe_qname is None: | ||||
|                     value_type = self._get_type(path) | ||||
|                 else: | ||||
|                     value_type = self._get_child_type(parent, maybe_qname) | ||||
| 
 | ||||
|                 if 'identityref' in value_type: | ||||
|                     if isinstance(value, list): | ||||
|                         value = [ll_v for ll_v, t_ll_v | ||||
|                                  in [self._get_prefix_name(v) for v in value]] | ||||
|                     else: | ||||
|                         value, t_value = self._get_prefix_name(value) | ||||
|                 self._add_value(path, State.SET, value) | ||||
|         elif isinstance(value, dict): | ||||
|  | @ -349,6 +359,18 @@ class ValueBuilder(object): | |||
|             child_schema = self._find_child(path, schema, qname) | ||||
|             self.build(path, dict_key, dict_value, child_schema) | ||||
| 
 | ||||
|     def _build_leaf_list(self, path, schema, value): | ||||
|         entry_type = self._get_type(path, schema) | ||||
|         # remove leaf list if treated as a list and then re-create the | ||||
|         # expected list entries. | ||||
|         self._add_value(path, State.ABSENT, None) | ||||
| 
 | ||||
|         for entry in value: | ||||
|             if 'identityref' in entry_type: | ||||
|                 entry, t_entry = self._get_prefix_name(entry) | ||||
|             entry_path = '{0}{{{1}}}'.format(path, entry) | ||||
|             self._add_value(entry_path, State.PRESENT, None) | ||||
| 
 | ||||
|     def _build_list(self, path, schema, value): | ||||
|         for entry in value: | ||||
|             entry_key = self._build_key(path, entry, schema['key']) | ||||
|  | @ -376,8 +398,8 @@ class ValueBuilder(object): | |||
|                     'required leaf {0} in {1} not set in data'.format( | ||||
|                         key, path)) | ||||
| 
 | ||||
|             value_type = self._get_type(path, key) | ||||
|             if value_type == 'identityref': | ||||
|             value_type = self._get_child_type(path, key) | ||||
|             if 'identityref' in value_type: | ||||
|                 value, t_value = self._get_prefix_name(value) | ||||
|             key_parts.append(self._quote_key(value)) | ||||
|         return ' '.join(key_parts) | ||||
|  | @ -422,8 +444,8 @@ class ValueBuilder(object): | |||
|         self._values_dirty = True | ||||
| 
 | ||||
|     def _get_prefix_name(self, qname): | ||||
|         if qname is None: | ||||
|             return None, None | ||||
|         if not isinstance(qname, (str, unicode)): | ||||
|             return qname, None | ||||
|         if ':' not in qname: | ||||
|             return qname, qname | ||||
| 
 | ||||
|  | @ -439,16 +461,23 @@ class ValueBuilder(object): | |||
|     def _get_schema(self, path): | ||||
|         return self._ensure_schema_cached(path)['data'] | ||||
| 
 | ||||
|     def _get_type(self, parent_path, key): | ||||
|     def _get_child_type(self, parent_path, key): | ||||
|         all_schema = self._ensure_schema_cached(parent_path) | ||||
|         parent_schema = all_schema['data'] | ||||
|         meta = all_schema['meta'] | ||||
| 
 | ||||
|         schema = self._find_child(parent_path, parent_schema, key) | ||||
|         return self._get_type(parent_path, schema, meta) | ||||
| 
 | ||||
|     def _get_type(self, path, schema=None, meta=None): | ||||
|         if schema is None or meta is None: | ||||
|             all_schema = self._ensure_schema_cached(path) | ||||
|             schema = all_schema['data'] | ||||
|             meta = all_schema['meta'] | ||||
| 
 | ||||
|         if self._is_leaf(schema): | ||||
|             def get_type(meta, curr_type): | ||||
|                 if curr_type.get('primitive', False): | ||||
|                     return curr_type['name'] | ||||
|                     return [curr_type['name']] | ||||
|                 if 'namespace' in curr_type: | ||||
|                     curr_type_key = '{0}:{1}'.format( | ||||
|                         curr_type['namespace'], curr_type['name']) | ||||
|  | @ -456,10 +485,14 @@ class ValueBuilder(object): | |||
|                     return get_type(meta, type_info) | ||||
|                 if 'leaf_type' in curr_type: | ||||
|                     return get_type(meta, curr_type['leaf_type'][-1]) | ||||
|                 return curr_type['name'] | ||||
|                 if 'union' in curr_type: | ||||
|                     union_types = [] | ||||
|                     for union_type in curr_type['union']: | ||||
|                         union_types.extend(get_type(meta, union_type[-1])) | ||||
|                     return union_types | ||||
|                 return [curr_type.get('name', 'unknown')] | ||||
| 
 | ||||
|             return get_type(meta, schema['type']) | ||||
| 
 | ||||
|         return None | ||||
| 
 | ||||
|     def _ensure_schema_cached(self, path): | ||||
|  | @ -502,7 +535,12 @@ class ValueBuilder(object): | |||
|                     return choice_child_schema | ||||
|         return None | ||||
| 
 | ||||
|     def _is_leaf_list(self, schema): | ||||
|         return schema.get('kind', None) == 'leaf-list' | ||||
| 
 | ||||
|     def _is_leaf(self, schema): | ||||
|         # still checking for leaf-list here to be compatible with pre | ||||
|         # 4.5 versions of NSO. | ||||
|         return schema.get('kind', None) in ('key', 'leaf', 'leaf-list') | ||||
| 
 | ||||
|     def _is_empty_leaf(self, schema): | ||||
|  | @ -531,6 +569,11 @@ def verify_version(client, required_versions=None): | |||
|                 version_str, supported_versions)) | ||||
| 
 | ||||
| 
 | ||||
| def is_version(client, required_versions): | ||||
|     version_str = client.get_system_setting('version') | ||||
|     return verify_version_str(version_str, required_versions) | ||||
| 
 | ||||
| 
 | ||||
| def verify_version_str(version_str, required_versions): | ||||
|     version = [int(p) for p in version_str.split('.')] | ||||
|     if len(version) < 2: | ||||
|  |  | |||
|  | @ -237,6 +237,38 @@ SCHEMA_DATA = { | |||
|         ] | ||||
|     } | ||||
| } | ||||
| ''', | ||||
|     '/test:test/device-list': ''' | ||||
| { | ||||
|     "meta": { | ||||
|         "types": { | ||||
|             "http://example.com/test:t15": [ | ||||
|                { | ||||
|                   "leaf_type":[ | ||||
|                      { | ||||
|                         "name":"string" | ||||
|                      } | ||||
|                   ], | ||||
|                   "list_type":[ | ||||
|                      { | ||||
|                         "name":"http://example.com/test:t15", | ||||
|                         "leaf-list":true | ||||
|                      } | ||||
|                   ] | ||||
|                } | ||||
|             ] | ||||
|         } | ||||
|     }, | ||||
|     "data": { | ||||
|         "kind":"leaf-list", | ||||
|         "name":"device-list", | ||||
|         "qname":"test:device-list", | ||||
|         "type": { | ||||
|            "namespace":"http://example.com/test", | ||||
|            "name":"t15" | ||||
|         } | ||||
|     } | ||||
| } | ||||
| ''' | ||||
| } | ||||
| 
 | ||||
|  | @ -342,7 +374,7 @@ class TestValueBuilder(unittest.TestCase): | |||
|             MockResponse('new_trans', {}, 200, '{"result": {"th": 1}}'), | ||||
|             get_schema_response('/an:id-name-values/id-name-value'), | ||||
|             MockResponse('get_module_prefix_map', {}, 200, '{{"result": {0}}}'.format(MODULE_PREFIX_MAP)), | ||||
|             MockResponse('exists', {'path': '/an:id-name-values/id-name-value{an:id-one}'}, 200, '{"result": {"exists": true}}'), | ||||
|             MockResponse('exists', {'path': '/an:id-name-values/id-name-value{an:id-one}'}, 200, '{"result": {"exists": true}}') | ||||
|         ] | ||||
|         open_url_mock.side_effect = lambda *args, **kwargs: mock_call(calls, *args, **kwargs) | ||||
| 
 | ||||
|  | @ -395,6 +427,7 @@ class TestValueBuilder(unittest.TestCase): | |||
|     @patch('ansible.module_utils.network.nso.nso.open_url') | ||||
|     def test_leaf_list_type(self, open_url_mock): | ||||
|         calls = [ | ||||
|             MockResponse('get_system_setting', {'operation': 'version'}, 200, '{"result": "4.4"}'), | ||||
|             MockResponse('new_trans', {}, 200, '{"result": {"th": 1}}'), | ||||
|             get_schema_response('/test:test') | ||||
|         ] | ||||
|  | @ -414,6 +447,35 @@ class TestValueBuilder(unittest.TestCase): | |||
| 
 | ||||
|         self.assertEqual(0, len(calls)) | ||||
| 
 | ||||
|     @patch('ansible.module_utils.network.nso.nso.open_url') | ||||
|     def test_leaf_list_type_45(self, open_url_mock): | ||||
|         calls = [ | ||||
|             MockResponse('get_system_setting', {'operation': 'version'}, 200, '{"result": "4.5"}'), | ||||
|             MockResponse('new_trans', {}, 200, '{"result": {"th": 1}}'), | ||||
|             get_schema_response('/test:test/device-list') | ||||
|         ] | ||||
|         open_url_mock.side_effect = lambda *args, **kwargs: mock_call(calls, *args, **kwargs) | ||||
| 
 | ||||
|         parent = "/test:test" | ||||
|         schema_data = json.loads( | ||||
|             SCHEMA_DATA['/test:test']) | ||||
|         schema = schema_data['data'] | ||||
| 
 | ||||
|         vb = nso.ValueBuilder(nso.JsonRpc('http://localhost:8080/jsonrpc')) | ||||
|         vb.build(parent, None, {'device-list': ['one', 'two']}, schema) | ||||
|         self.assertEquals(3, len(vb.values)) | ||||
|         value = vb.values[0] | ||||
|         self.assertEquals('{0}/device-list'.format(parent), value.path) | ||||
|         self.assertEquals(nso.State.ABSENT, value.state) | ||||
|         value = vb.values[1] | ||||
|         self.assertEquals('{0}/device-list{{one}}'.format(parent), value.path) | ||||
|         self.assertEquals(nso.State.PRESENT, value.state) | ||||
|         value = vb.values[2] | ||||
|         self.assertEquals('{0}/device-list{{two}}'.format(parent), value.path) | ||||
|         self.assertEquals(nso.State.PRESENT, value.state) | ||||
| 
 | ||||
|         self.assertEqual(0, len(calls)) | ||||
| 
 | ||||
| 
 | ||||
| class TestVerifyVersion(unittest.TestCase): | ||||
|     def test_valid_versions(self): | ||||
|  |  | |||
		Loading…
	
	Add table
		Add a link
		
	
		Reference in a new issue