update iosxr_config with new arguments

* add src argument to provide path to config file
* add new choice to match used to ignore current running config
* add update argument with choices merge, replace or check
* add backup argument to backup current running config to control host
* add comment argument to provide comment to commit
* deprecated force argument, use match=none instead
This commit is contained in:
Peter Sprygada 2016-07-13 14:48:28 -07:00 committed by Matt Clay
parent c4988262b5
commit d138b94c70

View file

@ -26,9 +26,8 @@ description:
- Cisco IOS XR configurations use a simple block indent file syntax - Cisco IOS XR configurations use a simple block indent file syntax
for segmenting configuration into sections. This module provides for segmenting configuration into sections. This module provides
an implementation for working with IOS XR configuration sections in an implementation for working with IOS XR configuration sections in
a deterministic way. This module works with either CLI or NXAPI a deterministic way.
transports. extends_documentation_fragment: iosxr
extends_documentation_fragment: ios
options: options:
lines: lines:
description: description:
@ -37,7 +36,9 @@ options:
in the device running-config. Be sure to note the configuration in the device running-config. Be sure to note the configuration
command syntax as some commands are automatically modified by the command syntax as some commands are automatically modified by the
device config parser. device config parser.
required: true required: false
default: null
aliases: ['commands']
parents: parents:
description: description:
- The ordered set of parents that uniquely identify the section - The ordered set of parents that uniquely identify the section
@ -46,6 +47,16 @@ options:
level or global commands. level or global commands.
required: false required: false
default: null default: null
src:
description:
- Specifies the source path to the file that contains the configuration
or configuration template to load. The path to the source file can
either be the full path on the Ansible control host or a relative
path from the playbook or role root directory. This argument is mutually
exclusive with I(lines).
required: false
default: null
version_added: "2.2"
before: before:
description: description:
- The ordered set of commands to push on to the command stack if - The ordered set of commands to push on to the command stack if
@ -69,11 +80,13 @@ options:
the set of commands against the current device config. If the set of commands against the current device config. If
match is set to I(line), commands are matched line by line. If match is set to I(line), commands are matched line by line. If
match is set to I(strict), command lines are matched with respect match is set to I(strict), command lines are matched with respect
to position. Finally if match is set to I(exact), command lines to position. If match is set to I(exact), command lines
must be an equal match. must be an equal match. Finally, if match is set to I(none), the
module will not attempt to compare the source configuration with
the running configuration on the remote device.
required: false required: false
default: line default: line
choices: ['line', 'strict', 'exact'] choices: ['line', 'strict', 'exact', 'none']
replace: replace:
description: description:
- Instructs the module on the way to perform the configuration - Instructs the module on the way to perform the configuration
@ -91,9 +104,26 @@ options:
current devices running-config. When set to true, this will current devices running-config. When set to true, this will
cause the module to push the contents of I(src) into the device cause the module to push the contents of I(src) into the device
without first checking if already configured. without first checking if already configured.
- Note this argument should be considered deprecated. To achieve
the equivalent, set the match argument to none. This argument
will be removed in a future release.
required: false required: false
default: false default: false
choices: [ "true", "false" ] choices: [ "yes", "no" ]
version_added: "2.2"
update:
description:
- The I(update) argument controls how the configuration statements
are processed on the remote device. Valid choices for the I(update)
argument are I(merge) and I(check). When the argument is set to
I(merge), the configuration changes are merged with the current
device running configuration. When the argument is set to I(check)
the configuration updates are determined but not actually configured
on the remote device.
required: false
default: merge
choices: ['merge', 'replace', 'check']
version_added: "2.2"
config: config:
description: description:
- The module, by default, will connect to the remote device and - The module, by default, will connect to the remote device and
@ -105,25 +135,56 @@ options:
config for comparison. config for comparison.
required: false required: false
default: null default: null
backup:
description:
- This argument will cause the module to create a full backup of
the current C(running-config) from the remote device before any
changes are made. The backup file is written to the C(backup)
folder in the playbook root directory. If the directory does not
exist, it is created.
required: false
default: no
choices: ['yes', 'no']
version_added: "2.2"
comment:
description:
- Allows a commit description to be specified to be included
when the configuration is committed. If the configuration is
not changed or committed, this argument is ignored.
required: false
default: 'configured by iosxr_config'
version_added: "2.2"
""" """
EXAMPLES = """ EXAMPLES = """
- iosxr_config: # Note: examples below use the following provider dict to handle
lines: ['hostname {{ inventory_hostname }}'] # transport and authentication to the node.
force: yes vars:
cli:
host: "{{ inventory_hostname }}"
username: cisco
password: cisco
transport: cli
- iosxr_config: - name: configure top level configuration
iosxr_config:
lines: hostname {{ inventory_hostname }}
provider: "{{ cli }}"
- name: configure interface settings
iosxr_config:
lines: lines:
- description configured by ansible - description test interface
- ipv4 address 10.0.0.25 255.255.255.0 - ip address 172.31.1.1 255.255.255.0
parents: ['interface GigabitEthernet0/0/0/0'] parents: interface GigabitEthernet0/0/0/0
provider: "{{ cli }}"
- iosxr_config:
commands: "{{lookup('file', 'datcenter1.txt')}}"
parents: ['ipv4 access-list test']
before: ['no ip access-listv4 test']
replace: block
- name: load a config from disk and replace the current config
iosxr_config:
src: config.cfg
update: replace
backup: yes
provider: "{{ cli }}"
""" """
RETURN = """ RETURN = """
@ -132,82 +193,144 @@ updates:
returned: always returned: always
type: list type: list
sample: ['...', '...'] sample: ['...', '...']
backup_path:
responses: description: The full path to the backup file
description: The set of responses from issuing the commands on the device returned: when backup is yes
retured: always type: path
type: list sample: /playbooks/ansible/backup/iosxr01.2016-07-16@22:28:34
sample: ['...', '...']
""" """
from ansible.module_utils.basic import get_exception
from ansible.module_utils.netcfg import NetworkConfig, dumps
from ansible.module_utils.iosxr import NetworkModule, NetworkError
def get_config(module): DEFAULT_COMMIT_COMMENT = 'configured by iosxr_config'
config = module.params['config'] or dict()
if not config and not module.params['force']:
config = module.config
return config
def main():
argument_spec = dict( def invoke(name, *args, **kwargs):
lines=dict(aliases=['commands'], required=True, type='list'), func = globals().get(name)
parents=dict(type='list'), if func:
before=dict(type='list'), return func(*args, **kwargs)
after=dict(type='list'),
match=dict(default='line', choices=['line', 'strict', 'exact']),
replace=dict(default='line', choices=['line', 'block']),
force=dict(default=False, type='bool'),
config=dict()
)
module = get_module(argument_spec=argument_spec, def check_args(module, warnings):
supports_check_mode=True) if module.params['parents']:
if not module.params['lines'] or module.params['src']:
warnings.append('ignoring unnecessary argument parents')
if module.params['match'] == 'none' and module.params['replace']:
warnings.append('ignoring unnecessary argument replace')
if module.params['update'] == 'replace' and not module.params['src']:
module.fail_json(msg='Must specify src when update is `replace`')
if module.params['force']:
warnings.append('The force argument is deprecated, please use '
'match=none instead. This argument will be '
'removed in the future')
lines = module.params['lines']
parents = module.params['parents'] or list()
before = module.params['before'] def get_config(module, result):
after = module.params['after'] contents = module.params['config'] or result.get('__config__')
if not contents:
contents = module.config.get_config()
return NetworkConfig(indent=1, contents=contents)
def get_candidate(module):
candidate = NetworkConfig(indent=1)
if module.params['src']:
candidate.load(module.params['src'])
elif module.params['lines']:
parents = module.params['parents'] or list()
candidate.add(module.params['lines'], parents=parents)
return candidate
def load_config(module, commands, result):
replace = module.params['update'] == 'replace'
comment = module.params['comment']
commit = not module.check_mode
diff = module.config.load_config(commands, replace=replace, commit=commit,
comment=comment)
result['diff'] = dict(prepared=diff)
result['changed'] = True
def run(module, result):
match = module.params['match'] match = module.params['match']
replace = module.params['replace'] replace = module.params['replace']
update = module.params['update']
contents = get_config(module) candidate = get_candidate(module)
config = module.parse_config(contents)
if not module.params['force']: if match != 'none' and update != 'replace':
contents = get_config(module) config = get_config(module, result)
config = NetworkConfig(contents=contents, indent=1) configobjs = candidate.difference(config, match=match, replace=replace)
candidate = NetworkConfig(indent=1)
candidate.add(lines, parents=parents)
commands = candidate.difference(config, path=parents, match=match, replace=replace)
else: else:
commands = parents config = None
commands.extend(lines) configobjs = candidate.items
result = dict(changed=False) if configobjs:
commands = dumps(configobjs, 'commands')
if commands: if module.params['before']:
if before: commands[:0] = module.params['before']
commands[:0] = before
if after: if module.params['after']:
commands.extend(after) commands.extend(module.params['after'])
if not module.check_mode: result['updates'] = commands.split('\n')
commands = [str(c).strip() for c in commands]
response = module.configure(commands) if update != 'check':
result['responses'] = response load_config(module, commands, result)
result['changed'] = True
def main():
"""main entry point for module execution
"""
argument_spec = dict(
lines=dict(aliases=['commands'], type='list'),
parents=dict(type='list'),
src=dict(type='path'),
before=dict(type='list'),
after=dict(type='list'),
match=dict(default='line', choices=['line', 'strict', 'exact', 'none']),
replace=dict(default='line', choices=['line', 'block']),
update=dict(choices=['merge', 'replace', 'check'], default='merge'),
backup=dict(type='bool', default=False),
comment=dict(default=DEFAULT_COMMIT_COMMENT),
# this argument is deprecated in favor of setting match: none
# it will be removed in a future version
force=dict(default=False, type='bool'),
config=dict(),
)
mutually_exclusive = [('lines', 'src')]
module = NetworkModule(argument_spec=argument_spec,
connect_on_load=False,
mutually_exclusive=mutually_exclusive,
supports_check_mode=True)
if module.params['force'] is True:
module.params['match'] = 'none'
warnings = list()
check_args(module, warnings)
result = dict(changed=False, warnings=warnings)
if module.params['backup']:
config = module.config.get_config()
result['__config__'] = config
result['__backup__'] = config
try:
run(module, result)
except NetworkError:
exc = get_exception()
module.fail_json(msg=str(exc), **exc.kwargs)
result['updates'] = commands
module.exit_json(**result) module.exit_json(**result)
from ansible.module_utils.basic import *
from ansible.module_utils.shell import *
from ansible.module_utils.netcfg import *
from ansible.module_utils.iosxr import *
if __name__ == '__main__': if __name__ == '__main__':
main() main()