mirror of
				https://github.com/ansible-collections/community.general.git
				synced 2025-10-25 21:44:00 -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: |         if schema is None: | ||||||
|             schema = self._get_schema(path) |             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): |             if self._is_empty_leaf(schema): | ||||||
|                 exists = self._client.exists(path) |                 exists = self._client.exists(path) | ||||||
|                 if exists and value != [None]: |                 if exists and value != [None]: | ||||||
|  | @ -318,8 +320,16 @@ class ValueBuilder(object): | ||||||
|                 elif not exists and value == [None]: |                 elif not exists and value == [None]: | ||||||
|                     self._add_value(path, State.PRESENT, None) |                     self._add_value(path, State.PRESENT, None) | ||||||
|             else: |             else: | ||||||
|                 value_type = self._get_type(parent, maybe_qname) |                 if maybe_qname is None: | ||||||
|                 if value_type == 'identityref': |                     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) |                         value, t_value = self._get_prefix_name(value) | ||||||
|                 self._add_value(path, State.SET, value) |                 self._add_value(path, State.SET, value) | ||||||
|         elif isinstance(value, dict): |         elif isinstance(value, dict): | ||||||
|  | @ -349,6 +359,18 @@ class ValueBuilder(object): | ||||||
|             child_schema = self._find_child(path, schema, qname) |             child_schema = self._find_child(path, schema, qname) | ||||||
|             self.build(path, dict_key, dict_value, child_schema) |             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): |     def _build_list(self, path, schema, value): | ||||||
|         for entry in value: |         for entry in value: | ||||||
|             entry_key = self._build_key(path, entry, schema['key']) |             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( |                     'required leaf {0} in {1} not set in data'.format( | ||||||
|                         key, path)) |                         key, path)) | ||||||
| 
 | 
 | ||||||
|             value_type = self._get_type(path, key) |             value_type = self._get_child_type(path, key) | ||||||
|             if value_type == 'identityref': |             if 'identityref' in value_type: | ||||||
|                 value, t_value = self._get_prefix_name(value) |                 value, t_value = self._get_prefix_name(value) | ||||||
|             key_parts.append(self._quote_key(value)) |             key_parts.append(self._quote_key(value)) | ||||||
|         return ' '.join(key_parts) |         return ' '.join(key_parts) | ||||||
|  | @ -422,8 +444,8 @@ class ValueBuilder(object): | ||||||
|         self._values_dirty = True |         self._values_dirty = True | ||||||
| 
 | 
 | ||||||
|     def _get_prefix_name(self, qname): |     def _get_prefix_name(self, qname): | ||||||
|         if qname is None: |         if not isinstance(qname, (str, unicode)): | ||||||
|             return None, None |             return qname, None | ||||||
|         if ':' not in qname: |         if ':' not in qname: | ||||||
|             return qname, qname |             return qname, qname | ||||||
| 
 | 
 | ||||||
|  | @ -439,16 +461,23 @@ class ValueBuilder(object): | ||||||
|     def _get_schema(self, path): |     def _get_schema(self, path): | ||||||
|         return self._ensure_schema_cached(path)['data'] |         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) |         all_schema = self._ensure_schema_cached(parent_path) | ||||||
|         parent_schema = all_schema['data'] |         parent_schema = all_schema['data'] | ||||||
|         meta = all_schema['meta'] |         meta = all_schema['meta'] | ||||||
| 
 |  | ||||||
|         schema = self._find_child(parent_path, parent_schema, key) |         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): |         if self._is_leaf(schema): | ||||||
|             def get_type(meta, curr_type): |             def get_type(meta, curr_type): | ||||||
|                 if curr_type.get('primitive', False): |                 if curr_type.get('primitive', False): | ||||||
|                     return curr_type['name'] |                     return [curr_type['name']] | ||||||
|                 if 'namespace' in curr_type: |                 if 'namespace' in curr_type: | ||||||
|                     curr_type_key = '{0}:{1}'.format( |                     curr_type_key = '{0}:{1}'.format( | ||||||
|                         curr_type['namespace'], curr_type['name']) |                         curr_type['namespace'], curr_type['name']) | ||||||
|  | @ -456,10 +485,14 @@ class ValueBuilder(object): | ||||||
|                     return get_type(meta, type_info) |                     return get_type(meta, type_info) | ||||||
|                 if 'leaf_type' in curr_type: |                 if 'leaf_type' in curr_type: | ||||||
|                     return get_type(meta, curr_type['leaf_type'][-1]) |                     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 get_type(meta, schema['type']) | ||||||
| 
 |  | ||||||
|         return None |         return None | ||||||
| 
 | 
 | ||||||
|     def _ensure_schema_cached(self, path): |     def _ensure_schema_cached(self, path): | ||||||
|  | @ -502,7 +535,12 @@ class ValueBuilder(object): | ||||||
|                     return choice_child_schema |                     return choice_child_schema | ||||||
|         return None |         return None | ||||||
| 
 | 
 | ||||||
|  |     def _is_leaf_list(self, schema): | ||||||
|  |         return schema.get('kind', None) == 'leaf-list' | ||||||
|  | 
 | ||||||
|     def _is_leaf(self, schema): |     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') |         return schema.get('kind', None) in ('key', 'leaf', 'leaf-list') | ||||||
| 
 | 
 | ||||||
|     def _is_empty_leaf(self, schema): |     def _is_empty_leaf(self, schema): | ||||||
|  | @ -531,6 +569,11 @@ def verify_version(client, required_versions=None): | ||||||
|                 version_str, supported_versions)) |                 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): | def verify_version_str(version_str, required_versions): | ||||||
|     version = [int(p) for p in version_str.split('.')] |     version = [int(p) for p in version_str.split('.')] | ||||||
|     if len(version) < 2: |     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}}'), |             MockResponse('new_trans', {}, 200, '{"result": {"th": 1}}'), | ||||||
|             get_schema_response('/an:id-name-values/id-name-value'), |             get_schema_response('/an:id-name-values/id-name-value'), | ||||||
|             MockResponse('get_module_prefix_map', {}, 200, '{{"result": {0}}}'.format(MODULE_PREFIX_MAP)), |             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) |         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') |     @patch('ansible.module_utils.network.nso.nso.open_url') | ||||||
|     def test_leaf_list_type(self, open_url_mock): |     def test_leaf_list_type(self, open_url_mock): | ||||||
|         calls = [ |         calls = [ | ||||||
|  |             MockResponse('get_system_setting', {'operation': 'version'}, 200, '{"result": "4.4"}'), | ||||||
|             MockResponse('new_trans', {}, 200, '{"result": {"th": 1}}'), |             MockResponse('new_trans', {}, 200, '{"result": {"th": 1}}'), | ||||||
|             get_schema_response('/test:test') |             get_schema_response('/test:test') | ||||||
|         ] |         ] | ||||||
|  | @ -414,6 +447,35 @@ class TestValueBuilder(unittest.TestCase): | ||||||
| 
 | 
 | ||||||
|         self.assertEqual(0, len(calls)) |         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): | class TestVerifyVersion(unittest.TestCase): | ||||||
|     def test_valid_versions(self): |     def test_valid_versions(self): | ||||||
|  |  | ||||||
		Loading…
	
	Add table
		Add a link
		
	
		Reference in a new issue