mirror of https://github.com/ansible/ansible.git
Contributing lib/ansible/modules/network/cloudengine/ce_vrf_interface.py module to manage HUAWEI data center CloudEngine (#22079)
* add ce_vrf_interface.py * metadata_version update * fix code review issues * fix ci_verified * tab to spacepull/24799/head
parent
cb75f2a43d
commit
a8475f6ef5
@ -0,0 +1,516 @@
|
||||
#!/usr/bin/python
|
||||
#
|
||||
# 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/>.
|
||||
#
|
||||
|
||||
ANSIBLE_METADATA = {'status': ['preview'],
|
||||
'supported_by': 'community',
|
||||
'metadata_version': '1.0'}
|
||||
|
||||
DOCUMENTATION = '''
|
||||
---
|
||||
module: ce_vrf_interface
|
||||
version_added: "2.4"
|
||||
short_description: Manages interface specific VPN configuration on HUAWEI CloudEngine switches.
|
||||
description:
|
||||
- Manages interface specific VPN configuration of HUAWEI CloudEngine switches.
|
||||
author: Zhijin Zhou (@CloudEngine-Ansible)
|
||||
notes:
|
||||
- Ensure that a VPN instance has been created and the IPv4 address family has been enabled for the VPN instance.
|
||||
options:
|
||||
vrf:
|
||||
description:
|
||||
- VPN instance, the length of vrf name is 1 ~ 31, i.e. "test", but can not be C(_public_).
|
||||
required: true
|
||||
vpn_interface:
|
||||
description:
|
||||
- An interface that can binding VPN instance, i.e. 40GE1/0/22, Vlanif10.
|
||||
Must be fully qualified interface name.
|
||||
Interface types, such as 10GE, 40GE, 100GE, LoopBack, MEth, Tunnel, Vlanif....
|
||||
required: true
|
||||
state:
|
||||
description:
|
||||
- Manage the state of the resource.
|
||||
required: false
|
||||
choices: ['present','absent']
|
||||
default: present
|
||||
'''
|
||||
|
||||
EXAMPLES = '''
|
||||
- name: VRF interface test
|
||||
hosts: cloudengine
|
||||
connection: local
|
||||
gather_facts: no
|
||||
vars:
|
||||
cli:
|
||||
host: "{{ inventory_hostname }}"
|
||||
port: "{{ ansible_ssh_port }}"
|
||||
username: "{{ username }}"
|
||||
password: "{{ password }}"
|
||||
transport: cli
|
||||
|
||||
tasks:
|
||||
|
||||
- name: "Configure a VPN instance for the interface"
|
||||
ce_vrf_interface:
|
||||
vpn_interface: 40GE1/0/2
|
||||
vrf: test
|
||||
state: present
|
||||
provider: "{{ cli }}"
|
||||
|
||||
- name: "Disable the association between a VPN instance and an interface"
|
||||
ce_vrf_interface:
|
||||
vpn_interface: 40GE1/0/2
|
||||
vrf: test
|
||||
state: absent
|
||||
provider: "{{ cli }}"
|
||||
'''
|
||||
|
||||
RETURN = '''
|
||||
proposed:
|
||||
description: k/v pairs of parameters passed into module
|
||||
returned: verbose mode
|
||||
type: dict
|
||||
sample: {
|
||||
"state": "present",
|
||||
"vpn_interface": "40GE2/0/17",
|
||||
"vrf": "jss"
|
||||
}
|
||||
existing:
|
||||
description: k/v pairs of existing attributes on the interface
|
||||
returned: verbose mode
|
||||
type: dict
|
||||
sample: {
|
||||
"vpn_interface": "40GE2/0/17",
|
||||
"vrf": null
|
||||
}
|
||||
end_state:
|
||||
description: k/v pairs of end attributes on the interface
|
||||
returned: verbose mode
|
||||
type: dict
|
||||
sample: {
|
||||
"vpn_interface": "40GE2/0/17",
|
||||
"vrf": "jss"
|
||||
}
|
||||
updates:
|
||||
description: command list sent to the device
|
||||
returned: always
|
||||
type: list
|
||||
sample: [
|
||||
"ip binding vpn-instance jss",
|
||||
]
|
||||
changed:
|
||||
description: check to see if a change was made on the device
|
||||
returned: always
|
||||
type: boolean
|
||||
sample: true
|
||||
'''
|
||||
|
||||
|
||||
from xml.etree import ElementTree
|
||||
from ansible.module_utils.basic import AnsibleModule
|
||||
from ansible.module_utils.ce import ce_argument_spec, get_nc_config, set_nc_config
|
||||
|
||||
CE_NC_GET_VRF = """
|
||||
<filter type="subtree">
|
||||
<l3vpn xmlns="http://www.huawei.com/netconf/vrp" content-version="1.0" format-version="1.0">
|
||||
<l3vpncomm>
|
||||
<l3vpnInstances>
|
||||
<l3vpnInstance>
|
||||
<vrfName>%s</vrfName>
|
||||
</l3vpnInstance>
|
||||
</l3vpnInstances>
|
||||
</l3vpncomm>
|
||||
</l3vpn>
|
||||
</filter>
|
||||
"""
|
||||
|
||||
CE_NC_GET_VRF_INTERFACE = """
|
||||
<filter type="subtree">
|
||||
<l3vpn xmlns="http://www.huawei.com/netconf/vrp" content-version="1.0" format-version="1.0">
|
||||
<l3vpncomm>
|
||||
<l3vpnInstances>
|
||||
<l3vpnInstance>
|
||||
<vrfName></vrfName>
|
||||
<l3vpnIfs>
|
||||
<l3vpnIf>
|
||||
<ifName></ifName>
|
||||
</l3vpnIf>
|
||||
</l3vpnIfs>
|
||||
</l3vpnInstance>
|
||||
</l3vpnInstances>
|
||||
</l3vpncomm>
|
||||
</l3vpn>
|
||||
</filter>
|
||||
"""
|
||||
|
||||
CE_NC_MERGE_VRF_INTERFACE = """
|
||||
<config>
|
||||
<l3vpn xmlns="http://www.huawei.com/netconf/vrp" content-version="1.0" format-version="1.0">
|
||||
<l3vpncomm>
|
||||
<l3vpnInstances>
|
||||
<l3vpnInstance>
|
||||
<vrfName>%s</vrfName>
|
||||
<l3vpnIfs>
|
||||
<l3vpnIf operation="merge">
|
||||
<ifName>%s</ifName>
|
||||
</l3vpnIf>
|
||||
</l3vpnIfs>
|
||||
</l3vpnInstance>
|
||||
</l3vpnInstances>
|
||||
</l3vpncomm>
|
||||
</l3vpn>
|
||||
</config>
|
||||
"""
|
||||
|
||||
CE_NC_GET_INTF = """
|
||||
<filter type="subtree">
|
||||
<ifm xmlns="http://www.huawei.com/netconf/vrp" content-version="1.0" format-version="1.0">
|
||||
<interfaces>
|
||||
<interface>
|
||||
<ifName>%s</ifName>
|
||||
<isL2SwitchPort></isL2SwitchPort>
|
||||
</interface>
|
||||
</interfaces>
|
||||
</ifm>
|
||||
</filter>
|
||||
"""
|
||||
|
||||
CE_NC_DEL_INTF_VPN = """
|
||||
<config>
|
||||
<l3vpn xmlns="http://www.huawei.com/netconf/vrp" content-version="1.0" format-version="1.0">
|
||||
<l3vpncomm>
|
||||
<l3vpnInstances>
|
||||
<l3vpnInstance>
|
||||
<vrfName>%s</vrfName>
|
||||
<l3vpnIfs>
|
||||
<l3vpnIf operation="delete">
|
||||
<ifName>%s</ifName>
|
||||
</l3vpnIf>
|
||||
</l3vpnIfs>
|
||||
</l3vpnInstance>
|
||||
</l3vpnInstances>
|
||||
</l3vpncomm>
|
||||
</l3vpn>
|
||||
</config>
|
||||
"""
|
||||
|
||||
|
||||
def get_interface_type(interface):
|
||||
"""Gets the type of interface, such as 10GE, ETH-TRUNK, VLANIF..."""
|
||||
|
||||
if interface is None:
|
||||
return None
|
||||
|
||||
iftype = None
|
||||
|
||||
if interface.upper().startswith('GE'):
|
||||
iftype = 'ge'
|
||||
elif interface.upper().startswith('10GE'):
|
||||
iftype = '10ge'
|
||||
elif interface.upper().startswith('25GE'):
|
||||
iftype = '25ge'
|
||||
elif interface.upper().startswith('4X10GE'):
|
||||
iftype = '4x10ge'
|
||||
elif interface.upper().startswith('40GE'):
|
||||
iftype = '40ge'
|
||||
elif interface.upper().startswith('100GE'):
|
||||
iftype = '100ge'
|
||||
elif interface.upper().startswith('VLANIF'):
|
||||
iftype = 'vlanif'
|
||||
elif interface.upper().startswith('LOOPBACK'):
|
||||
iftype = 'loopback'
|
||||
elif interface.upper().startswith('METH'):
|
||||
iftype = 'meth'
|
||||
elif interface.upper().startswith('ETH-TRUNK'):
|
||||
iftype = 'eth-trunk'
|
||||
elif interface.upper().startswith('VBDIF'):
|
||||
iftype = 'vbdif'
|
||||
elif interface.upper().startswith('NVE'):
|
||||
iftype = 'nve'
|
||||
elif interface.upper().startswith('TUNNEL'):
|
||||
iftype = 'tunnel'
|
||||
elif interface.upper().startswith('ETHERNET'):
|
||||
iftype = 'ethernet'
|
||||
elif interface.upper().startswith('FCOE-PORT'):
|
||||
iftype = 'fcoe-port'
|
||||
elif interface.upper().startswith('FABRIC-PORT'):
|
||||
iftype = 'fabric-port'
|
||||
elif interface.upper().startswith('STACK-PORT'):
|
||||
iftype = 'stack-Port'
|
||||
elif interface.upper().startswith('NULL'):
|
||||
iftype = 'null'
|
||||
else:
|
||||
return None
|
||||
|
||||
return iftype.lower()
|
||||
|
||||
|
||||
class VrfInterface(object):
|
||||
"""Manange vpn instance"""
|
||||
|
||||
def __init__(self, argument_spec):
|
||||
self.spec = argument_spec
|
||||
self.module = None
|
||||
self.init_module()
|
||||
|
||||
# vpn instance info
|
||||
self.vrf = self.module.params['vrf']
|
||||
self.vpn_interface = self.module.params['vpn_interface']
|
||||
self.vpn_interface = self.vpn_interface.upper().replace(' ', '')
|
||||
self.state = self.module.params['state']
|
||||
self.intf_info = dict()
|
||||
self.intf_info['isL2SwitchPort'] = None
|
||||
self.intf_info['vrfName'] = None
|
||||
self.conf_exist = False
|
||||
|
||||
# state
|
||||
self.changed = False
|
||||
self.updates_cmd = list()
|
||||
self.results = dict()
|
||||
self.proposed = dict()
|
||||
self.existing = dict()
|
||||
self.end_state = dict()
|
||||
|
||||
def init_module(self):
|
||||
"""init_module"""
|
||||
|
||||
required_one_of = [("vrf", "vpn_interface")]
|
||||
self.module = AnsibleModule(
|
||||
argument_spec=self.spec, required_one_of=required_one_of, supports_check_mode=True)
|
||||
|
||||
def check_response(self, xml_str, xml_name):
|
||||
"""Check if response message is already succeed."""
|
||||
|
||||
if "<ok/>" not in xml_str:
|
||||
self.module.fail_json(msg='Error: %s failed.' % xml_name)
|
||||
|
||||
def get_update_cmd(self):
|
||||
""" get updated command"""
|
||||
|
||||
if self.conf_exist:
|
||||
return
|
||||
|
||||
if self.state == 'absent':
|
||||
self.updates_cmd.append(
|
||||
"undo ip binding vpn-instance %s" % self.vrf)
|
||||
return
|
||||
|
||||
if self.vrf != self.intf_info['vrfName']:
|
||||
self.updates_cmd.append("ip binding vpn-instance %s" % self.vrf)
|
||||
|
||||
return
|
||||
|
||||
def check_params(self):
|
||||
"""Check all input params"""
|
||||
|
||||
if not self.is_vrf_exist():
|
||||
self.module.fail_json(
|
||||
msg='Error: The VPN instance is not existed.')
|
||||
|
||||
if self.state == 'absent':
|
||||
if self.vrf != self.intf_info['vrfName']:
|
||||
self.module.fail_json(
|
||||
msg='Error: The VPN instance is not bound to the interface.')
|
||||
|
||||
if self.intf_info['isL2SwitchPort'] == 'true':
|
||||
self.module.fail_json(
|
||||
msg='Error: L2Switch Port can not binding a VPN instance.')
|
||||
|
||||
# interface type check
|
||||
if self.vpn_interface:
|
||||
intf_type = get_interface_type(self.vpn_interface)
|
||||
if not intf_type:
|
||||
self.module.fail_json(
|
||||
msg='Error: interface name of %s'
|
||||
' is error.' % self.vpn_interface)
|
||||
|
||||
# vrf check
|
||||
if self.vrf == '_public_':
|
||||
self.module.fail_json(
|
||||
msg='Error: The vrf name _public_ is reserved.')
|
||||
if len(self.vrf) < 1 or len(self.vrf) > 31:
|
||||
self.module.fail_json(
|
||||
msg='Error: The vrf name length must be between 1 and 31.')
|
||||
|
||||
def get_interface_vpn_name(self, vpninfo, vpn_name):
|
||||
""" get vpn instance name"""
|
||||
|
||||
l3vpn_if = vpninfo.findall("l3vpnIf")
|
||||
for l3vpn_ifinfo in l3vpn_if:
|
||||
for ele in l3vpn_ifinfo:
|
||||
if ele.tag in ['ifName']:
|
||||
if ele.text == self.vpn_interface:
|
||||
self.intf_info['vrfName'] = vpn_name
|
||||
|
||||
def get_interface_vpn(self):
|
||||
""" get the VPN instance associated with the interface"""
|
||||
|
||||
xml_str = CE_NC_GET_VRF_INTERFACE
|
||||
con_obj = get_nc_config(self.module, xml_str)
|
||||
if "<data/>" in con_obj:
|
||||
return
|
||||
|
||||
xml_str = con_obj.replace('\r', '').replace('\n', '').\
|
||||
replace('xmlns="urn:ietf:params:xml:ns:netconf:base:1.0"', "").\
|
||||
replace('xmlns="http://www.huawei.com/netconf/vrp"', "")
|
||||
|
||||
# get global vrf interface info
|
||||
root = ElementTree.fromstring(xml_str)
|
||||
vpns = root.findall(
|
||||
"data/l3vpn/l3vpncomm/l3vpnInstances/l3vpnInstance")
|
||||
if vpns:
|
||||
for vpnele in vpns:
|
||||
vpn_name = None
|
||||
for vpninfo in vpnele:
|
||||
if vpninfo.tag == 'vrfName':
|
||||
vpn_name = vpninfo.text
|
||||
|
||||
if vpninfo.tag == 'l3vpnIfs':
|
||||
self.get_interface_vpn_name(vpninfo, vpn_name)
|
||||
|
||||
return
|
||||
|
||||
def is_vrf_exist(self):
|
||||
""" judge whether the VPN instance is existed"""
|
||||
|
||||
conf_str = CE_NC_GET_VRF % self.vrf
|
||||
con_obj = get_nc_config(self.module, conf_str)
|
||||
if "<data/>" in con_obj:
|
||||
return False
|
||||
|
||||
return True
|
||||
|
||||
def get_intf_conf_info(self):
|
||||
""" get related configuration of the interface"""
|
||||
|
||||
conf_str = CE_NC_GET_INTF % self.vpn_interface
|
||||
con_obj = get_nc_config(self.module, conf_str)
|
||||
if "<data/>" in con_obj:
|
||||
return
|
||||
|
||||
# get interface base info
|
||||
xml_str = con_obj.replace('\r', '').replace('\n', '').\
|
||||
replace('xmlns="urn:ietf:params:xml:ns:netconf:base:1.0"', "").\
|
||||
replace('xmlns="http://www.huawei.com/netconf/vrp"', "")
|
||||
|
||||
root = ElementTree.fromstring(xml_str)
|
||||
interface = root.find("data/ifm/interfaces/interface")
|
||||
if interface:
|
||||
for eles in interface:
|
||||
if eles.tag in ["isL2SwitchPort"]:
|
||||
self.intf_info[eles.tag] = eles.text
|
||||
|
||||
self.get_interface_vpn()
|
||||
return
|
||||
|
||||
def get_existing(self):
|
||||
"""get existing config"""
|
||||
|
||||
self.existing = dict(vrf=self.intf_info['vrfName'],
|
||||
vpn_interface=self.vpn_interface)
|
||||
|
||||
def get_proposed(self):
|
||||
"""get_proposed"""
|
||||
|
||||
self.proposed = dict(vrf=self.vrf,
|
||||
vpn_interface=self.vpn_interface,
|
||||
state=self.state)
|
||||
|
||||
def get_end_state(self):
|
||||
"""get_end_state"""
|
||||
|
||||
self.intf_info['vrfName'] = None
|
||||
self.get_intf_conf_info()
|
||||
|
||||
self.end_state = dict(vrf=self.intf_info['vrfName'],
|
||||
vpn_interface=self.vpn_interface)
|
||||
|
||||
def show_result(self):
|
||||
""" show result"""
|
||||
|
||||
self.results['changed'] = self.changed
|
||||
self.results['proposed'] = self.proposed
|
||||
self.results['existing'] = self.existing
|
||||
self.results['end_state'] = self.end_state
|
||||
if self.changed:
|
||||
self.results['updates'] = self.updates_cmd
|
||||
else:
|
||||
self.results['updates'] = list()
|
||||
|
||||
self.module.exit_json(**self.results)
|
||||
|
||||
def judge_if_config_exist(self):
|
||||
""" judge whether configuration has existed"""
|
||||
|
||||
if self.state == 'absent':
|
||||
return False
|
||||
|
||||
delta = set(self.proposed.items()).difference(
|
||||
self.existing.items())
|
||||
delta = dict(delta)
|
||||
if len(delta) == 1 and delta['state']:
|
||||
return True
|
||||
|
||||
return False
|
||||
|
||||
def config_interface_vrf(self):
|
||||
""" configure VPN instance of the interface"""
|
||||
|
||||
if not self.conf_exist and self.state == 'present':
|
||||
|
||||
xml_str = CE_NC_MERGE_VRF_INTERFACE % (
|
||||
self.vrf, self.vpn_interface)
|
||||
ret_xml = set_nc_config(self.module, xml_str)
|
||||
self.check_response(ret_xml, "VRF_INTERFACE_CONFIG")
|
||||
self.changed = True
|
||||
elif self.state == 'absent':
|
||||
xml_str = CE_NC_DEL_INTF_VPN % (self.vrf, self.vpn_interface)
|
||||
ret_xml = set_nc_config(self.module, xml_str)
|
||||
self.check_response(ret_xml, "DEL_VRF_INTERFACE_CONFIG")
|
||||
self.changed = True
|
||||
|
||||
def work(self):
|
||||
"""excute task"""
|
||||
|
||||
self.get_intf_conf_info()
|
||||
self.check_params()
|
||||
self.get_existing()
|
||||
self.get_proposed()
|
||||
self.conf_exist = self.judge_if_config_exist()
|
||||
|
||||
self.config_interface_vrf()
|
||||
|
||||
self.get_update_cmd()
|
||||
self.get_end_state()
|
||||
self.show_result()
|
||||
|
||||
|
||||
def main():
|
||||
"""main"""
|
||||
|
||||
argument_spec = dict(
|
||||
vrf=dict(required=True, type='str'),
|
||||
vpn_interface=dict(required=True, type='str'),
|
||||
state=dict(choices=['absent', 'present'],
|
||||
default='present', required=False),
|
||||
)
|
||||
argument_spec.update(ce_argument_spec)
|
||||
vrf_intf = VrfInterface(argument_spec)
|
||||
vrf_intf.work()
|
||||
|
||||
if __name__ == '__main__':
|
||||
main()
|
Loading…
Reference in New Issue