mirror of
https://github.com/ansible-collections/community.general.git
synced 2024-09-14 20:13:21 +02:00
255 lines
7.4 KiB
Python
255 lines
7.4 KiB
Python
|
#!/usr/bin/python
|
||
|
# -*- coding: utf-8 -*-
|
||
|
|
||
|
# (c) 2016-2018, Cumulus Networks <ce-ceng@cumulusnetworks.com>
|
||
|
# GNU General Public License v3.0+ (see COPYING or https://www.gnu.org/licenses/gpl-3.0.txt)
|
||
|
|
||
|
from __future__ import absolute_import, division, print_function
|
||
|
__metaclass__ = type
|
||
|
|
||
|
|
||
|
ANSIBLE_METADATA = {'metadata_version': '1.1',
|
||
|
'status': ['preview'],
|
||
|
'supported_by': 'community'}
|
||
|
|
||
|
DOCUMENTATION = '''
|
||
|
---
|
||
|
module: nclu
|
||
|
author: "Cumulus Networks (@isharacomix)"
|
||
|
short_description: Configure network interfaces using NCLU
|
||
|
description:
|
||
|
- Interface to the Network Command Line Utility, developed to make it easier
|
||
|
to configure operating systems running ifupdown2 and Quagga, such as
|
||
|
Cumulus Linux. Command documentation is available at
|
||
|
U(https://docs.cumulusnetworks.com/cumulus-linux/System-Configuration/Network-Command-Line-Utility-NCLU/)
|
||
|
options:
|
||
|
commands:
|
||
|
description:
|
||
|
- A list of strings containing the net commands to run. Mutually
|
||
|
exclusive with I(template).
|
||
|
template:
|
||
|
description:
|
||
|
- A single, multi-line string with jinja2 formatting. This string
|
||
|
will be broken by lines, and each line will be run through net.
|
||
|
Mutually exclusive with I(commands).
|
||
|
commit:
|
||
|
description:
|
||
|
- When true, performs a 'net commit' at the end of the block.
|
||
|
Mutually exclusive with I(atomic).
|
||
|
default: false
|
||
|
type: bool
|
||
|
abort:
|
||
|
description:
|
||
|
- Boolean. When true, perform a 'net abort' before the block.
|
||
|
This cleans out any uncommitted changes in the buffer.
|
||
|
Mutually exclusive with I(atomic).
|
||
|
default: false
|
||
|
type: bool
|
||
|
atomic:
|
||
|
description:
|
||
|
- When true, equivalent to both I(commit) and I(abort) being true.
|
||
|
Mutually exclusive with I(commit) and I(atomic).
|
||
|
default: false
|
||
|
type: bool
|
||
|
description:
|
||
|
description:
|
||
|
- Commit description that will be recorded to the commit log if
|
||
|
I(commit) or I(atomic) are true.
|
||
|
default: "Ansible-originated commit"
|
||
|
'''
|
||
|
|
||
|
EXAMPLES = '''
|
||
|
|
||
|
- name: Add two interfaces without committing any changes
|
||
|
nclu:
|
||
|
commands:
|
||
|
- add int swp1
|
||
|
- add int swp2
|
||
|
|
||
|
- name: Modify hostname to Cumulus-1 and commit the change
|
||
|
nclu:
|
||
|
commands:
|
||
|
- add hostname Cumulus-1
|
||
|
commit: true
|
||
|
|
||
|
- name: Add 48 interfaces and commit the change.
|
||
|
nclu:
|
||
|
template: |
|
||
|
{% for iface in range(1,49) %}
|
||
|
add int swp{{iface}}
|
||
|
{% endfor %}
|
||
|
commit: true
|
||
|
description: "Ansible - add swps1-48"
|
||
|
|
||
|
- name: Fetch Status Of Interface
|
||
|
nclu:
|
||
|
commands:
|
||
|
- show interface swp1
|
||
|
register: output
|
||
|
|
||
|
- name: Print Status Of Interface
|
||
|
debug:
|
||
|
var: output
|
||
|
|
||
|
- name: Fetch Details From All Interfaces In JSON Format
|
||
|
nclu:
|
||
|
commands:
|
||
|
- show interface json
|
||
|
register: output
|
||
|
|
||
|
- name: Print Interface Details
|
||
|
debug:
|
||
|
var: output["msg"]
|
||
|
|
||
|
- name: Atomically add an interface
|
||
|
nclu:
|
||
|
commands:
|
||
|
- add int swp1
|
||
|
atomic: true
|
||
|
description: "Ansible - add swp1"
|
||
|
|
||
|
- name: Remove IP address from interface swp1
|
||
|
nclu:
|
||
|
commands:
|
||
|
- del int swp1 ip address 1.1.1.1/24
|
||
|
|
||
|
- name: Configure BGP AS and add 2 EBGP neighbors using BGP Unnumbered
|
||
|
nclu:
|
||
|
commands:
|
||
|
- add bgp autonomous-system 65000
|
||
|
- add bgp neighbor swp51 interface remote-as external
|
||
|
- add bgp neighbor swp52 interface remote-as external
|
||
|
commit: true
|
||
|
|
||
|
- name: Configure BGP AS and Add 2 EBGP neighbors Using BGP Unnumbered via Template
|
||
|
nclu:
|
||
|
template: |
|
||
|
{% for neighbor in range(51,53) %}
|
||
|
add bgp neighbor swp{{neighbor}} interface remote-as external
|
||
|
add bgp autonomous-system 65000
|
||
|
{% endfor %}
|
||
|
atomic: true
|
||
|
|
||
|
- name: Check BGP Status
|
||
|
nclu:
|
||
|
commands:
|
||
|
- show bgp summary json
|
||
|
register: output
|
||
|
|
||
|
- name: Print BGP Status In JSON
|
||
|
debug:
|
||
|
var: output["msg"]
|
||
|
'''
|
||
|
|
||
|
RETURN = '''
|
||
|
changed:
|
||
|
description: whether the interface was changed
|
||
|
returned: changed
|
||
|
type: bool
|
||
|
sample: True
|
||
|
msg:
|
||
|
description: human-readable report of success or failure
|
||
|
returned: always
|
||
|
type: str
|
||
|
sample: "interface bond0 config updated"
|
||
|
'''
|
||
|
|
||
|
from ansible.module_utils.basic import AnsibleModule
|
||
|
|
||
|
|
||
|
def command_helper(module, command, errmsg=None):
|
||
|
"""Run a command, catch any nclu errors"""
|
||
|
(_rc, output, _err) = module.run_command("/usr/bin/net %s" % command)
|
||
|
if _rc or 'ERROR' in output or 'ERROR' in _err:
|
||
|
module.fail_json(msg=errmsg or output)
|
||
|
return str(output)
|
||
|
|
||
|
|
||
|
def check_pending(module):
|
||
|
"""Check the pending diff of the nclu buffer."""
|
||
|
pending = command_helper(module, "pending", "Error in pending config. You may want to view `net pending` on this target.")
|
||
|
|
||
|
delimeter1 = "net add/del commands since the last 'net commit'"
|
||
|
color1 = '\x1b[94m'
|
||
|
if delimeter1 in pending:
|
||
|
pending = pending.split(delimeter1)[0]
|
||
|
pending = pending.replace(color1, '')
|
||
|
return pending.strip()
|
||
|
|
||
|
|
||
|
def run_nclu(module, command_list, command_string, commit, atomic, abort, description):
|
||
|
_changed = False
|
||
|
|
||
|
commands = []
|
||
|
if command_list:
|
||
|
commands = command_list
|
||
|
elif command_string:
|
||
|
commands = command_string.splitlines()
|
||
|
|
||
|
do_commit = False
|
||
|
do_abort = abort
|
||
|
if commit or atomic:
|
||
|
do_commit = True
|
||
|
if atomic:
|
||
|
do_abort = True
|
||
|
|
||
|
if do_abort:
|
||
|
command_helper(module, "abort")
|
||
|
|
||
|
# First, look at the staged commands.
|
||
|
before = check_pending(module)
|
||
|
# Run all of the net commands
|
||
|
output_lines = []
|
||
|
for line in commands:
|
||
|
if line.strip():
|
||
|
output_lines += [command_helper(module, line.strip(), "Failed on line %s" % line)]
|
||
|
output = "\n".join(output_lines)
|
||
|
|
||
|
# If pending changes changed, report a change.
|
||
|
after = check_pending(module)
|
||
|
if before == after:
|
||
|
_changed = False
|
||
|
else:
|
||
|
_changed = True
|
||
|
|
||
|
# Do the commit.
|
||
|
if do_commit:
|
||
|
result = command_helper(module, "commit description '%s'" % description)
|
||
|
if "commit ignored" in result:
|
||
|
_changed = False
|
||
|
command_helper(module, "abort")
|
||
|
elif command_helper(module, "show commit last") == "":
|
||
|
_changed = False
|
||
|
|
||
|
return _changed, output
|
||
|
|
||
|
|
||
|
def main(testing=False):
|
||
|
module = AnsibleModule(argument_spec=dict(
|
||
|
commands=dict(required=False, type='list'),
|
||
|
template=dict(required=False, type='str'),
|
||
|
description=dict(required=False, type='str', default="Ansible-originated commit"),
|
||
|
abort=dict(required=False, type='bool', default=False),
|
||
|
commit=dict(required=False, type='bool', default=False),
|
||
|
atomic=dict(required=False, type='bool', default=False)),
|
||
|
mutually_exclusive=[('commands', 'template'),
|
||
|
('commit', 'atomic'),
|
||
|
('abort', 'atomic')]
|
||
|
)
|
||
|
command_list = module.params.get('commands', None)
|
||
|
command_string = module.params.get('template', None)
|
||
|
commit = module.params.get('commit')
|
||
|
atomic = module.params.get('atomic')
|
||
|
abort = module.params.get('abort')
|
||
|
description = module.params.get('description')
|
||
|
|
||
|
_changed, output = run_nclu(module, command_list, command_string, commit, atomic, abort, description)
|
||
|
if not testing:
|
||
|
module.exit_json(changed=_changed, msg=output)
|
||
|
elif testing:
|
||
|
return {"changed": _changed, "msg": output}
|
||
|
|
||
|
|
||
|
if __name__ == '__main__':
|
||
|
main()
|