mirror of
https://github.com/ansible-collections/community.general.git
synced 2024-09-14 20:13:21 +02:00
Add new filter to parse xml output for network use cases (#31562)
* Add new filter to parse xml output for network use cases Fixes #31026 * Add parse_xml filter * Add documentation for parse_xml filter * Edited for clarity. * Fix review comment and add unit tests * Fix unit test CI failure * Fix CI issues * Fix unit test failures * Fix review comments * More copy edits.
This commit is contained in:
parent
50c9f91060
commit
0ddf092ae3
8 changed files with 393 additions and 6 deletions
|
@ -345,8 +345,7 @@ output, use the ``parse_cli`` filter::
|
|||
{{ output | parse_cli('path/to/spec') }}
|
||||
|
||||
The ``parse_cli`` filter will load the spec file and pass the command output
|
||||
through, it returning JSON output. The spec file is a YAML yaml that defines
|
||||
how to parse the CLI output.
|
||||
through it, returning JSON output. The YAML spec file defines how to parse the CLI output.
|
||||
|
||||
The spec file should be valid formatted YAML. It defines how to parse the CLI
|
||||
output and return JSON data. Below is an example of a valid spec file that
|
||||
|
@ -362,7 +361,6 @@ will parse the output from the ``show vlan`` command.::
|
|||
|
||||
keys:
|
||||
vlans:
|
||||
type: list
|
||||
value: "{{ vlan }}"
|
||||
items: "^(?P<vlan_id>\\d+)\\s+(?P<name>\\w+)\\s+(?P<state>active|act/lshut|suspended)"
|
||||
state_static:
|
||||
|
@ -387,7 +385,6 @@ value using the same ``show vlan`` command.::
|
|||
|
||||
keys:
|
||||
vlans:
|
||||
type: list
|
||||
value: "{{ vlan }}"
|
||||
items: "^(?P<vlan_id>\\d+)\\s+(?P<name>\\w+)\\s+(?P<state>active|act/lshut|suspended)"
|
||||
state_static:
|
||||
|
@ -426,6 +423,101 @@ filter::
|
|||
|
||||
Use of the TextFSM filter requires the TextFSM library to be installed.
|
||||
|
||||
Network XML filters
|
||||
```````````````````
|
||||
|
||||
.. versionadded:: 2.5
|
||||
|
||||
To convert the XML output of a network device command into structured JSON
|
||||
output, use the ``parse_xml`` filter::
|
||||
|
||||
{{ output | parse_xml('path/to/spec') }}
|
||||
|
||||
The ``parse_xml`` filter will load the spec file and pass the command output
|
||||
through formatted as JSON.
|
||||
|
||||
The spec file should be valid formatted YAML. It defines how to parse the XML
|
||||
output and return JSON data.
|
||||
|
||||
Below is an example of a valid spec file that
|
||||
will parse the output from the ``show vlan | display xml`` command.::
|
||||
|
||||
---
|
||||
vars:
|
||||
vlan:
|
||||
vlan_id: "{{ item.vlan_id }}"
|
||||
name: "{{ item.name }}"
|
||||
desc: "{{ item.desc }}"
|
||||
enabled: "{{ item.state.get('inactive') != 'inactive' }}"
|
||||
state: "{% if item.state.get('inactive') == 'inactive'%} inactive {% else %} active {% endif %}"
|
||||
|
||||
keys:
|
||||
vlans:
|
||||
value: "{{ vlan }}"
|
||||
top: configuration/vlans/vlan
|
||||
items:
|
||||
vlan_id: vlan-id
|
||||
name: name
|
||||
desc: description
|
||||
state: ".[@inactive='inactive']"
|
||||
|
||||
The spec file above will return a JSON data structure that is a list of hashes
|
||||
with the parsed VLAN information.
|
||||
|
||||
The same command could be parsed into a hash by using the key and values
|
||||
directives. Here is an example of how to parse the output into a hash
|
||||
value using the same ``show vlan | display xml`` command.::
|
||||
|
||||
---
|
||||
vars:
|
||||
vlan:
|
||||
key: "{{ item.vlan_id }}"
|
||||
values:
|
||||
vlan_id: "{{ item.vlan_id }}"
|
||||
name: "{{ item.name }}"
|
||||
desc: "{{ item.desc }}"
|
||||
enabled: "{{ item.state.get('inactive') != 'inactive' }}"
|
||||
state: "{% if item.state.get('inactive') == 'inactive'%} inactive {% else %} active {% endif %}"
|
||||
|
||||
keys:
|
||||
vlans:
|
||||
value: "{{ vlan }}"
|
||||
top: configuration/vlans/vlan
|
||||
items:
|
||||
vlan_id: vlan-id
|
||||
name: name
|
||||
desc: description
|
||||
state: ".[@inactive='inactive']"
|
||||
|
||||
|
||||
The value of ``top`` is the XPath relative to the XML root node.
|
||||
In the example XML output given below, the value of ``top`` is ``configuration/vlans/vlan``,
|
||||
which is an XPath expression relative to the root node (<rpc-reply>).
|
||||
``configuration`` in the value of ``top`` is the outer most container node, and ``vlan``
|
||||
is the inner-most container node.
|
||||
|
||||
``items`` is a dictionary of key-value pairs that map user-defined names to XPath expressions
|
||||
that select elements. The Xpath expression is relative to the value of the XPath value contained in ``top``.
|
||||
For example, the ``vlan_id`` in the spec file is a user defined name and its value ``vlan-id`` is the
|
||||
relative to the value of XPath in ``top``
|
||||
|
||||
Attributes of XML tags can be extracted using XPath expressions. The value of ``state`` in the spec
|
||||
is an XPath expression used to get the attributes of the ``vlan`` tag in output XML.::
|
||||
|
||||
<rpc-reply>
|
||||
<configuration>
|
||||
<vlans>
|
||||
<vlan inactive="inactive">
|
||||
<name>vlan-1</name>
|
||||
<vlan-id>200</vlan-id>
|
||||
<description>This is vlan-1</description>
|
||||
</vlan>
|
||||
</vlans>
|
||||
</configuration>
|
||||
</rpc-reply>
|
||||
|
||||
.. note:: For more information on supported XPath expressions, see `<https://docs.python.org/2/library/xml.etree.elementtree.html#xpath-support>`_.
|
||||
|
||||
.. _hash_filters:
|
||||
|
||||
Hashing filters
|
||||
|
|
|
@ -22,9 +22,10 @@ __metaclass__ = type
|
|||
|
||||
import re
|
||||
import os
|
||||
import json
|
||||
import traceback
|
||||
|
||||
from collections import Mapping
|
||||
from xml.etree.ElementTree import fromstring
|
||||
|
||||
from ansible.module_utils.network_common import Template
|
||||
from ansible.module_utils.six import iteritems, string_types
|
||||
|
@ -242,12 +243,115 @@ def parse_cli_textfsm(value, template):
|
|||
return results
|
||||
|
||||
|
||||
def _extract_param(template, root, attrs, value):
|
||||
|
||||
key = None
|
||||
when = attrs.get('when')
|
||||
conditional = "{%% if %s %%}True{%% else %%}False{%% endif %%}" % when
|
||||
param_to_xpath_map = attrs['items']
|
||||
|
||||
if isinstance(value, Mapping):
|
||||
key = value.get('key', None)
|
||||
if key:
|
||||
value = value['values']
|
||||
|
||||
entries = dict() if key else list()
|
||||
|
||||
for element in root.findall(attrs['top']):
|
||||
entry = dict()
|
||||
item_dict = dict()
|
||||
for param, param_xpath in iteritems(param_to_xpath_map):
|
||||
fields = None
|
||||
try:
|
||||
fields = element.findall(param_xpath)
|
||||
except:
|
||||
display.warning("Failed to evaluate value of '%s' with XPath '%s'.\nUnexpected error: %s." % (param, param_xpath, traceback.format_exc()))
|
||||
|
||||
tags = param_xpath.split('/')
|
||||
|
||||
# check if xpath ends with attribute.
|
||||
# If yes set attribute key/value dict to param value in case attribute matches
|
||||
# else if it is a normal xpath assign matched element text value.
|
||||
if len(tags) and tags[-1].endswith(']'):
|
||||
if fields:
|
||||
if len(fields) > 1:
|
||||
item_dict[param] = [field.attrib for field in fields]
|
||||
else:
|
||||
item_dict[param] = fields[0].attrib
|
||||
else:
|
||||
item_dict[param] = {}
|
||||
else:
|
||||
if fields:
|
||||
if len(fields) > 1:
|
||||
item_dict[param] = [field.text for field in fields]
|
||||
else:
|
||||
item_dict[param] = fields[0].text
|
||||
else:
|
||||
item_dict[param] = None
|
||||
|
||||
if isinstance(value, Mapping):
|
||||
for item_key, item_value in iteritems(value):
|
||||
entry[item_key] = template(item_value, {'item': item_dict})
|
||||
else:
|
||||
entry = template(value, {'item': item_dict})
|
||||
|
||||
if key:
|
||||
expanded_key = template(key, {'item': item_dict})
|
||||
if when:
|
||||
if template(conditional, {'item': {'key': expanded_key, 'value': entry}}):
|
||||
entries[expanded_key] = entry
|
||||
else:
|
||||
entries[expanded_key] = entry
|
||||
else:
|
||||
if when:
|
||||
if template(conditional, {'item': entry}):
|
||||
entries.append(entry)
|
||||
else:
|
||||
entries.append(entry)
|
||||
|
||||
return entries
|
||||
|
||||
|
||||
def parse_xml(output, tmpl):
|
||||
if not os.path.exists(tmpl):
|
||||
raise AnsibleError('unable to locate parse_cli template: %s' % tmpl)
|
||||
|
||||
if not isinstance(output, string_types):
|
||||
raise AnsibleError('parse_xml works on string input, but given input of : %s' % type(output))
|
||||
|
||||
root = fromstring(output)
|
||||
try:
|
||||
template = Template()
|
||||
except ImportError as exc:
|
||||
raise AnsibleError(str(exc))
|
||||
|
||||
spec = yaml.safe_load(open(tmpl).read())
|
||||
obj = {}
|
||||
|
||||
for name, attrs in iteritems(spec['keys']):
|
||||
value = attrs['value']
|
||||
|
||||
try:
|
||||
variables = spec.get('vars', {})
|
||||
value = template(value, variables)
|
||||
except:
|
||||
pass
|
||||
|
||||
if 'items' in attrs:
|
||||
obj[name] = _extract_param(template, root, attrs, value)
|
||||
else:
|
||||
obj[name] = value
|
||||
|
||||
return obj
|
||||
|
||||
|
||||
class FilterModule(object):
|
||||
"""Filters for working with output from network devices"""
|
||||
|
||||
filter_map = {
|
||||
'parse_cli': parse_cli,
|
||||
'parse_cli_textfsm': parse_cli_textfsm
|
||||
'parse_cli_textfsm': parse_cli_textfsm,
|
||||
'parse_xml': parse_xml
|
||||
}
|
||||
|
||||
def filters(self):
|
||||
|
|
|
@ -0,0 +1,34 @@
|
|||
<rpc-reply>
|
||||
<configuration>
|
||||
<vlans>
|
||||
<vlan>
|
||||
<name>test-1</name>
|
||||
<vlan-id>100</vlan-id>
|
||||
</vlan>
|
||||
<vlan>
|
||||
<name>test-2</name>
|
||||
</vlan>
|
||||
<vlan>
|
||||
<name>test-3</name>
|
||||
<vlan-id>300</vlan-id>
|
||||
<description>test vlan-3</description>
|
||||
<interface>
|
||||
<name>em3.0</name>
|
||||
</interface>
|
||||
</vlan>
|
||||
<vlan inactive="inactive">
|
||||
<name>test-4</name>
|
||||
<description>test vlan-4</description>
|
||||
<vlan-id>400</vlan-id>
|
||||
</vlan>
|
||||
<vlan inactive="inactive">
|
||||
<name>test-5</name>
|
||||
<description>test vlan-5</description>
|
||||
<vlan-id>500</vlan-id>
|
||||
<interface>
|
||||
<name>em5.0</name>
|
||||
</interface>
|
||||
</vlan>
|
||||
</vlans>
|
||||
</configuration>
|
||||
</rpc-reply>
|
|
@ -0,0 +1,11 @@
|
|||
---
|
||||
vars:
|
||||
vlan: "{{ item.name }}"
|
||||
|
||||
keys:
|
||||
vlans:
|
||||
type: list
|
||||
value: "{{ vlan }}"
|
||||
top: configuration/vlans/vlan
|
||||
items:
|
||||
name: name
|
|
@ -0,0 +1,21 @@
|
|||
---
|
||||
vars:
|
||||
vlan:
|
||||
vlan_id: "{{ item.vlan_id }}"
|
||||
name: "{{ item.name }}"
|
||||
desc: "{{ item.desc }}"
|
||||
interface: "{{ item.intf }}"
|
||||
enabled: "{{ item.state.get('inactive') != 'inactive' }}"
|
||||
state: "{% if item.state.get('inactive') == 'inactive'%}inactive{% else %}active{% endif %}"
|
||||
|
||||
keys:
|
||||
vlans:
|
||||
type: list
|
||||
value: "{{ vlan }}"
|
||||
top: configuration/vlans/vlan
|
||||
items:
|
||||
vlan_id: vlan-id
|
||||
name: name
|
||||
desc: description
|
||||
intf: interface/name
|
||||
state: ".[@inactive='inactive']"
|
|
@ -0,0 +1,22 @@
|
|||
---
|
||||
vars:
|
||||
vlan:
|
||||
vlan_id: "{{ item.vlan_id }}"
|
||||
name: "{{ item.name }}"
|
||||
desc: "{{ item.desc }}"
|
||||
interface: "{{ item.intf }}"
|
||||
enabled: "{{ item.state.get('inactive') != 'inactive' }}"
|
||||
state: "{% if item.state.get('inactive') == 'inactive'%}inactive{% else %}active{% endif %}"
|
||||
|
||||
keys:
|
||||
vlans:
|
||||
type: list
|
||||
value: "{{ vlan }}"
|
||||
top: configuration/vlans/vlan
|
||||
items:
|
||||
vlan_id: vlan-id
|
||||
name: name
|
||||
desc: description
|
||||
intf: interface/name
|
||||
state: ".[@inactive='inactive']"
|
||||
when: item.name == 'test-5'
|
|
@ -0,0 +1,23 @@
|
|||
---
|
||||
vars:
|
||||
vlan:
|
||||
key: "{{ item.name }}"
|
||||
values:
|
||||
vlan_id: "{{ item.vlan_id }}"
|
||||
name: "{{ item.name }}"
|
||||
desc: "{{ item.desc }}"
|
||||
interface: "{{ item.intf }}"
|
||||
enabled: "{{ item.state.get('inactive') != 'inactive' }}"
|
||||
state: "{% if item.state.get('inactive') == 'inactive'%}inactive{% else %}active{% endif %}"
|
||||
|
||||
keys:
|
||||
vlans:
|
||||
type: list
|
||||
value: "{{ vlan }}"
|
||||
top: configuration/vlans/vlan
|
||||
items:
|
||||
vlan_id: vlan-id
|
||||
name: name
|
||||
desc: description
|
||||
intf: interface/name
|
||||
state: ".[@inactive='inactive']"
|
80
test/units/plugins/filter/test_network.py
Normal file
80
test/units/plugins/filter/test_network.py
Normal file
|
@ -0,0 +1,80 @@
|
|||
# This file is part of Ansible
|
||||
#
|
||||
# Ansible is free software: you can redistribute it and/or modify
|
||||
# it under the terms of the GNU General Public License as published by
|
||||
# the Free Software Foundation, either version 3 of the License, or
|
||||
# (at your option) any later version.
|
||||
#
|
||||
# Ansible is distributed in the hope that it will be useful,
|
||||
# but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||
# GNU General Public License for more details.
|
||||
#
|
||||
# You should have received a copy of the GNU General Public License
|
||||
# along with Ansible. If not, see <http://www.gnu.org/licenses/>.
|
||||
|
||||
# Make coding more python3-ish
|
||||
from __future__ import (absolute_import, division, print_function)
|
||||
__metaclass__ = type
|
||||
|
||||
import os
|
||||
import sys
|
||||
|
||||
from ansible.compat.tests import unittest
|
||||
from ansible.plugins.filter.network import parse_xml
|
||||
|
||||
fixture_path = os.path.join(os.path.dirname(__file__), 'fixtures', 'network')
|
||||
|
||||
with open(os.path.join(fixture_path, 'show_vlans_xml_output.txt')) as f:
|
||||
output_xml = f.read()
|
||||
|
||||
|
||||
class TestNetworkParseFilter(unittest.TestCase):
|
||||
|
||||
@unittest.skipIf(sys.version_info[:2] == (2, 6), 'XPath expression not supported in this version')
|
||||
def test_parse_xml_to_list_of_dict(self):
|
||||
spec_file_path = os.path.join(fixture_path, 'show_vlans_xml_spec.yml')
|
||||
parsed = parse_xml(output_xml, spec_file_path)
|
||||
expected = {'vlans': [{'name': 'test-1', 'enabled': True, 'state': 'active', 'interface': None, 'vlan_id': 100, 'desc': None},
|
||||
{'name': 'test-2', 'enabled': True, 'state': 'active', 'interface': None, 'vlan_id': None, 'desc': None},
|
||||
{'name': 'test-3', 'enabled': True, 'state': 'active', 'interface': 'em3.0', 'vlan_id': 300, 'desc': 'test vlan-3'},
|
||||
{'name': 'test-4', 'enabled': False, 'state': 'inactive', 'interface': None, 'vlan_id': 400, 'desc': 'test vlan-4'},
|
||||
{'name': 'test-5', 'enabled': False, 'state': 'inactive', 'interface': 'em5.0', 'vlan_id': 500, 'desc': 'test vlan-5'}]}
|
||||
self.assertEqual(parsed, expected)
|
||||
|
||||
@unittest.skipIf(sys.version_info[:2] == (2, 6), 'XPath expression not supported in this version')
|
||||
def test_parse_xml_to_dict(self):
|
||||
spec_file_path = os.path.join(fixture_path, 'show_vlans_xml_with_key_spec.yml')
|
||||
parsed = parse_xml(output_xml, spec_file_path)
|
||||
expected = {'vlans': {'test-4': {'name': 'test-4', 'enabled': False, 'state': 'inactive', 'interface': None, 'vlan_id': 400, 'desc': 'test vlan-4'},
|
||||
'test-3': {'name': 'test-3', 'enabled': True, 'state': 'active', 'interface': 'em3.0', 'vlan_id': 300, 'desc': 'test vlan-3'},
|
||||
'test-1': {'name': 'test-1', 'enabled': True, 'state': 'active', 'interface': None, 'vlan_id': 100, 'desc': None},
|
||||
'test-5': {'name': 'test-5', 'enabled': False, 'state': 'inactive', 'interface': 'em5.0', 'vlan_id': 500, 'desc': 'test vlan-5'},
|
||||
'test-2': {'name': 'test-2', 'enabled': True, 'state': 'active', 'interface': None, 'vlan_id': None, 'desc': None}}
|
||||
}
|
||||
self.assertEqual(parsed, expected)
|
||||
|
||||
@unittest.skipIf(sys.version_info[:2] == (2, 6), 'XPath expression not supported in this version')
|
||||
def test_parse_xml_with_condition_spec(self):
|
||||
spec_file_path = os.path.join(fixture_path, 'show_vlans_xml_with_condition_spec.yml')
|
||||
parsed = parse_xml(output_xml, spec_file_path)
|
||||
expected = {'vlans': [{'name': 'test-5', 'enabled': False, 'state': 'inactive', 'interface': 'em5.0', 'vlan_id': 500, 'desc': 'test vlan-5'}]}
|
||||
self.assertEqual(parsed, expected)
|
||||
|
||||
def test_parse_xml_with_single_value_spec(self):
|
||||
spec_file_path = os.path.join(fixture_path, 'show_vlans_xml_single_value_spec.yml')
|
||||
parsed = parse_xml(output_xml, spec_file_path)
|
||||
expected = {'vlans': ['test-1', 'test-2', 'test-3', 'test-4', 'test-5']}
|
||||
self.assertEqual(parsed, expected)
|
||||
|
||||
def test_parse_xml_validate_input(self):
|
||||
spec_file_path = os.path.join(fixture_path, 'show_vlans_xml_spec.yml')
|
||||
output = 10
|
||||
|
||||
with self.assertRaises(Exception) as e:
|
||||
parse_xml(output_xml, 'junk_path')
|
||||
self.assertEqual("unable to locate parse_cli template: junk_path", str(e.exception))
|
||||
|
||||
with self.assertRaises(Exception) as e:
|
||||
parse_xml(output, spec_file_path)
|
||||
self.assertEqual("parse_xml works on string input, but given input of : %s" % type(output), str(e.exception))
|
Loading…
Reference in a new issue