2016-08-10 17:45:54 +00:00
|
|
|
# This code is part of Ansible, but is an independent component.
|
|
|
|
# This particular file snippet, and this file snippet only, is BSD licensed.
|
|
|
|
# Modules you write using this snippet, which is embedded dynamically by Ansible
|
|
|
|
# still belong to the author of the module, and may assign their own license
|
|
|
|
# to the complete work.
|
2016-01-08 19:00:23 +00:00
|
|
|
#
|
2016-08-10 17:45:54 +00:00
|
|
|
# Copyright (c) 2015 Peter Sprygada, <psprygada@ansible.com>
|
2016-01-08 19:00:23 +00:00
|
|
|
#
|
2016-08-10 17:45:54 +00:00
|
|
|
# Redistribution and use in source and binary forms, with or without modification,
|
|
|
|
# are permitted provided that the following conditions are met:
|
2016-01-08 19:00:23 +00:00
|
|
|
#
|
2016-08-10 17:45:54 +00:00
|
|
|
# * Redistributions of source code must retain the above copyright
|
|
|
|
# notice, this list of conditions and the following disclaimer.
|
|
|
|
# * Redistributions in binary form must reproduce the above copyright notice,
|
|
|
|
# this list of conditions and the following disclaimer in the documentation
|
|
|
|
# and/or other materials provided with the distribution.
|
2016-01-08 19:00:23 +00:00
|
|
|
#
|
2016-08-10 17:45:54 +00:00
|
|
|
# THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND
|
|
|
|
# ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
|
|
|
|
# WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED.
|
|
|
|
# IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT,
|
|
|
|
# INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
|
|
|
|
# PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
|
|
|
|
# INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
|
|
|
|
# LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE
|
|
|
|
# USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
2016-01-08 19:00:23 +00:00
|
|
|
#
|
|
|
|
|
2016-04-05 18:06:17 +00:00
|
|
|
import re
|
|
|
|
|
2016-09-06 17:49:48 +00:00
|
|
|
from ansible.module_utils.netcli import Command
|
|
|
|
from ansible.module_utils.network import NetworkError, NetworkModule
|
2016-07-07 17:46:33 +00:00
|
|
|
from ansible.module_utils.network import register_transport, to_list
|
2016-09-06 17:49:48 +00:00
|
|
|
from ansible.module_utils.shell import CliBase
|
2016-04-05 18:06:17 +00:00
|
|
|
|
2016-08-24 15:03:01 +00:00
|
|
|
|
|
|
|
class Cli(CliBase):
|
|
|
|
|
2016-07-07 17:46:33 +00:00
|
|
|
CLI_PROMPTS_RE = [
|
|
|
|
re.compile(r"[\r\n]?[\w+\-\.:\/\[\]]+(?:\([^\)]+\)){,3}(?:>|#) ?$"),
|
|
|
|
re.compile(r"\[\w+\@[\w\-\.]+(?: [^\]])\] ?[>#\$] ?$")
|
|
|
|
]
|
2016-01-08 19:00:23 +00:00
|
|
|
|
2016-07-07 17:46:33 +00:00
|
|
|
CLI_ERRORS_RE = [
|
|
|
|
re.compile(r"% ?Error"),
|
|
|
|
re.compile(r"% ?Bad secret"),
|
|
|
|
re.compile(r"invalid input", re.I),
|
|
|
|
re.compile(r"(?:incomplete|ambiguous) command", re.I),
|
|
|
|
re.compile(r"connection timed out", re.I),
|
|
|
|
re.compile(r"[^\r\n]+ not found", re.I),
|
|
|
|
re.compile(r"'[^']' +returned error code: ?\d+"),
|
|
|
|
]
|
2016-01-08 19:00:23 +00:00
|
|
|
|
2016-07-07 17:46:33 +00:00
|
|
|
NET_PASSWD_RE = re.compile(r"[\r\n]?password: $", re.I)
|
2016-04-04 20:06:01 +00:00
|
|
|
|
2016-07-07 17:46:33 +00:00
|
|
|
def connect(self, params, **kwargs):
|
|
|
|
super(Cli, self).connect(params, kickstart=False, **kwargs)
|
2016-09-06 17:49:48 +00:00
|
|
|
self.shell.send(['terminal length 0', 'terminal exec prompt no-timestamp'])
|
2016-08-24 15:03:01 +00:00
|
|
|
|
2016-09-06 17:49:48 +00:00
|
|
|
### Config methods ###
|
2016-04-25 20:04:19 +00:00
|
|
|
|
2016-09-12 03:32:05 +00:00
|
|
|
def configure(self, commands):
|
2016-08-24 15:03:01 +00:00
|
|
|
cmds = ['configure terminal']
|
2016-08-26 19:58:06 +00:00
|
|
|
if commands[-1] == 'end':
|
|
|
|
commands.pop()
|
2016-07-07 17:46:33 +00:00
|
|
|
cmds.extend(to_list(commands))
|
2016-08-26 19:58:06 +00:00
|
|
|
cmds.extend(['commit', 'end'])
|
2016-07-07 17:46:33 +00:00
|
|
|
responses = self.execute(cmds)
|
2016-08-24 15:03:01 +00:00
|
|
|
return responses[1:]
|
2016-04-25 20:04:19 +00:00
|
|
|
|
2016-09-12 03:32:05 +00:00
|
|
|
def get_config(self, flags=None):
|
2016-08-24 15:03:01 +00:00
|
|
|
cmd = 'show running-config'
|
|
|
|
if flags:
|
|
|
|
if isinstance(flags, list):
|
|
|
|
cmd += ' %s' % ' '.join(flags)
|
|
|
|
else:
|
|
|
|
cmd += ' %s' % flags
|
|
|
|
return self.execute([cmd])[0]
|
2016-01-08 19:00:23 +00:00
|
|
|
|
2016-08-26 17:16:18 +00:00
|
|
|
def load_config(self, config, commit=False, replace=False, comment=None):
|
2016-08-24 15:03:01 +00:00
|
|
|
commands = ['configure terminal']
|
|
|
|
commands.extend(config)
|
2016-01-08 19:00:23 +00:00
|
|
|
|
2016-08-24 15:03:01 +00:00
|
|
|
if commands[-1] == 'end':
|
|
|
|
commands.pop()
|
2016-01-08 19:00:23 +00:00
|
|
|
|
2016-08-24 15:03:01 +00:00
|
|
|
try:
|
|
|
|
self.execute(commands)
|
|
|
|
diff = self.execute(['show commit changes diff'])
|
|
|
|
if commit:
|
|
|
|
if replace:
|
|
|
|
prompt = re.compile(r'\[no\]:\s$')
|
2016-08-26 17:16:18 +00:00
|
|
|
commit = 'commit replace'
|
|
|
|
if comment:
|
|
|
|
commit += ' comment %s' % comment
|
|
|
|
cmd = Command(commit, prompt=prompt, response='yes')
|
2016-08-24 15:03:01 +00:00
|
|
|
self.execute([cmd, 'end'])
|
|
|
|
else:
|
2016-08-26 17:16:18 +00:00
|
|
|
commit = 'commit'
|
|
|
|
if comment:
|
|
|
|
commit += ' comment %s' % comment
|
|
|
|
self.execute([commit, 'end'])
|
|
|
|
else:
|
|
|
|
self.execute(['abort'])
|
2016-08-24 15:03:01 +00:00
|
|
|
except NetworkError:
|
|
|
|
self.execute(['abort'])
|
|
|
|
diff = None
|
|
|
|
raise
|
2016-09-12 03:32:05 +00:00
|
|
|
|
2016-08-24 15:03:01 +00:00
|
|
|
return diff[0]
|
|
|
|
|
2016-07-07 17:46:33 +00:00
|
|
|
Cli = register_transport('cli', default=True)(Cli)
|