diff options
author | Daniel Baumann <daniel.baumann@progress-linux.org> | 2024-05-14 20:03:01 +0000 |
---|---|---|
committer | Daniel Baumann <daniel.baumann@progress-linux.org> | 2024-05-14 20:03:01 +0000 |
commit | a453ac31f3428614cceb99027f8efbdb9258a40b (patch) | |
tree | f61f87408f32a8511cbd91799f9cececb53e0374 /collections-debian-merged/ansible_collections/arista/eos/plugins | |
parent | Initial commit. (diff) | |
download | ansible-upstream.tar.xz ansible-upstream.zip |
Adding upstream version 2.10.7+merged+base+2.10.8+dfsg.upstream/2.10.7+merged+base+2.10.8+dfsgupstream
Signed-off-by: Daniel Baumann <daniel.baumann@progress-linux.org>
Diffstat (limited to 'collections-debian-merged/ansible_collections/arista/eos/plugins')
177 files changed, 40750 insertions, 0 deletions
diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/action/__init__.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/action/__init__.py new file mode 100644 index 00000000..e69de29b --- /dev/null +++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/action/__init__.py diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/action/eos.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/action/eos.py new file mode 100644 index 00000000..03d317f0 --- /dev/null +++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/action/eos.py @@ -0,0 +1,194 @@ +# +# (c) 2016 Red Hat Inc. +# +# 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/>. +# + +from __future__ import absolute_import, division, print_function + +__metaclass__ = type + +import sys +import copy + +from ansible import constants as C +from ansible_collections.arista.eos.plugins.module_utils.network.eos.eos import ( + eos_provider_spec, +) +from ansible_collections.ansible.netcommon.plugins.action.network import ( + ActionModule as ActionNetworkModule, +) +from ansible_collections.ansible.netcommon.plugins.module_utils.network.common.utils import ( + load_provider, +) +from ansible.utils.display import Display + +display = Display() + + +class ActionModule(ActionNetworkModule): + def run(self, tmp=None, task_vars=None): + del tmp # tmp no longer has any effect + + module_name = self._task.action.split(".")[-1] + self._config_module = ( + True if module_name in ["eos_config", "config"] else False + ) + persistent_connection = self._play_context.connection.split(".")[-1] + warnings = [] + + if persistent_connection in ("network_cli", "httpapi"): + provider = self._task.args.get("provider", {}) + if any(provider.values()): + display.warning( + "provider is unnecessary when using %s and will be ignored" + % self._play_context.connection + ) + del self._task.args["provider"] + if self._task.args.get("transport"): + display.warning( + "transport is unnecessary when using %s and will be ignored" + % self._play_context.connection + ) + del self._task.args["transport"] + elif self._play_context.connection == "local": + provider = load_provider(eos_provider_spec, self._task.args) + transport = provider["transport"] or "cli" + + display.vvvv( + "connection transport is %s" % transport, + self._play_context.remote_addr, + ) + + if transport == "cli": + pc = copy.deepcopy(self._play_context) + pc.connection = "ansible.netcommon.network_cli" + pc.network_os = "arista.eos.eos" + pc.remote_addr = ( + provider["host"] or self._play_context.remote_addr + ) + pc.port = int( + provider["port"] or self._play_context.port or 22 + ) + pc.remote_user = ( + provider["username"] or self._play_context.connection_user + ) + pc.password = ( + provider["password"] or self._play_context.password + ) + pc.private_key_file = ( + provider["ssh_keyfile"] + or self._play_context.private_key_file + ) + pc.become = provider["authorize"] or False + if pc.become: + pc.become_method = "enable" + pc.become_pass = provider["auth_pass"] + + connection = self._shared_loader_obj.connection_loader.get( + "ansible.netcommon.persistent", + pc, + sys.stdin, + task_uuid=self._task._uuid, + ) + + # TODO: Remove below code after ansible minimal is cut out + if connection is None: + pc.connection = "network_cli" + pc.network_os = "eos" + connection = self._shared_loader_obj.connection_loader.get( + "persistent", pc, sys.stdin, task_uuid=self._task._uuid + ) + + display.vvv( + "using connection plugin %s (was local)" % pc.connection, + pc.remote_addr, + ) + + command_timeout = ( + int(provider["timeout"]) + if provider["timeout"] + else connection.get_option("persistent_command_timeout") + ) + connection.set_options( + direct={"persistent_command_timeout": command_timeout} + ) + + socket_path = connection.run() + display.vvvv("socket_path: %s" % socket_path, pc.remote_addr) + if not socket_path: + return { + "failed": True, + "msg": "unable to open shell. Please see: " + + "https://docs.ansible.com/ansible/network_debug_troubleshooting.html#unable-to-open-shell", + } + + task_vars["ansible_socket"] = socket_path + warnings.append( + [ + "connection local support for this module is deprecated and will be removed in version 2.14," + " use connection %s" % pc.connection + ] + ) + else: + self._task.args["provider"] = ActionModule.eapi_implementation( + provider, self._play_context + ) + warnings.append( + [ + "connection local support for this module is deprecated and will be removed in version 2.14," + " use connection either httpapi or ansible.netcommon.httpapi (whichever is applicable)" + ] + ) + else: + return { + "failed": True, + "msg": "Connection type %s is not valid for this module" + % self._play_context.connection, + } + + result = super(ActionModule, self).run(task_vars=task_vars) + if warnings: + if "warnings" in result: + result["warnings"].extend(warnings) + else: + result["warnings"] = warnings + return result + + @staticmethod + def eapi_implementation(provider, play_context): + provider["transport"] = "eapi" + + if provider.get("host") is None: + provider["host"] = play_context.remote_addr + + if provider.get("port") is None: + default_port = 443 if provider["use_ssl"] else 80 + provider["port"] = int(play_context.port or default_port) + + if provider.get("timeout") is None: + provider["timeout"] = C.PERSISTENT_COMMAND_TIMEOUT + + if provider.get("username") is None: + provider["username"] = play_context.connection_user + + if provider.get("password") is None: + provider["password"] = play_context.password + + if provider.get("authorize") is None: + provider["authorize"] = False + + return provider diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/cliconf/__init__.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/cliconf/__init__.py new file mode 100644 index 00000000..e69de29b --- /dev/null +++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/cliconf/__init__.py diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/cliconf/eos.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/cliconf/eos.py new file mode 100644 index 00000000..0e62fc03 --- /dev/null +++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/cliconf/eos.py @@ -0,0 +1,374 @@ +# +# (c) 2017 Red Hat Inc. +# +# 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/>. +# + +from __future__ import absolute_import, division, print_function + +__metaclass__ = type + +DOCUMENTATION = """ +author: Ansible Networking Team +cliconf: eos +short_description: Use eos cliconf to run command on Arista EOS platform +description: +- This eos plugin provides low level abstraction apis for sending and receiving CLI + commands from Arista EOS network devices. +version_added: 1.0.0 +options: + eos_use_sessions: + type: boolean + default: true + description: + - Specifies if sessions should be used on remote host or not + env: + - name: ANSIBLE_EOS_USE_SESSIONS + vars: + - name: ansible_eos_use_sessions +""" + +import json +import time +import re + +from ansible.errors import AnsibleConnectionFailure +from ansible.module_utils._text import to_text +from ansible.module_utils.common._collections_compat import Mapping +from ansible_collections.ansible.netcommon.plugins.module_utils.network.common.utils import ( + to_list, +) +from ansible_collections.ansible.netcommon.plugins.module_utils.network.common.config import ( + NetworkConfig, + dumps, +) +from ansible.plugins.cliconf import CliconfBase, enable_mode + + +class Cliconf(CliconfBase): + def __init__(self, *args, **kwargs): + super(Cliconf, self).__init__(*args, **kwargs) + self._session_support = None + + @enable_mode + def get_config(self, source="running", format="text", flags=None): + options_values = self.get_option_values() + if format not in options_values["format"]: + raise ValueError( + "'format' value %s is invalid. Valid values are %s" + % (format, ",".join(options_values["format"])) + ) + + lookup = {"running": "running-config", "startup": "startup-config"} + if source not in lookup: + raise ValueError( + "fetching configuration from %s is not supported" % source + ) + + cmd = "show %s " % lookup[source] + if format and format != "text": + cmd += "| %s " % format + + cmd += " ".join(to_list(flags)) + cmd = cmd.strip() + return self.send_command(cmd) + + @enable_mode + def edit_config( + self, candidate=None, commit=True, replace=None, comment=None + ): + + operations = self.get_device_operations() + self.check_edit_config_capability( + operations, candidate, commit, replace, comment + ) + + if (commit is False) and (not self.supports_sessions()): + raise ValueError( + "check mode is not supported without configuration session" + ) + + resp = {} + session = None + if self.supports_sessions(): + session = "ansible_%s" % int(time.time()) + resp.update({"session": session}) + self.send_command("configure session %s" % session) + if replace: + self.send_command("rollback clean-config") + else: + self.send_command("configure") + + results = [] + requests = [] + multiline = False + for line in to_list(candidate): + if not isinstance(line, Mapping): + line = {"command": line} + + cmd = line["command"] + if cmd == "end": + continue + if cmd.startswith("banner") or multiline: + multiline = True + elif cmd == "EOF" and multiline: + multiline = False + + if multiline: + line["sendonly"] = True + + if cmd != "end" and not cmd.startswith("!"): + try: + results.append(self.send_command(**line)) + requests.append(cmd) + except AnsibleConnectionFailure as e: + self.discard_changes(session) + raise AnsibleConnectionFailure(e.message) + + resp["request"] = requests + resp["response"] = results + if self.supports_sessions(): + out = self.send_command("show session-config diffs") + if out: + resp["diff"] = out.strip() + + if commit: + self.commit() + else: + self.discard_changes(session) + else: + self.send_command("end") + return resp + + def get( + self, + command, + prompt=None, + answer=None, + sendonly=False, + output=None, + newline=True, + check_all=False, + ): + if output: + command = self._get_command_with_output(command, output) + return self.send_command( + command=command, + prompt=prompt, + answer=answer, + sendonly=sendonly, + newline=newline, + check_all=check_all, + ) + + def commit(self): + self.send_command("commit") + + def discard_changes(self, session=None): + commands = ["end"] + if self.supports_sessions(): + # to close session gracefully execute abort in top level session prompt. + commands.extend(["configure session %s" % session, "abort"]) + + for cmd in commands: + self.send_command(cmd) + + def run_commands(self, commands=None, check_rc=True): + if commands is None: + raise ValueError("'commands' value is required") + responses = list() + for cmd in to_list(commands): + if not isinstance(cmd, Mapping): + cmd = {"command": cmd} + + output = cmd.pop("output", None) + if output: + cmd["command"] = self._get_command_with_output( + cmd["command"], output + ) + + try: + out = self.send_command(**cmd) + except AnsibleConnectionFailure as e: + if check_rc: + raise + out = getattr(e, "err", e) + out = to_text(out, errors="surrogate_or_strict") + + if out is not None: + try: + out = json.loads(out) + except ValueError: + out = out.strip() + + responses.append(out) + return responses + + def get_diff( + self, + candidate=None, + running=None, + diff_match="line", + diff_ignore_lines=None, + path=None, + diff_replace="line", + ): + diff = {} + device_operations = self.get_device_operations() + option_values = self.get_option_values() + + if candidate is None and device_operations["supports_generate_diff"]: + raise ValueError( + "candidate configuration is required to generate diff" + ) + + if diff_match not in option_values["diff_match"]: + raise ValueError( + "'match' value %s in invalid, valid values are %s" + % (diff_match, ", ".join(option_values["diff_match"])) + ) + + if diff_replace not in option_values["diff_replace"]: + raise ValueError( + "'replace' value %s in invalid, valid values are %s" + % (diff_replace, ", ".join(option_values["diff_replace"])) + ) + + # prepare candidate configuration + candidate_obj = NetworkConfig(indent=3) + candidate_obj.load(candidate) + + if running and diff_match != "none" and diff_replace != "config": + # running configuration + running_obj = NetworkConfig( + indent=3, contents=running, ignore_lines=diff_ignore_lines + ) + configdiffobjs = candidate_obj.difference( + running_obj, path=path, match=diff_match, replace=diff_replace + ) + + else: + configdiffobjs = candidate_obj.items + + diff["config_diff"] = ( + dumps(configdiffobjs, "commands") if configdiffobjs else "" + ) + return diff + + def supports_sessions(self): + if not self.get_option("eos_use_sessions"): + self._session_support = False + else: + if self._session_support: + return self._session_support + + try: + self.get("show configuration sessions") + self._session_support = True + except AnsibleConnectionFailure: + self._session_support = False + + return self._session_support + + def get_device_info(self): + device_info = {} + + device_info["network_os"] = "eos" + reply = self.get("show version | json") + data = json.loads(reply) + + device_info["network_os_version"] = data["version"] + device_info["network_os_model"] = data["modelName"] + + reply = self.get("show hostname | json") + data = json.loads(reply) + + device_info["network_os_hostname"] = data["hostname"] + + try: + reply = self.get("bash timeout 5 cat /mnt/flash/boot-config") + + match = re.search(r"SWI=(.+)$", reply, re.M) + if match: + device_info["network_os_image"] = match.group(1) + except AnsibleConnectionFailure: + # This requires enable mode to run + self._connection.queue_message( + "vvv", "Unable to gather network_os_image without enable mode" + ) + + return device_info + + def get_device_operations(self): + return { + "supports_diff_replace": True, + "supports_commit": bool(self.supports_sessions()), + "supports_rollback": False, + "supports_defaults": False, + "supports_onbox_diff": bool(self.supports_sessions()), + "supports_commit_comment": False, + "supports_multiline_delimiter": False, + "supports_diff_match": True, + "supports_diff_ignore_lines": True, + "supports_generate_diff": not bool(self.supports_sessions()), + "supports_replace": bool(self.supports_sessions()), + } + + def get_option_values(self): + return { + "format": ["text", "json"], + "diff_match": ["line", "strict", "exact", "none"], + "diff_replace": ["line", "block", "config"], + "output": ["text", "json"], + } + + def get_capabilities(self): + result = super(Cliconf, self).get_capabilities() + result["rpc"] += [ + "commit", + "discard_changes", + "get_diff", + "run_commands", + "supports_sessions", + ] + result["device_operations"] = self.get_device_operations() + result.update(self.get_option_values()) + + return json.dumps(result) + + def set_cli_prompt_context(self): + """ + Make sure we are in the operational cli mode + :return: None + """ + if self._connection.connected: + self._update_cli_prompt_context( + config_context="(config", exit_command="abort" + ) + + def _get_command_with_output(self, command, output): + options_values = self.get_option_values() + if output not in options_values["output"]: + raise ValueError( + "'output' value %s is invalid. Valid values are %s" + % (output, ",".join(options_values["output"])) + ) + + if output == "json" and not command.endswith("| json"): + cmd = "%s | json" % command + else: + cmd = command + return cmd diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/doc_fragments/__init__.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/doc_fragments/__init__.py new file mode 100644 index 00000000..e69de29b --- /dev/null +++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/doc_fragments/__init__.py diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/doc_fragments/eos.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/doc_fragments/eos.py new file mode 100644 index 00000000..a9eeccd7 --- /dev/null +++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/doc_fragments/eos.py @@ -0,0 +1,118 @@ +# -*- coding: utf-8 -*- +from __future__ import absolute_import, division, print_function + +__metaclass__ = type + +# Copyright: (c) 2015, Peter Sprygada <psprygada@ansible.com> +# GNU General Public License v3.0+ (see COPYING or https://www.gnu.org/licenses/gpl-3.0.txt) + + +class ModuleDocFragment(object): + + # Standard files documentation fragment + DOCUMENTATION = r"""options: + provider: + description: + - B(Deprecated) + - 'Starting with Ansible 2.5 we recommend using C(connection: network_cli).' + - 'Starting with Ansible 2.6 we recommend using C(connection: httpapi) for eAPI.' + - This option will be removed in a release after 2022-06-01. + - For more information please see the L(EOS Platform Options guide, ../network/user_guide/platform_eos.html). + - HORIZONTALLINE + - A dict object containing connection details. + type: dict + suboptions: + host: + description: + - Specifies the DNS host name or address for connecting to the remote device + over the specified transport. The value of host is used as the destination + address for the transport. + type: str + port: + description: + - Specifies the port to use when building the connection to the remote device. This + value applies to either I(cli) or I(eapi). + - The port value will default to the appropriate transport common port if + none is provided in the task (cli=22, http=80, https=443). + type: int + default: 0 + username: + description: + - Configures the username to use to authenticate the connection to the remote + device. This value is used to authenticate either the CLI login or the + eAPI authentication depending on which transport is used. If the value is + not specified in the task, the value of environment variable C(ANSIBLE_NET_USERNAME) + will be used instead. + type: str + password: + description: + - Specifies the password to use to authenticate the connection to the remote + device. This is a common argument used for either I(cli) or I(eapi) transports. + If the value is not specified in the task, the value of environment variable + C(ANSIBLE_NET_PASSWORD) will be used instead. + type: str + timeout: + description: + - Specifies the timeout in seconds for communicating with the network device + for either connecting or sending commands. If the timeout is exceeded before + the operation is completed, the module will error. + type: int + ssh_keyfile: + description: + - Specifies the SSH keyfile to use to authenticate the connection to the remote + device. This argument is only used for I(cli) transports. If the value + is not specified in the task, the value of environment variable C(ANSIBLE_NET_SSH_KEYFILE) + will be used instead. + type: path + authorize: + description: + - Instructs the module to enter privileged mode on the remote device before + sending any commands. If not specified, the device will attempt to execute + all commands in non-privileged mode. If the value is not specified in the + task, the value of environment variable C(ANSIBLE_NET_AUTHORIZE) will be + used instead. + type: bool + default: false + auth_pass: + description: + - Specifies the password to use if required to enter privileged mode on the + remote device. If I(authorize) is false, then this argument does nothing. + If the value is not specified in the task, the value of environment variable + C(ANSIBLE_NET_AUTH_PASS) will be used instead. + type: str + transport: + description: + - Configures the transport connection to use when connecting to the remote + device. + type: str + choices: + - cli + - eapi + default: cli + use_ssl: + description: + - Configures the I(transport) to use SSL if set to C(yes) only when the C(transport=eapi). If + the transport argument is not eapi, this value is ignored. + type: bool + default: true + validate_certs: + description: + - If C(no), SSL certificates will not be validated. This should only be used + on personally controlled sites using self-signed certificates. If the transport + argument is not eapi, this value is ignored. + type: bool + default: true + use_proxy: + description: + - If C(no), the environment variables C(http_proxy) and C(https_proxy) will + be ignored. + type: bool + default: true +notes: +- For information on using CLI, eAPI and privileged mode see the :ref:`EOS Platform + Options guide <eos_platform_options>` +- For more information on using Ansible to manage network devices see the :ref:`Ansible + Network Guide <network_guide>` +- For more information on using Ansible to manage Arista EOS devices see the `Arista + integration page <https://www.ansible.com/ansible-arista-networks>`_. +""" diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/filter/__init__.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/filter/__init__.py new file mode 100644 index 00000000..e69de29b --- /dev/null +++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/filter/__init__.py diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/httpapi/__init__.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/httpapi/__init__.py new file mode 100644 index 00000000..e69de29b --- /dev/null +++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/httpapi/__init__.py diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/httpapi/eos.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/httpapi/eos.py new file mode 100644 index 00000000..ae4203db --- /dev/null +++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/httpapi/eos.py @@ -0,0 +1,207 @@ +# (c) 2018 Red Hat Inc. +# 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 + +DOCUMENTATION = """ +author: Ansible Networking Team +httpapi: eos +short_description: Use eAPI to run command on eos platform +description: +- This eos plugin provides low level abstraction api's for sending and receiving CLI + commands with eos network devices. +version_added: 1.0.0 +options: + eos_use_sessions: + type: int + default: 1 + description: + - Specifies if sessions should be used on remote host or not + env: + - name: ANSIBLE_EOS_USE_SESSIONS + vars: + - name: ansible_eos_use_sessions +""" + +import json +import time + +from ansible.module_utils._text import to_text +from ansible.module_utils.connection import ConnectionError +from ansible_collections.ansible.netcommon.plugins.module_utils.network.common.utils import ( + to_list, +) +from ansible.plugins.httpapi import HttpApiBase + + +OPTIONS = { + "format": ["text", "json"], + "diff_match": ["line", "strict", "exact", "none"], + "diff_replace": ["line", "block", "config"], + "output": ["text", "json"], +} + + +class HttpApi(HttpApiBase): + def __init__(self, *args, **kwargs): + super(HttpApi, self).__init__(*args, **kwargs) + self._device_info = None + self._session_support = None + + def supports_sessions(self): + use_session = self.get_option("eos_use_sessions") + try: + use_session = int(use_session) + except ValueError: + pass + + if not bool(use_session): + self._session_support = False + else: + if self._session_support: + return self._session_support + + response = self.send_request("show configuration sessions") + self._session_support = "error" not in response + + return self._session_support + + def send_request(self, data, **message_kwargs): + data = to_list(data) + become = self._become + if become: + self.connection.queue_message("vvvv", "firing event: on_become") + data.insert(0, {"cmd": "enable", "input": self._become_pass}) + + output = message_kwargs.get("output") or "text" + request = request_builder(data, output) + headers = {"Content-Type": "application/json-rpc"} + + _response, response_data = self.connection.send( + "/command-api", request, headers=headers, method="POST" + ) + + try: + response_data = json.loads(to_text(response_data.getvalue())) + except ValueError: + raise ConnectionError( + "Response was not valid JSON, got {0}".format( + to_text(response_data.getvalue()) + ) + ) + + results = handle_response(response_data) + + if become: + results = results[1:] + if len(results) == 1: + results = results[0] + + return results + + def get_device_info(self): + if self._device_info: + return self._device_info + + device_info = {} + + device_info["network_os"] = "eos" + reply = self.send_request("show version", output="json") + data = json.loads(reply) + + device_info["network_os_version"] = data["version"] + device_info["network_os_model"] = data["modelName"] + + reply = self.send_request("show hostname", output="json") + data = json.loads(reply) + + device_info["network_os_hostname"] = data["hostname"] + + self._device_info = device_info + return self._device_info + + def get_device_operations(self): + return { + "supports_diff_replace": True, + "supports_commit": bool(self.supports_sessions()), + "supports_rollback": False, + "supports_defaults": False, + "supports_onbox_diff": bool(self.supports_sessions()), + "supports_commit_comment": False, + "supports_multiline_delimiter": False, + "supports_diff_match": True, + "supports_diff_ignore_lines": True, + "supports_generate_diff": not bool(self.supports_sessions()), + "supports_replace": bool(self.supports_sessions()), + } + + def get_capabilities(self): + result = {} + result["rpc"] = [] + result["device_info"] = self.get_device_info() + result["device_operations"] = self.get_device_operations() + result.update(OPTIONS) + result["network_api"] = "eapi" + + return json.dumps(result) + + # Shims for resource module support + def get(self, command, output=None): + # This method is ONLY here to support resource modules. Therefore most + # arguments are unsupported and not present. + + return self.send_request(data=command, output=output) + + def edit_config(self, candidate): + # This method is ONLY here to support resource modules. Therefore most + # arguments are unsupported and not present. + + session = None + if self.supports_sessions(): + session = "ansible_%d" % int(time.time()) + candidate = ["configure session %s" % session] + candidate + else: + candidate = ["configure"] + candidate + candidate.append("commit") + + try: + responses = self.send_request(candidate) + except ConnectionError: + if session: + self.send_request(["configure session %s" % session, "abort"]) + raise + + return [resp for resp in to_list(responses) if resp != "{}"] + + +def handle_response(response): + if "error" in response: + error = response["error"] + + error_text = [] + for data in error.get("data", []): + error_text.extend(data.get("errors", [])) + error_text = "\n".join(error_text) or error["message"] + + raise ConnectionError(error_text, code=error["code"]) + + results = [] + + for result in response["result"]: + if "messages" in result: + results.append(result["messages"][0]) + elif "output" in result: + results.append(result["output"].strip()) + else: + results.append(json.dumps(result)) + + return results + + +def request_builder(commands, output, reqid=None): + params = dict(version=1, cmds=commands, format=output) + return json.dumps( + dict(jsonrpc="2.0", id=reqid, method="runCmds", params=params) + ) diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/inventory/__init__.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/inventory/__init__.py new file mode 100644 index 00000000..e69de29b --- /dev/null +++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/inventory/__init__.py diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/__init__.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/__init__.py new file mode 100644 index 00000000..e69de29b --- /dev/null +++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/__init__.py diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/__init__.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/__init__.py new file mode 100644 index 00000000..e69de29b --- /dev/null +++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/__init__.py diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/__init__.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/__init__.py new file mode 100644 index 00000000..e69de29b --- /dev/null +++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/__init__.py diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/__init__.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/__init__.py new file mode 100644 index 00000000..e69de29b --- /dev/null +++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/__init__.py diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/acl_interfaces/__init__.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/acl_interfaces/__init__.py new file mode 100644 index 00000000..e69de29b --- /dev/null +++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/acl_interfaces/__init__.py diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/acl_interfaces/acl_interfaces.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/acl_interfaces/acl_interfaces.py new file mode 100644 index 00000000..ae20ee94 --- /dev/null +++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/acl_interfaces/acl_interfaces.py @@ -0,0 +1,85 @@ +# +# -*- coding: utf-8 -*- +# Copyright 2019 Red Hat +# GNU General Public License v3.0+ +# (see COPYING or https://www.gnu.org/licenses/gpl-3.0.txt) + +############################################# +# WARNING # +############################################# +# +# This file is auto generated by the resource +# module builder playbook. +# +# Do not edit this file manually. +# +# Changes to this file will be over written +# by the resource module builder. +# +# Changes should be made in the model used to +# generate this file or in the resource module +# builder template. +# +############################################# +""" +The arg spec for the eos_acl_interfaces module +""" + +from __future__ import absolute_import, division, print_function + +__metaclass__ = type + + +class Acl_interfacesArgs(object): # pylint: disable=R0903 + """The arg spec for the eos_acl_interfaces module + """ + + def __init__(self, **kwargs): + pass + + argument_spec = { + "config": { + "elements": "dict", + "options": { + "access_groups": { + "elements": "dict", + "options": { + "acls": { + "elements": "dict", + "options": { + "direction": { + "required": True, + "choices": ["in", "out"], + "type": "str", + }, + "name": {"required": True, "type": "str"}, + }, + "type": "list", + }, + "afi": { + "required": True, + "choices": ["ipv4", "ipv6"], + "type": "str", + }, + }, + "type": "list", + }, + "name": {"required": True, "type": "str"}, + }, + "type": "list", + }, + "running_config": {"type": "str"}, + "state": { + "choices": [ + "merged", + "replaced", + "overridden", + "deleted", + "gathered", + "parsed", + "rendered", + ], + "default": "merged", + "type": "str", + }, + } # pylint: disable=C0301 diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/acls/__init__.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/acls/__init__.py new file mode 100644 index 00000000..e69de29b --- /dev/null +++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/acls/__init__.py diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/acls/acls.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/acls/acls.py new file mode 100644 index 00000000..b6d09a38 --- /dev/null +++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/acls/acls.py @@ -0,0 +1,416 @@ +# +# -*- coding: utf-8 -*- +# Copyright 2019 Red Hat +# GNU General Public License v3.0+ +# (see COPYING or https://www.gnu.org/licenses/gpl-3.0.txt) + +############################################# +# WARNING # +############################################# +# +# This file is auto generated by the resource +# module builder playbook. +# +# Do not edit this file manually. +# +# Changes to this file will be over written +# by the resource module builder. +# +# Changes should be made in the model used to +# generate this file or in the resource module +# builder template. +# +############################################# +""" +The arg spec for the eos_acls module +""" + +from __future__ import absolute_import, division, print_function + +__metaclass__ = type + + +class AclsArgs(object): # pylint: disable=R0903 + """The arg spec for the eos_acls module + """ + + def __init__(self, **kwargs): + pass + + argument_spec = { + "config": { + "elements": "dict", + "options": { + "acls": { + "elements": "dict", + "options": { + "aces": { + "elements": "dict", + "options": { + "destination": { + "mutually_exclusive": [ + [ + "address", + "subnet_address", + "any", + "host", + ], + [ + "wildcard_bits", + "subnet_address", + "any", + "host", + ], + ], + "options": { + "address": {"type": "str"}, + "any": {"type": "bool"}, + "host": {"type": "str"}, + "port_protocol": {"type": "dict"}, + "subnet_address": {"type": "str"}, + "wildcard_bits": {"type": "str"}, + }, + "required_together": [ + ["address", "wildcard_bits"] + ], + "type": "dict", + }, + "fragment_rules": {"type": "bool"}, + "fragments": {"type": "bool"}, + "grant": { + "choices": ["permit", "deny"], + "type": "str", + }, + "line": {"type": "str", "aliases": ["ace"]}, + "hop_limit": {"type": "dict"}, + "log": {"type": "bool"}, + "protocol": {"type": "str"}, + "protocol_options": { + "options": { + "icmp": { + "options": { + "administratively_prohibited": { + "type": "bool" + }, + "alternate_address": { + "type": "bool" + }, + "conversion_error": { + "type": "bool" + }, + "dod_host_prohibited": { + "type": "bool" + }, + "dod_net_prohibited": { + "type": "bool" + }, + "echo": {"type": "bool"}, + "echo_reply": {"type": "bool"}, + "general_parameter_problem": { + "type": "bool" + }, + "host_isolated": { + "type": "bool" + }, + "host_precedence_unreachable": { + "type": "bool" + }, + "host_redirect": { + "type": "bool" + }, + "host_tos_redirect": { + "type": "bool" + }, + "host_tos_unreachable": { + "type": "bool" + }, + "host_unknown": { + "type": "bool" + }, + "host_unreachable": { + "type": "bool" + }, + "information_reply": { + "type": "bool" + }, + "information_request": { + "type": "bool" + }, + "mask_reply": {"type": "bool"}, + "mask_request": { + "type": "bool" + }, + "message_code": { + "type": "int" + }, + "message_num": {"type": "int"}, + "message_type": { + "type": "int" + }, + "mobile_redirect": { + "type": "bool" + }, + "net_redirect": { + "type": "bool" + }, + "net_tos_redirect": { + "type": "bool" + }, + "net_tos_unreachable": { + "type": "bool" + }, + "net_unreachable": { + "type": "bool" + }, + "network_unknown": { + "type": "bool" + }, + "no_room_for_option": { + "type": "bool" + }, + "option_missing": { + "type": "bool" + }, + "packet_too_big": { + "type": "bool" + }, + "parameter_problem": { + "type": "bool" + }, + "port_unreachable": { + "type": "bool" + }, + "precedence_unreachable": { + "type": "bool" + }, + "protocol_unreachable": { + "type": "bool" + }, + "reassembly_timeout": { + "type": "bool" + }, + "redirect": {"type": "bool"}, + "router_advertisement": { + "type": "bool" + }, + "router_solicitation": { + "type": "bool" + }, + "source_quench": { + "type": "bool" + }, + "source_route_failed": { + "type": "bool" + }, + "time_exceeded": { + "type": "bool" + }, + "timestamp_reply": { + "type": "bool" + }, + "timestamp_request": { + "type": "bool" + }, + "traceroute": {"type": "bool"}, + "ttl_exceeded": { + "type": "bool" + }, + "unreachable": { + "type": "bool" + }, + }, + "type": "dict", + }, + "icmpv6": { + "options": { + "address_unreachable": { + "type": "bool" + }, + "beyond_scope": { + "type": "bool" + }, + "echo_reply": {"type": "bool"}, + "echo_request": { + "type": "bool" + }, + "erroneous_header": { + "type": "bool" + }, + "fragment_reassembly_exceeded": { + "type": "bool" + }, + "hop_limit_exceeded": { + "type": "bool" + }, + "neighbor_advertisement": { + "type": "bool" + }, + "neighbor_solicitation": { + "type": "bool" + }, + "no_admin": {"type": "bool"}, + "no_route": {"type": "bool"}, + "packet_too_big": { + "type": "bool" + }, + "parameter_problem": { + "type": "bool" + }, + "port_unreachable": { + "type": "bool" + }, + "redirect_message": { + "type": "bool" + }, + "reject_route": { + "type": "bool" + }, + "router_advertisement": { + "type": "bool" + }, + "router_solicitation": { + "type": "bool" + }, + "source_address_failed": { + "type": "bool" + }, + "source_routing_error": { + "type": "bool" + }, + "time_exceeded": { + "type": "bool" + }, + "unreachable": { + "type": "bool" + }, + "unrecognized_ipv6_option": { + "type": "bool" + }, + "unrecognized_next_header": { + "type": "bool" + }, + }, + "type": "dict", + }, + "ip": { + "options": { + "nexthop_group": { + "type": "str" + } + }, + "type": "dict", + }, + "ipv6": { + "options": { + "nexthop_group": { + "type": "str" + } + }, + "type": "dict", + }, + "tcp": { + "options": { + "flags": { + "options": { + "ack": { + "type": "bool" + }, + "established": { + "type": "bool" + }, + "fin": { + "type": "bool" + }, + "psh": { + "type": "bool" + }, + "rst": { + "type": "bool" + }, + "syn": { + "type": "bool" + }, + "urg": { + "type": "bool" + }, + }, + "type": "dict", + } + }, + "type": "dict", + }, + }, + "type": "dict", + }, + "remark": {"type": "str"}, + "sequence": {"type": "int"}, + "source": { + "mutually_exclusive": [ + [ + "address", + "subnet_address", + "any", + "host", + ], + [ + "wildcard_bits", + "subnet_address", + "any", + "host", + ], + ], + "options": { + "address": {"type": "str"}, + "any": {"type": "bool"}, + "host": {"type": "str"}, + "port_protocol": {"type": "dict"}, + "subnet_address": {"type": "str"}, + "wildcard_bits": {"type": "str"}, + }, + "required_together": [ + ["address", "wildcard_bits"] + ], + "type": "dict", + }, + "tracked": {"type": "bool"}, + "ttl": { + "options": { + "eq": {"type": "int"}, + "gt": {"type": "int"}, + "lt": {"type": "int"}, + "neq": {"type": "int"}, + }, + "type": "dict", + }, + "vlan": {"type": "str"}, + }, + "type": "list", + }, + "name": {"required": True, "type": "str"}, + "standard": {"type": "bool"}, + }, + "type": "list", + }, + "afi": { + "choices": ["ipv4", "ipv6"], + "required": True, + "type": "str", + }, + }, + "type": "list", + }, + "running_config": {"type": "str"}, + "state": { + "choices": [ + "deleted", + "merged", + "overridden", + "replaced", + "gathered", + "rendered", + "parsed", + ], + "default": "merged", + "type": "str", + }, + } # pylint: disable=C0301 diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/bgp_address_family/__init__.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/bgp_address_family/__init__.py new file mode 100644 index 00000000..e69de29b --- /dev/null +++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/bgp_address_family/__init__.py diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/bgp_address_family/bgp_address_family.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/bgp_address_family/bgp_address_family.py new file mode 100644 index 00000000..aa488920 --- /dev/null +++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/bgp_address_family/bgp_address_family.py @@ -0,0 +1,195 @@ +# -*- coding: utf-8 -*- +# Copyright 2021 Red Hat +# 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 + +############################################# +# WARNING # +############################################# +# +# This file is auto generated by the resource +# module builder playbook. +# +# Do not edit this file manually. +# +# Changes to this file will be over written +# by the resource module builder. +# +# Changes should be made in the model used to +# generate this file or in the resource module +# builder template. +# +############################################# + +""" +The arg spec for the eos_bgp_address_family module +""" + + +class Bgp_afArgs(object): # pylint: disable=R0903 + """The arg spec for the eos_bgp_address_family module + """ + + def __init__(self, **kwargs): + pass + + argument_spec = { + "running_config": {"type": "str"}, + "state": { + "default": "merged", + "type": "str", + "choices": [ + "deleted", + "merged", + "overridden", + "replaced", + "gathered", + "rendered", + "parsed", + ], + }, + "config": { + "type": "dict", + "options": { + "as_number": {"type": "str"}, + "address_family": { + "elements": "dict", + "type": "list", + "options": { + "network": { + "type": "list", + "elements": "dict", + "options": { + "route_map": {"type": "str"}, + "address": {"type": "str"}, + }, + }, + "redistribute": { + "elements": "dict", + "type": "list", + "options": { + "ospf_route": { + "type": "str", + "choices": [ + "internal", + "external", + "nssa_external_1", + "nssa_external_2", + ], + }, + "route_map": {"type": "str"}, + "protocol": { + "type": "str", + "choices": ["isis", "ospf3", "dhcp"], + }, + "isis_level": { + "type": "str", + "choices": [ + "level-1", + "level-2", + "level-1-2", + ], + }, + }, + }, + "route_target": { + "type": "dict", + "options": { + "mode": { + "type": "str", + "choices": ["both", "import", "export"], + }, + "target": {"type": "str"}, + }, + }, + "graceful_restart": {"type": "bool"}, + "bgp_params": { + "type": "dict", + "options": { + "next_hop_address_family": { + "type": "str", + "choices": ["ipv6"], + }, + "redistribute_internal": {"type": "bool"}, + "route": {"type": "str"}, + "additional_paths": { + "type": "str", + "choices": ["install", "send", "receive"], + }, + "next_hop_unchanged": {"type": "bool"}, + }, + }, + "safi": { + "type": "str", + "choices": ["labeled-unicast", "multicast"], + }, + "neighbor": { + "elements": "dict", + "type": "list", + "options": { + "activate": {"type": "bool"}, + "graceful_restart": {"type": "bool"}, + "weight": {"type": "int"}, + "default_originate": { + "type": "dict", + "options": { + "route_map": {"type": "str"}, + "always": {"type": "bool"}, + }, + }, + "route_map": { + "type": "dict", + "options": { + "direction": { + "type": "str", + "choices": ["in", "out"], + }, + "name": {"type": "str"}, + }, + }, + "next_hop_address_family": { + "type": "str", + "choices": ["ipv6"], + }, + "additional_paths": { + "type": "str", + "choices": ["send", "receive"], + }, + "peer": {"type": "str"}, + "encapsulation": { + "type": "dict", + "options": { + "transport": { + "type": "str", + "choices": ["mpls", "vxlan"], + }, + "source_interface": {"type": "str"}, + }, + }, + "prefix_list": { + "type": "dict", + "options": { + "direction": { + "type": "str", + "choices": ["in", "out"], + }, + "name": {"type": "str"}, + }, + }, + "next_hop_unchanged": {"type": "bool"}, + }, + }, + "afi": { + "type": "str", + "choices": ["ipv4", "ipv6", "evpn"], + }, + "vrf": {"type": "str"}, + }, + }, + }, + }, + } # pylint: disable=C0301 diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/bgp_global/__init__.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/bgp_global/__init__.py new file mode 100644 index 00000000..e69de29b --- /dev/null +++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/bgp_global/__init__.py diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/bgp_global/bgp_global.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/bgp_global/bgp_global.py new file mode 100644 index 00000000..cd291d05 --- /dev/null +++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/bgp_global/bgp_global.py @@ -0,0 +1,999 @@ +# -*- coding: utf-8 -*- +# Copyright 2020 Red Hat +# 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 + +############################################# +# WARNING # +############################################# +# +# This file is auto generated by the resource +# module builder playbook. +# +# Do not edit this file manually. +# +# Changes to this file will be over written +# by the resource module builder. +# +# Changes should be made in the model used to +# generate this file or in the resource module +# builder template. +# +############################################# + +""" +The arg spec for the eos_bgp_global module +""" + + +class Bgp_globalArgs(object): # pylint: disable=R0903 + """The arg spec for the eos_bgp_global module + """ + + def __init__(self, **kwargs): + pass + + argument_spec = { + "running_config": {"type": "str"}, + "state": { + "default": "merged", + "type": "str", + "choices": [ + "deleted", + "purged", + "merged", + "replaced", + "gathered", + "rendered", + "parsed", + ], + }, + "config": { + "type": "dict", + "options": { + "router_id": {"type": "str"}, + "as_number": {"type": "str"}, + "redistribute": { + "elements": "dict", + "type": "list", + "options": { + "ospf_route": { + "type": "str", + "choices": [ + "internal", + "external", + "nssa_external_1", + "nssa_external_2", + ], + }, + "route_map": {"type": "str"}, + "protocol": { + "type": "str", + "choices": [ + "isis", + "ospf3", + "ospf", + "attached-host", + "connected", + "rip", + "static", + ], + }, + "isis_level": { + "type": "str", + "choices": ["level-1", "level-2", "level-1-2"], + }, + }, + }, + "monitoring": { + "type": "dict", + "options": { + "received": { + "type": "str", + "choices": ["post_policy", "pre_policy"], + }, + "station": {"type": "str"}, + "port": {"type": "int"}, + "timestamp": { + "type": "str", + "choices": ["none", "send_time"], + }, + }, + }, + "default_metric": {"type": "int"}, + "bgp_params": { + "type": "dict", + "options": { + "labeled_unicast": { + "type": "str", + "choices": ["ip", "tunnel"], + }, + "host_routes": {"type": "bool"}, + "transport": {"type": "int"}, + "next_hop_unchanged": {"type": "bool"}, + "missing_policy": { + "type": "dict", + "options": { + "action": { + "type": "str", + "choices": [ + "deny", + "permit", + "deny-in-out", + ], + }, + "direction": { + "type": "str", + "choices": ["in", "out"], + }, + }, + }, + "monitoring": {"type": "bool"}, + "allowas_in": { + "type": "dict", + "options": { + "count": {"type": "int"}, + "set": {"type": "bool"}, + }, + }, + "additional_paths": { + "type": "str", + "choices": ["install", "send", "receive"], + }, + "advertise_inactive": {"type": "bool"}, + "listen": { + "type": "dict", + "options": { + "range": { + "type": "dict", + "options": { + "peer_group": { + "type": "dict", + "options": { + "peer_filter": {"type": "str"}, + "remote_as": {"type": "str"}, + "name": {"type": "str"}, + }, + }, + "address": {"type": "str"}, + }, + }, + "limit": {"type": "int"}, + }, + }, + "route_reflector": { + "type": "dict", + "options": { + "preserve": {"type": "bool"}, + "set": {"type": "bool"}, + }, + }, + "always_compare_med": {"type": "bool"}, + "client_to_client": {"type": "bool"}, + "bestpath": { + "type": "dict", + "options": { + "ecmp_fast": {"type": "bool"}, + "tie_break": { + "type": "str", + "choices": [ + "cluster_list_length", + "router_id", + ], + }, + "skip": {"type": "bool"}, + "as_path": { + "type": "str", + "choices": ["ignore", "multipath_relax"], + }, + "med": { + "type": "dict", + "options": { + "confed": {"type": "bool"}, + "missing_as_worst": {"type": "bool"}, + }, + }, + }, + }, + "convergence": { + "type": "dict", + "options": { + "slow_peer": {"type": "bool"}, + "time": {"type": "int"}, + }, + }, + "log_neighbor_changes": {"type": "bool"}, + "asn": { + "type": "str", + "choices": ["asdot", "asplain"], + }, + "default": { + "type": "str", + "choices": ["ipv4_unicast", "ipv6_unicast"], + }, + "route": {"type": "str"}, + "enforce_first_as": {"type": "bool"}, + "auto_local_addr": {"type": "bool"}, + "redistribute_internal": {"type": "bool"}, + "cluster_id": {"type": "str"}, + "control_plan_filter": {"type": "bool"}, + "confederation": { + "type": "dict", + "options": { + "peers": {"type": "str"}, + "identifier": {"type": "str"}, + }, + }, + }, + }, + "vlan": {"type": "int"}, + "update": { + "type": "dict", + "options": { + "wait_for": { + "type": "str", + "choices": [ + "wait_for_convergence", + "wait_install", + ], + }, + "batch_size": {"type": "int"}, + }, + }, + "vlan_aware_bundle": {"type": "str"}, + "aggregate_address": { + "elements": "dict", + "type": "list", + "options": { + "advertise_only": {"type": "bool"}, + "match_map": {"type": "str"}, + "attribute_map": {"type": "str"}, + "as_set": {"type": "bool"}, + "summary_only": {"type": "bool"}, + "address": {"type": "str"}, + }, + }, + "neighbor": { + "elements": "dict", + "type": "list", + "options": { + "weight": {"type": "int"}, + "default_originate": { + "type": "dict", + "options": { + "route_map": {"type": "str"}, + "always": {"type": "bool"}, + }, + }, + "next_hop_v6_address": {"type": "str"}, + "route_reflector_client": {"type": "bool"}, + "ttl": {"type": "int"}, + "remove_private_as": { + "type": "dict", + "options": { + "all": {"type": "bool"}, + "set": {"type": "bool"}, + "replace_as": {"type": "bool"}, + }, + }, + "local_v6_addr": {"type": "str"}, + "transport": { + "type": "dict", + "options": { + "connection_mode": {"type": "str"}, + "remote_port": {"type": "int"}, + }, + }, + "next_hop_unchanged": {"type": "bool"}, + "monitoring": {"type": "bool"}, + "ebgp_multihop": { + "type": "dict", + "options": { + "set": {"type": "bool"}, + "ttl": {"type": "int"}, + }, + }, + "shut_down": {"type": "bool"}, + "fall_over": {"type": "bool"}, + "idle_restart_timer": {"type": "int"}, + "allowas_in": { + "type": "dict", + "options": { + "count": {"type": "int"}, + "set": {"type": "bool"}, + }, + }, + "additional_paths": { + "type": "str", + "choices": ["send", "receive"], + }, + "peer_group": {"type": "str"}, + "out_delay": {"type": "int"}, + "import_localpref": {"type": "int"}, + "prefix_list": { + "type": "dict", + "options": { + "direction": { + "type": "str", + "choices": ["in", "out"], + }, + "name": {"type": "str"}, + }, + }, + "dont_capability_negotiate": {"type": "bool"}, + "update_source": {"type": "str"}, + "export_localpref": {"type": "int"}, + "local_as": { + "type": "dict", + "options": { + "as_number": {"type": "str"}, + "fallback": {"type": "bool"}, + }, + }, + "maximum_received_routes": { + "type": "dict", + "options": { + "count": {"type": "int"}, + "warning_limit": { + "type": "dict", + "options": { + "limit_count": {"type": "int"}, + "limit_percent": {"type": "int"}, + }, + }, + "warning_only": {"type": "bool"}, + }, + }, + "encryption_password": { + "type": "dict", + "options": { + "password": {"type": "str"}, + "type": {"type": "int", "choices": [0, 7]}, + }, + }, + "link_bandwidth": { + "type": "dict", + "options": { + "default": {"type": "str"}, + "auto": {"type": "bool"}, + "set": {"type": "bool"}, + "update_delay": {"type": "int"}, + }, + }, + "graceful_restart_helper": {"type": "bool"}, + "peer": {"type": "str"}, + "next_hop_self": {"type": "bool"}, + "route_to_peer": {"type": "bool"}, + "soft_recognition": { + "type": "str", + "choices": ["all", "None"], + }, + "graceful_restart": {"type": "bool"}, + "enforce_first_as": {"type": "bool"}, + "send_community": { + "type": "dict", + "options": { + "community_attribute": {"type": "str"}, + "sub_attribute": { + "type": "str", + "choices": [ + "extended", + "link-bandwidth", + "standard", + ], + }, + "speed": {"type": "str"}, + "divide": { + "type": "str", + "choices": ["equal", "ratio"], + }, + "link_bandwidth_attribute": { + "type": "str", + "choices": ["aggregate", "divide"], + }, + }, + }, + "description": {"type": "str"}, + "maximum_accepted_routes": { + "type": "dict", + "options": { + "count": {"type": "int"}, + "warning_limit": {"type": "int"}, + }, + }, + "auto_local_addr": {"type": "bool"}, + "metric_out": {"type": "int"}, + "timers": { + "type": "dict", + "options": { + "holdtime": {"type": "int"}, + "keepalive": {"type": "int"}, + }, + }, + "route_map": { + "type": "dict", + "options": { + "direction": { + "type": "str", + "choices": ["in", "out"], + }, + "name": {"type": "str"}, + }, + }, + "remote_as": {"type": "str"}, + }, + }, + "graceful_restart": { + "type": "dict", + "options": { + "stalepath_time": {"type": "int"}, + "restart_time": {"type": "int"}, + "set": {"type": "bool"}, + }, + }, + "timers": { + "type": "dict", + "options": { + "holdtime": {"type": "int"}, + "keepalive": {"type": "int"}, + }, + }, + "distance": { + "type": "dict", + "options": { + "internal": {"type": "int"}, + "local": {"type": "int"}, + "external": {"type": "int"}, + }, + }, + "route_target": { + "type": "dict", + "options": { + "action": { + "type": "str", + "choices": ["both", "import", "export"], + }, + "target": {"type": "str"}, + }, + }, + "vrfs": { + "elements": "dict", + "type": "list", + "options": { + "router_id": {"type": "str"}, + "vrf": {"type": "str"}, + "route_target": { + "type": "dict", + "options": { + "action": { + "type": "str", + "choices": ["both", "import", "export"], + }, + "target": {"type": "str"}, + }, + }, + "redistribute": { + "elements": "dict", + "type": "list", + "options": { + "ospf_route": { + "type": "str", + "choices": [ + "internal", + "external", + "nssa_external_1", + "nssa_external_2", + ], + }, + "route_map": {"type": "str"}, + "protocol": { + "type": "str", + "choices": [ + "isis", + "ospf3", + "ospf", + "attached-host", + "connected", + "rip", + "static", + ], + }, + "isis_level": { + "type": "str", + "choices": [ + "level-1", + "level-2", + "level-1-2", + ], + }, + }, + }, + "distance": { + "type": "dict", + "options": { + "internal": {"type": "int"}, + "local": {"type": "int"}, + "external": {"type": "int"}, + }, + }, + "default_metric": {"type": "int"}, + "bgp_params": { + "type": "dict", + "options": { + "control_plane_filter": {"type": "bool"}, + "convergence": { + "type": "dict", + "options": { + "slow_peer": {"type": "bool"}, + "time": {"type": "int"}, + }, + }, + "host_routes": {"type": "bool"}, + "transport": {"type": "int"}, + "next_hop_unchanged": {"type": "bool"}, + "missing_policy": { + "type": "dict", + "options": { + "action": { + "type": "str", + "choices": [ + "deny", + "permit", + "deny-in-out", + ], + }, + "direction": { + "type": "str", + "choices": ["in", "out"], + }, + }, + }, + "monitoring": {"type": "bool"}, + "allowas_in": { + "type": "dict", + "options": { + "count": {"type": "int"}, + "set": {"type": "bool"}, + }, + }, + "additional_paths": { + "type": "str", + "choices": ["install", "send", "receive"], + }, + "advertise_inactive": {"type": "bool"}, + "listen": { + "type": "dict", + "options": { + "range": { + "type": "dict", + "options": { + "peer_group": { + "type": "dict", + "options": { + "peer_filter": { + "type": "str" + }, + "remote_as": { + "type": "str" + }, + "name": { + "type": "str" + }, + }, + }, + "address": {"type": "str"}, + }, + }, + "limit": {"type": "int"}, + }, + }, + "route_reflector": { + "type": "dict", + "options": { + "preserve": {"type": "bool"}, + "set": {"type": "bool"}, + }, + }, + "always_compare_med": {"type": "bool"}, + "client_to_client": {"type": "bool"}, + "bestpath": { + "type": "dict", + "options": { + "ecmp_fast": {"type": "bool"}, + "tie_break": { + "type": "str", + "choices": [ + "cluster_list_length", + "router_id", + ], + }, + "skip": {"type": "bool"}, + "as_path": { + "type": "str", + "choices": [ + "ignore", + "multipath_relax", + ], + }, + "med": { + "type": "dict", + "options": { + "confed": {"type": "bool"}, + "missing_as_worst": { + "type": "bool" + }, + }, + }, + }, + }, + "labeled_unicast": { + "type": "str", + "choices": ["ip", "tunnel"], + }, + "log_neighbor_changes": {"type": "bool"}, + "asn": { + "type": "str", + "choices": ["asdot", "asplain"], + }, + "default": { + "type": "str", + "choices": [ + "ipv4_unicast", + "ipv6_unicast", + ], + }, + "route": {"type": "str"}, + "enforce_first_as": {"type": "bool"}, + "auto_local_addr": {"type": "bool"}, + "redistribute_internal": {"type": "bool"}, + "cluster_id": {"type": "str"}, + "confederation": { + "type": "dict", + "options": { + "peers": {"type": "str"}, + "identifier": {"type": "str"}, + }, + }, + }, + }, + "update": { + "type": "dict", + "options": { + "wait_for": { + "type": "str", + "choices": [ + "wait_for_convergence", + "wait_install", + ], + }, + "batch_size": {"type": "int"}, + }, + }, + "aggregate_address": { + "elements": "dict", + "type": "list", + "options": { + "advertise_only": {"type": "bool"}, + "match_map": {"type": "str"}, + "attribute_map": {"type": "str"}, + "as_set": {"type": "bool"}, + "summary_only": {"type": "bool"}, + "address": {"type": "str"}, + }, + }, + "neighbor": { + "elements": "dict", + "type": "list", + "options": { + "weight": {"type": "int"}, + "default_originate": { + "type": "dict", + "options": { + "route_map": {"type": "str"}, + "always": {"type": "bool"}, + }, + }, + "next_hop_v6_address": {"type": "str"}, + "route_reflector_client": {"type": "bool"}, + "ttl": {"type": "int"}, + "remove_private_as": { + "type": "dict", + "options": { + "all": {"type": "bool"}, + "set": {"type": "bool"}, + "replace_as": {"type": "bool"}, + }, + }, + "local_v6_addr": {"type": "str"}, + "transport": { + "type": "dict", + "options": { + "connection_mode": {"type": "str"}, + "remote_port": {"type": "int"}, + }, + }, + "next_hop_unchanged": {"type": "bool"}, + "monitoring": {"type": "bool"}, + "ebgp_multihop": { + "type": "dict", + "options": { + "set": {"type": "bool"}, + "ttl": {"type": "int"}, + }, + }, + "shut_down": {"type": "bool"}, + "fall_over": {"type": "bool"}, + "idle_restart_timer": {"type": "int"}, + "allowas_in": { + "type": "dict", + "options": { + "count": {"type": "int"}, + "set": {"type": "bool"}, + }, + }, + "additional_paths": { + "type": "str", + "choices": ["send", "receive"], + }, + "peer_group": {"type": "str"}, + "out_delay": {"type": "int"}, + "import_localpref": {"type": "int"}, + "prefix_list": { + "type": "dict", + "options": { + "direction": { + "type": "str", + "choices": ["in", "out"], + }, + "name": {"type": "str"}, + }, + }, + "dont_capability_negotiate": {"type": "bool"}, + "update_source": {"type": "str"}, + "export_localpref": {"type": "int"}, + "local_as": { + "type": "dict", + "options": { + "as_number": {"type": "str"}, + "fallback": {"type": "bool"}, + }, + }, + "maximum_received_routes": { + "type": "dict", + "options": { + "count": {"type": "int"}, + "warning_limit": { + "type": "dict", + "options": { + "limit_count": {"type": "int"}, + "limit_percent": { + "type": "int" + }, + }, + }, + "warning_only": {"type": "bool"}, + }, + }, + "encryption_password": { + "type": "dict", + "options": { + "password": {"type": "str"}, + "type": { + "type": "int", + "choices": [0, 7], + }, + }, + }, + "link_bandwidth": { + "type": "dict", + "options": { + "default": {"type": "str"}, + "auto": {"type": "bool"}, + "set": {"type": "bool"}, + "update_delay": {"type": "int"}, + }, + }, + "graceful_restart_helper": {"type": "bool"}, + "peer": {"type": "str"}, + "next_hop_self": {"type": "bool"}, + "route_to_peer": {"type": "bool"}, + "soft_recognition": { + "type": "str", + "choices": ["all", "None"], + }, + "graceful_restart": {"type": "bool"}, + "enforce_first_as": {"type": "bool"}, + "send_community": { + "type": "dict", + "options": { + "community_attribute": {"type": "str"}, + "sub_attribute": { + "type": "str", + "choices": [ + "extended", + "link-bandwidth", + "standard", + ], + }, + "speed": {"type": "str"}, + "divide": { + "type": "str", + "choices": ["equal", "ratio"], + }, + "link_bandwidth_attribute": { + "type": "str", + "choices": ["aggregate", "divide"], + }, + }, + }, + "description": {"type": "str"}, + "maximum_accepted_routes": { + "type": "dict", + "options": { + "count": {"type": "int"}, + "warning_limit": {"type": "int"}, + }, + }, + "auto_local_addr": {"type": "bool"}, + "metric_out": {"type": "int"}, + "timers": { + "type": "dict", + "options": { + "holdtime": {"type": "int"}, + "keepalive": {"type": "int"}, + }, + }, + "route_map": { + "type": "dict", + "options": { + "direction": { + "type": "str", + "choices": ["in", "out"], + }, + "name": {"type": "str"}, + }, + }, + "remote_as": {"type": "str"}, + }, + }, + "graceful_restart": { + "type": "dict", + "options": { + "stalepath_time": {"type": "int"}, + "restart_time": {"type": "int"}, + "set": {"type": "bool"}, + }, + }, + "timers": { + "type": "dict", + "options": { + "holdtime": {"type": "int"}, + "keepalive": {"type": "int"}, + }, + }, + "shutdown": {"type": "bool"}, + "access_group": { + "type": "dict", + "options": { + "direction": {"type": "str"}, + "afi": { + "type": "str", + "choices": ["ip", "ipv6"], + }, + "acl_name": {"type": "str"}, + }, + }, + "graceful_restart_helper": {"type": "bool"}, + "ucmp": { + "type": "dict", + "options": { + "link_bandwidth": { + "type": "dict", + "options": { + "update_delay": {"type": "int"}, + "mode": { + "type": "str", + "choices": [ + "encoding_weighted", + "recursive", + ], + }, + }, + }, + "fec": { + "type": "dict", + "options": { + "clear": {"type": "int"}, + "trigger": {"type": "int"}, + }, + }, + "mode": { + "type": "dict", + "options": { + "set": {"type": "bool"}, + "nexthops": {"type": "int"}, + }, + }, + }, + }, + "maximum_paths": { + "type": "dict", + "options": { + "max_equal_cost_paths": {"type": "int"}, + "max_installed_ecmp_paths": {"type": "int"}, + }, + }, + "network": { + "type": "list", + "elements": "dict", + "options": { + "route_map": {"type": "str"}, + "address": {"type": "str"}, + }, + }, + }, + }, + "access_group": { + "type": "dict", + "options": { + "direction": {"type": "str"}, + "afi": {"type": "str", "choices": ["ipv4", "ipv6"]}, + "acl_name": {"type": "str"}, + }, + }, + "graceful_restart_helper": {"type": "bool"}, + "ucmp": { + "type": "dict", + "options": { + "link_bandwidth": { + "type": "dict", + "options": { + "update_delay": {"type": "int"}, + "mode": { + "type": "str", + "choices": [ + "encoding_weighted", + "recursive", + ], + }, + }, + }, + "fec": { + "type": "dict", + "options": { + "clear": {"type": "int"}, + "trigger": {"type": "int"}, + }, + }, + "mode": { + "type": "dict", + "options": { + "set": {"type": "bool"}, + "nexthops": {"type": "int"}, + }, + }, + }, + }, + "shutdown": {"type": "bool"}, + "maximum_paths": { + "type": "dict", + "options": { + "max_equal_cost_paths": {"type": "int"}, + "max_installed_ecmp_paths": {"type": "int"}, + }, + }, + "network": { + "type": "list", + "elements": "dict", + "options": { + "route_map": {"type": "str"}, + "address": {"type": "str"}, + }, + }, + }, + }, + } # pylint: disable=C0301 diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/facts/__init__.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/facts/__init__.py new file mode 100644 index 00000000..e69de29b --- /dev/null +++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/facts/__init__.py diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/facts/facts.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/facts/facts.py new file mode 100644 index 00000000..5d7566d9 --- /dev/null +++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/facts/facts.py @@ -0,0 +1,25 @@ +# -*- coding: utf-8 -*- +# Copyright 2019 Red Hat +# GNU General Public License v3.0+ (see COPYING or https://www.gnu.org/licenses/gpl-3.0.txt) +""" +The arg spec for the eos facts module. +""" + +from __future__ import absolute_import, division, print_function + +__metaclass__ = type + + +class FactsArgs(object): + """ The arg spec for the eos facts module + """ + + def __init__(self, **kwargs): + pass + + argument_spec = { + "gather_subset": dict( + default=["!config"], type="list", elements="str" + ), + "gather_network_resources": dict(type="list", elements="str"), + } diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/interfaces/__init__.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/interfaces/__init__.py new file mode 100644 index 00000000..e69de29b --- /dev/null +++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/interfaces/__init__.py diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/interfaces/interfaces.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/interfaces/interfaces.py new file mode 100644 index 00000000..00710be7 --- /dev/null +++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/interfaces/interfaces.py @@ -0,0 +1,72 @@ +# -*- coding: utf-8 -*- +# Copyright 2019 Red Hat +# GNU General Public License v3.0+ (see COPYING or https://www.gnu.org/licenses/gpl-3.0.txt) + +############################################## +# WARNING # +############################################## +# +# This file is auto generated by the resource +# module builder playbook. +# +# Do not edit this file manually. +# +# Changes to this file will be over written +# by the resource module builder. +# +# Changes should be made in the model used to +# generate this file or in the resource module +# builder template. +# +############################################## + +""" +The arg spec for the eos_interfaces module +""" + +from __future__ import absolute_import, division, print_function + +__metaclass__ = type + + +class InterfacesArgs(object): + """The arg spec for the eos_interfaces module + """ + + def __init__(self, **kwargs): + pass + + argument_spec = { + "config": { + "elements": "dict", + "options": { + "name": {"required": True, "type": "str"}, + "description": {"required": False, "type": "str"}, + "enabled": { + "default": True, + "required": False, + "type": "bool", + }, + "mtu": {"required": False, "type": "int"}, + "speed": {"required": False, "type": "str"}, + "duplex": {"required": False, "type": "str"}, + "mode": {"choices": ["layer2", "layer3"], "type": "str"}, + }, + "type": "list", + }, + "running_config": {"type": "str"}, + "state": { + "default": "merged", + "choices": [ + "merged", + "replaced", + "overridden", + "deleted", + "gathered", + "rendered", + "parsed", + ], + "required": False, + "type": "str", + }, + } diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/l2_interfaces/__init__.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/l2_interfaces/__init__.py new file mode 100644 index 00000000..e69de29b --- /dev/null +++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/l2_interfaces/__init__.py diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/l2_interfaces/l2_interfaces.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/l2_interfaces/l2_interfaces.py new file mode 100644 index 00000000..2a8b7f86 --- /dev/null +++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/l2_interfaces/l2_interfaces.py @@ -0,0 +1,77 @@ +# -*- coding: utf-8 -*- +# Copyright 2019 Red Hat +# GNU General Public License v3.0+ (see COPYING or https://www.gnu.org/licenses/gpl-3.0.txt) + +############################################## +# WARNING # +############################################## +# +# This file is auto generated by the resource +# module builder playbook. +# +# Do not edit this file manually. +# +# Changes to this file will be over written +# by the resource module builder. +# +# Changes should be made in the model used to +# generate this file or in the resource module +# builder template. +# +############################################## + +""" +The arg spec for the eos_l2_interfaces module +""" + +from __future__ import absolute_import, division, print_function + +__metaclass__ = type + + +class L2_interfacesArgs(object): + """The arg spec for the eos_l2_interfaces module + """ + + def __init__(self, **kwargs): + pass + + argument_spec = { + "config": { + "elements": "dict", + "options": { + "access": { + "options": {"vlan": {"type": "int"}}, + "type": "dict", + }, + "mode": {"type": "str", "choices": ["access", "trunk"]}, + "name": {"required": True, "type": "str"}, + "trunk": { + "options": { + "native_vlan": {"type": "int"}, + "trunk_allowed_vlans": { + "type": "list", + "elements": "str", + }, + }, + "type": "dict", + }, + }, + "type": "list", + }, + "running_config": {"type": "str"}, + "state": { + "default": "merged", + "choices": [ + "merged", + "replaced", + "overridden", + "deleted", + "gathered", + "rendered", + "parsed", + ], + "required": False, + "type": "str", + }, + } diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/l3_interfaces/__init__.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/l3_interfaces/__init__.py new file mode 100644 index 00000000..e69de29b --- /dev/null +++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/l3_interfaces/__init__.py diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/l3_interfaces/l3_interfaces.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/l3_interfaces/l3_interfaces.py new file mode 100644 index 00000000..ae8738fa --- /dev/null +++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/l3_interfaces/l3_interfaces.py @@ -0,0 +1,75 @@ +# -*- coding: utf-8 -*- +# Copyright 2019 Red Hat +# GNU General Public License v3.0+ +# (see COPYING or https://www.gnu.org/licenses/gpl-3.0.txt) + +############################################# +# WARNING # +############################################# +# +# This file is auto generated by the resource +# module builder playbook. +# +# Do not edit this file manually. +# +# Changes to this file will be over written +# by the resource module builder. +# +# Changes should be made in the model used to +# generate this file or in the resource module +# builder template. +# +############################################# + +""" +The arg spec for the eos_l3_interfaces module +""" + +from __future__ import absolute_import, division, print_function + +__metaclass__ = type + + +class L3_interfacesArgs(object): + """The arg spec for the eos_l3_interfaces module + """ + + def __init__(self, **kwargs): + pass + + argument_spec = { + "config": { + "elements": "dict", + "options": { + "name": {"required": True, "type": "str"}, + "ipv4": { + "elements": "dict", + "options": { + "address": {"type": "str"}, + "secondary": {"type": "bool"}, + }, + "type": "list", + }, + "ipv6": { + "elements": "dict", + "options": {"address": {"type": "str"}}, + "type": "list", + }, + }, + "type": "list", + }, + "running_config": {"type": "str"}, + "state": { + "default": "merged", + "choices": [ + "merged", + "replaced", + "overridden", + "deleted", + "gathered", + "rendered", + "parsed", + ], + "type": "str", + }, + } diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/lacp/__init__.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/lacp/__init__.py new file mode 100644 index 00000000..e69de29b --- /dev/null +++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/lacp/__init__.py diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/lacp/lacp.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/lacp/lacp.py new file mode 100644 index 00000000..6c913c01 --- /dev/null +++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/lacp/lacp.py @@ -0,0 +1,63 @@ +# -*- coding: utf-8 -*- +# Copyright 2019 Red Hat +# GNU General Public License v3.0+ +# (see COPYING or https://www.gnu.org/licenses/gpl-3.0.txt) + +############################################# +# WARNING # +############################################# +# +# This file is auto generated by the resource +# module builder playbook. +# +# Do not edit this file manually. +# +# Changes to this file will be over written +# by the resource module builder. +# +# Changes should be made in the model used to +# generate this file or in the resource module +# builder template. +# +############################################# + +""" +The arg spec for the eos_lacp module +""" + +from __future__ import absolute_import, division, print_function + +__metaclass__ = type + + +class LacpArgs(object): + """The arg spec for the eos_lacp module + """ + + def __init__(self, **kwargs): + pass + + argument_spec = { + "config": { + "options": { + "system": { + "options": {"priority": {"type": "int"}}, + "type": "dict", + } + }, + "type": "dict", + }, + "running_config": {"type": "str"}, + "state": { + "choices": [ + "merged", + "replaced", + "deleted", + "gathered", + "rendered", + "parsed", + ], + "default": "merged", + "type": "str", + }, + } diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/lacp_interfaces/__init__.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/lacp_interfaces/__init__.py new file mode 100644 index 00000000..e69de29b --- /dev/null +++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/lacp_interfaces/__init__.py diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/lacp_interfaces/lacp_interfaces.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/lacp_interfaces/lacp_interfaces.py new file mode 100644 index 00000000..802af365 --- /dev/null +++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/lacp_interfaces/lacp_interfaces.py @@ -0,0 +1,65 @@ +# +# -*- coding: utf-8 -*- +# Copyright 2019 Red Hat +# GNU General Public License v3.0+ +# (see COPYING or https://www.gnu.org/licenses/gpl-3.0.txt) + +############################################# +# WARNING # +############################################# +# +# This file is auto generated by the resource +# module builder playbook. +# +# Do not edit this file manually. +# +# Changes to this file will be over written +# by the resource module builder. +# +# Changes should be made in the model used to +# generate this file or in the resource module +# builder template. +# +############################################# + +""" +The arg spec for the eos_lacp_interfaces module +""" + +from __future__ import absolute_import, division, print_function + +__metaclass__ = type + + +class Lacp_interfacesArgs(object): + """The arg spec for the eos_lacp_interfaces module + """ + + def __init__(self, **kwargs): + pass + + argument_spec = { + "config": { + "elements": "dict", + "options": { + "name": {"type": "str"}, + "port_priority": {"type": "int"}, + "rate": {"choices": ["fast", "normal"], "type": "str"}, + }, + "type": "list", + }, + "running_config": {"type": "str"}, + "state": { + "choices": [ + "merged", + "replaced", + "overridden", + "deleted", + "gathered", + "rendered", + "parsed", + ], + "default": "merged", + "type": "str", + }, + } diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/lag_interfaces/__init__.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/lag_interfaces/__init__.py new file mode 100644 index 00000000..e69de29b --- /dev/null +++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/lag_interfaces/__init__.py diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/lag_interfaces/lag_interfaces.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/lag_interfaces/lag_interfaces.py new file mode 100644 index 00000000..f8792d5c --- /dev/null +++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/lag_interfaces/lag_interfaces.py @@ -0,0 +1,73 @@ +# -*- coding: utf-8 -*- +# Copyright 2019 Red Hat +# GNU General Public License v3.0+ +# (see COPYING or https://www.gnu.org/licenses/gpl-3.0.txt) + +############################################# +# WARNING # +############################################# +# +# This file is auto generated by the resource +# module builder playbook. +# +# Do not edit this file manually. +# +# Changes to this file will be over written +# by the resource module builder. +# +# Changes should be made in the model used to +# generate this file or in the resource module +# builder template. +# +############################################# + +""" +The arg spec for the eos_lag_interfaces module +""" + +from __future__ import absolute_import, division, print_function + +__metaclass__ = type + + +class Lag_interfacesArgs(object): + """The arg spec for the eos_lag_interfaces module + """ + + def __init__(self, **kwargs): + pass + + argument_spec = { + "config": { + "elements": "dict", + "options": { + "name": {"required": True, "type": "str"}, + "members": { + "elements": "dict", + "options": { + "member": {"type": "str"}, + "mode": { + "choices": ["active", "on", "passive"], + "type": "str", + }, + }, + "type": "list", + }, + }, + "type": "list", + }, + "running_config": {"type": "str"}, + "state": { + "default": "merged", + "choices": [ + "merged", + "replaced", + "overridden", + "deleted", + "gathered", + "rendered", + "parsed", + ], + "type": "str", + }, + } diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/lldp_global/__init__.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/lldp_global/__init__.py new file mode 100644 index 00000000..e69de29b --- /dev/null +++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/lldp_global/__init__.py diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/lldp_global/lldp_global.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/lldp_global/lldp_global.py new file mode 100644 index 00000000..97094fac --- /dev/null +++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/lldp_global/lldp_global.py @@ -0,0 +1,75 @@ +# +# -*- coding: utf-8 -*- +# Copyright 2019 Red Hat +# GNU General Public License v3.0+ +# (see COPYING or https://www.gnu.org/licenses/gpl-3.0.txt) + +############################################# +# WARNING # +############################################# +# +# This file is auto generated by the resource +# module builder playbook. +# +# Do not edit this file manually. +# +# Changes to this file will be over written +# by the resource module builder. +# +# Changes should be made in the model used to +# generate this file or in the resource module +# builder template. +# +############################################# + +""" +The arg spec for the eos_lldp_global module +""" + +from __future__ import absolute_import, division, print_function + +__metaclass__ = type + + +class Lldp_globalArgs(object): + """The arg spec for the eos_lldp_global module + """ + + def __init__(self, **kwargs): + pass + + argument_spec = { + "config": { + "options": { + "holdtime": {"type": "int"}, + "reinit": {"type": "int"}, + "timer": {"type": "int"}, + "tlv_select": { + "options": { + "link_aggregation": {"type": "bool"}, + "management_address": {"type": "bool"}, + "max_frame_size": {"type": "bool"}, + "port_description": {"type": "bool"}, + "system_capabilities": {"type": "bool"}, + "system_description": {"type": "bool"}, + "system_name": {"type": "bool"}, + }, + "type": "dict", + }, + }, + "type": "dict", + }, + "running_config": {"type": "str"}, + "state": { + "choices": [ + "merged", + "replaced", + "deleted", + "gathered", + "rendered", + "parsed", + ], + "default": "merged", + "type": "str", + }, + } diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/lldp_interfaces/__init__.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/lldp_interfaces/__init__.py new file mode 100644 index 00000000..e69de29b --- /dev/null +++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/lldp_interfaces/__init__.py diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/lldp_interfaces/lldp_interfaces.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/lldp_interfaces/lldp_interfaces.py new file mode 100644 index 00000000..01c00653 --- /dev/null +++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/lldp_interfaces/lldp_interfaces.py @@ -0,0 +1,65 @@ +# +# -*- coding: utf-8 -*- +# Copyright 2019 Red Hat +# GNU General Public License v3.0+ +# (see COPYING or https://www.gnu.org/licenses/gpl-3.0.txt) + +############################################# +# WARNING # +############################################# +# +# This file is auto generated by the resource +# module builder playbook. +# +# Do not edit this file manually. +# +# Changes to this file will be over written +# by the resource module builder. +# +# Changes should be made in the model used to +# generate this file or in the resource module +# builder template. +# +############################################# + +""" +The arg spec for the eos_lldp_interfaces module +""" + +from __future__ import absolute_import, division, print_function + +__metaclass__ = type + + +class Lldp_interfacesArgs(object): + """The arg spec for the eos_lldp_interfaces module + """ + + def __init__(self, **kwargs): + pass + + argument_spec = { + "config": { + "elements": "dict", + "options": { + "name": {"type": "str"}, + "receive": {"type": "bool"}, + "transmit": {"type": "bool"}, + }, + "type": "list", + }, + "running_config": {"type": "str"}, + "state": { + "choices": [ + "merged", + "replaced", + "overridden", + "deleted", + "gathered", + "rendered", + "parsed", + ], + "default": "merged", + "type": "str", + }, + } diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/ospf_interfaces/__init__.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/ospf_interfaces/__init__.py new file mode 100644 index 00000000..e69de29b --- /dev/null +++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/ospf_interfaces/__init__.py diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/ospf_interfaces/ospf_interfaces.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/ospf_interfaces/ospf_interfaces.py new file mode 100644 index 00000000..30a99536 --- /dev/null +++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/ospf_interfaces/ospf_interfaces.py @@ -0,0 +1,178 @@ +# -*- coding: utf-8 -*- +# Copyright 2020 Red Hat +# 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 + +############################################# +# WARNING # +############################################# +# +# This file is auto generated by the resource +# module builder playbook. +# +# Do not edit this file manually. +# +# Changes to this file will be over written +# by the resource module builder. +# +# Changes should be made in the model used to +# generate this file or in the resource module +# builder template. +# +############################################# + +""" +The arg spec for the eos_ospf_interfaces module +""" + + +class Ospf_interfacesArgs(object): # pylint: disable=R0903 + """The arg spec for the eos_ospf_interfaces module + """ + + def __init__(self, **kwargs): + pass + + argument_spec = { + "state": { + "default": "merged", + "type": "str", + "choices": [ + "merged", + "replaced", + "overridden", + "deleted", + "gathered", + "parsed", + "rendered", + ], + }, + "running_config": {"type": "str"}, + "config": { + "elements": "dict", + "type": "list", + "options": { + "name": {"type": "str"}, + "address_family": { + "elements": "dict", + "type": "list", + "options": { + "ip_params": { + "elements": "dict", + "type": "list", + "options": { + "retransmit_interval": {"type": "int"}, + "cost": {"type": "int"}, + "afi": { + "required": True, + "type": "str", + "choices": ["ipv4", "ipv6"], + }, + "area": { + "type": "dict", + "options": { + "area_id": { + "required": True, + "type": "str", + } + }, + }, + "bfd": {"type": "bool"}, + "mtu_ignore": {"type": "bool"}, + "priority": {"type": "int"}, + "dead_interval": {"type": "int"}, + "hello_interval": {"type": "int"}, + "passive_interface": {"type": "bool"}, + "transmit_delay": {"type": "int"}, + "network": {"type": "str"}, + }, + }, + "encryption_v3": { + "type": "dict", + "options": { + "key": {"type": "str"}, + "algorithm": { + "type": "str", + "choices": ["md5", "sha1"], + }, + "encryption": { + "type": "str", + "choices": [ + "3des-cbc", + "aes-128-cbc", + "aes-192-cbc", + "aes-256-cbc", + "null", + ], + }, + "keytype": {"type": "str"}, + "spi": {"type": "int"}, + "passphrase": {"type": "str"}, + }, + }, + "cost": {"type": "int"}, + "afi": { + "required": True, + "type": "str", + "choices": ["ipv4", "ipv6"], + }, + "authentication_v2": { + "type": "dict", + "options": { + "set": {"type": "bool"}, + "message_digest": {"type": "bool"}, + }, + }, + "bfd": {"type": "bool"}, + "authentication_v3": { + "type": "dict", + "options": { + "key": {"type": "str"}, + "spi": {"type": "int"}, + "keytype": {"type": "str"}, + "passphrase": {"type": "str"}, + "algorithm": { + "type": "str", + "choices": ["md5", "sha1"], + }, + }, + }, + "retransmit_interval": {"type": "int"}, + "message_digest_key": { + "type": "dict", + "options": { + "key_id": {"type": "int"}, + "key": {"type": "str"}, + "encryption": {"type": "str"}, + }, + }, + "mtu_ignore": {"type": "bool"}, + "priority": {"type": "int"}, + "area": { + "type": "dict", + "options": { + "area_id": {"required": True, "type": "str"} + }, + }, + "dead_interval": {"type": "int"}, + "shutdown": {"type": "bool"}, + "passive_interface": {"type": "bool"}, + "authentication_key": { + "type": "dict", + "options": { + "encryption": {"type": "str"}, + "key": {"type": "str"}, + }, + }, + "hello_interval": {"type": "int"}, + "transmit_delay": {"type": "int"}, + "network": {"type": "str"}, + }, + }, + }, + }, + } # pylint: disable=C0301 diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/ospfv2/__init__.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/ospfv2/__init__.py new file mode 100644 index 00000000..e69de29b --- /dev/null +++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/ospfv2/__init__.py diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/ospfv2/ospfv2.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/ospfv2/ospfv2.py new file mode 100644 index 00000000..44c0f4ca --- /dev/null +++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/ospfv2/ospfv2.py @@ -0,0 +1,329 @@ +# +# -*- coding: utf-8 -*- +# Copyright 2019 Red Hat +# 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 + +############################################# +# WARNING # +############################################# +# +# This file is auto generated by the resource +# module builder playbook. +# +# Do not edit this file manually. +# +# Changes to this file will be over written +# by the resource module builder. +# +# Changes should be made in the model used to +# generate this file or in the resource module +# builder template. +# +############################################# +""" +The arg spec for the eos_ospfv2 module +""" + + +class Ospfv2Args(object): # pylint: disable=R0903 + """The arg spec for the eos_ospfv2 module + """ + + def __init__(self, **kwargs): + pass + + argument_spec = { + "config": { + "options": { + "processes": { + "elements": "dict", + "options": { + "process_id": {"type": "int"}, + "vrf": {"type": "str"}, + "traffic_engineering": {"type": "bool"}, + "adjacency": { + "options": { + "exchange_start": { + "options": {"threshold": {"type": "int"}}, + "type": "dict", + } + }, + "type": "dict", + }, + "areas": { + "elements": "dict", + "options": { + "default_cost": {"type": "int"}, + "filter": { + "options": { + "address": {"type": "str"}, + "prefix_list": {"type": "str"}, + "subnet_address": {"type": "str"}, + "subnet_mask": {"type": "str"}, + }, + "type": "dict", + }, + "not_so_stubby": { + "options": { + "default_information_originate": { + "options": { + "metric": {"type": "int"}, + "metric_type": {"type": "int"}, + "nssa_only": {"type": "bool"}, + }, + "type": "dict", + }, + "no_summary": {"type": "bool"}, + "nssa_only": {"type": "bool"}, + "lsa": {"type": "bool"}, + "set": {"type": "bool"}, + }, + "type": "dict", + }, + "nssa": { + "options": { + "default_information_originate": { + "options": { + "metric": {"type": "int"}, + "metric_type": {"type": "int"}, + "nssa_only": {"type": "bool"}, + }, + "type": "dict", + }, + "no_summary": {"type": "bool"}, + "nssa_only": {"type": "bool"}, + "set": {"type": "bool"}, + }, + "type": "dict", + }, + "area_id": {"type": "str"}, + "range": { + "options": { + "address": {"type": "str"}, + "advertise": {"type": "bool"}, + "cost": {"type": "int"}, + "subnet_address": {"type": "str"}, + "subnet_mask": {"type": "str"}, + }, + "type": "dict", + }, + "stub": { + "options": { + "no_summary": {"type": "bool"}, + "set": {"type": "bool"}, + }, + "type": "dict", + }, + }, + "type": "list", + }, + "auto_cost": { + "options": { + "reference_bandwidth": {"type": "int"} + }, + "type": "dict", + }, + "bfd": { + "options": {"all_interfaces": {"type": "bool"}}, + "type": "dict", + }, + "default_information": { + "options": { + "always": {"type": "bool"}, + "metric": {"type": "int"}, + "metric_type": {"type": "int"}, + "originate": {"type": "bool"}, + "route_map": {"type": "str"}, + }, + "type": "dict", + }, + "default_metric": {"type": "int"}, + "distance": { + "options": { + "external": {"type": "int"}, + "inter_area": {"type": "int"}, + "intra_area": {"type": "int"}, + }, + "type": "dict", + }, + "distribute_list": { + "options": { + "prefix_list": {"type": "str"}, + "route_map": {"type": "str"}, + }, + "type": "dict", + }, + "dn_bit_ignore": {"type": "bool"}, + "fips_restrictions": {"type": "str"}, + "graceful_restart": { + "options": { + "grace_period": {"type": "int"}, + "set": {"type": "bool"}, + }, + "type": "dict", + }, + "graceful_restart_helper": {"type": "bool"}, + "log_adjacency_changes": { + "options": {"detail": {"type": "bool"}}, + "type": "dict", + }, + "max_lsa": { + "options": { + "count": {"type": "int"}, + "ignore_count": {"type": "int"}, + "ignore_time": {"type": "int"}, + "reset_time": {"type": "int"}, + "threshold": {"type": "int"}, + "warning": {"type": "bool"}, + }, + "type": "dict", + }, + "max_metric": { + "options": { + "router_lsa": { + "options": { + "set": {"type": "bool"}, + "include_stub": {"type": "bool"}, + "on_startup": { + "options": { + "wait_period": {"type": "int"} + }, + "type": "dict", + }, + "summary_lsa": { + "options": { + "max_metric_value": { + "type": "int" + }, + "set": {"type": "bool"}, + }, + "type": "dict", + }, + "external_lsa": { + "options": { + "max_metric_value": { + "type": "int" + }, + "set": {"type": "bool"}, + }, + "type": "dict", + }, + }, + "type": "dict", + } + }, + "type": "dict", + }, + "maximum_paths": {"type": "int"}, + "mpls_ldp": {"type": "bool"}, + "networks": { + "elements": "dict", + "options": { + "area": {"type": "str"}, + "mask": {"type": "str"}, + "network_address": {"type": "str"}, + "prefix": {"type": "str"}, + }, + "type": "list", + }, + "passive_interface": { + "type": "dict", + "options": { + "interface_list": {"type": "str"}, + "default": {"type": "bool"}, + }, + }, + "point_to_point": {"type": "bool"}, + "redistribute": { + "elements": "dict", + "options": { + "isis_level": {"type": "str"}, + "route_map": {"type": "str"}, + "routes": {"type": "str"}, + }, + "type": "list", + }, + "retransmission_threshold": {"type": "int"}, + "rfc1583compatibility": {"type": "bool"}, + "router_id": {"type": "str"}, + "shutdown": {"type": "bool"}, + "summary_address": { + "options": { + "address": {"type": "str"}, + "attribute_map": {"type": "str"}, + "mask": {"type": "str"}, + "not_advertise": {"type": "bool"}, + "prefix": {"type": "str"}, + "tag": {"type": "int"}, + }, + "type": "dict", + }, + "timers": { + "elements": "dict", + "options": { + "lsa": { + "options": { + "rx": { + "options": { + "min_interval": {"type": "int"} + }, + "type": "dict", + }, + "tx": { + "options": { + "delay": { + "options": { + "initial": { + "type": "int" + }, + "max": {"type": "int"}, + "min": {"type": "int"}, + }, + "type": "dict", + } + }, + "type": "dict", + }, + }, + "type": "dict", + }, + "out_delay": {"type": "int"}, + "pacing": {"type": "int"}, + "spf": { + "options": { + "initial": {"type": "int"}, + "max": {"type": "int"}, + "min": {"type": "int"}, + "seconds": {"type": "int"}, + }, + "type": "dict", + }, + }, + "type": "list", + }, + }, + "type": "list", + } + }, + "type": "dict", + }, + "running_config": {"type": "str"}, + "state": { + "choices": [ + "deleted", + "merged", + "overridden", + "replaced", + "gathered", + "rendered", + "parsed", + ], + "default": "merged", + "type": "str", + }, + } # pylint: disable=C0301 diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/ospfv3/__init__.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/ospfv3/__init__.py new file mode 100644 index 00000000..e69de29b --- /dev/null +++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/ospfv3/__init__.py diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/ospfv3/ospfv3.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/ospfv3/ospfv3.py new file mode 100644 index 00000000..f68afded --- /dev/null +++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/ospfv3/ospfv3.py @@ -0,0 +1,476 @@ +# -*- coding: utf-8 -*- +# Copyright 2020 Red Hat +# 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 + +############################################# +# WARNING # +############################################# +# +# This file is auto generated by the resource +# module builder playbook. +# +# Do not edit this file manually. +# +# Changes to this file will be over written +# by the resource module builder. +# +# Changes should be made in the model used to +# generate this file or in the resource module +# builder template. +# +############################################# + +""" +The arg spec for the eos_ospfv3 module +""" + + +class Ospfv3Args(object): # pylint: disable=R0903 + """The arg spec for the eos_ospfv3 module + """ + + def __init__(self, **kwargs): + pass + + argument_spec = { + "running_config": {"type": "str"}, + "state": { + "default": "merged", + "type": "str", + "choices": [ + "deleted", + "merged", + "overridden", + "replaced", + "gathered", + "rendered", + "parsed", + ], + }, + "config": { + "type": "dict", + "options": { + "processes": { + "elements": "dict", + "type": "list", + "options": { + "router_id": {"type": "str"}, + "shutdown": {"type": "bool"}, + "fips_restrictions": {"type": "bool"}, + "graceful_restart_helper": {"type": "bool"}, + "adjacency": { + "type": "dict", + "options": { + "exchange_start": { + "type": "dict", + "options": {"threshold": {"type": "int"}}, + } + }, + }, + "max_metric": { + "type": "dict", + "options": { + "router_lsa": { + "type": "dict", + "options": { + "external_lsa": { + "type": "dict", + "options": { + "set": {"type": "bool"}, + "max_metric_value": { + "type": "int" + }, + }, + }, + "summary_lsa": { + "type": "dict", + "options": { + "set": {"type": "bool"}, + "max_metric_value": { + "type": "int" + }, + }, + }, + "set": {"type": "bool"}, + "on_startup": { + "type": "dict", + "options": { + "wait_for_bgp": { + "type": "bool" + }, + "wait_period": {"type": "int"}, + }, + }, + "include_stub": {"type": "bool"}, + }, + } + }, + }, + "log_adjacency_changes": { + "type": "dict", + "options": { + "set": {"type": "bool"}, + "detail": {"type": "bool"}, + }, + }, + "graceful_restart": { + "type": "dict", + "options": { + "grace_period": {"type": "int"}, + "set": {"type": "bool"}, + }, + }, + "timers": { + "type": "dict", + "options": { + "throttle": { + "type": "dict", + "options": { + "max": {"type": "int"}, + "initial": {"type": "int"}, + "min": {"type": "int"}, + "spf": {"type": "bool"}, + "lsa": {"type": "bool"}, + }, + }, + "out_delay": {"type": "int"}, + "pacing": {"type": "int"}, + "lsa": {"type": "int"}, + }, + }, + "vrf": {"type": "str"}, + "auto_cost": { + "type": "dict", + "options": { + "reference_bandwidth": {"type": "int"} + }, + }, + "passive_interface": {"type": "bool"}, + "bfd": { + "type": "dict", + "options": {"all_interfaces": {"type": "bool"}}, + }, + "areas": { + "elements": "dict", + "type": "list", + "options": { + "area_id": {"type": "str"}, + "encryption": { + "type": "dict", + "options": { + "hidden_key": {"type": "bool"}, + "key": {"type": "str"}, + "algorithm": { + "type": "str", + "choices": ["sha1", "md5"], + }, + "encrypt_key": {"type": "bool"}, + "encryption": { + "type": "str", + "choices": [ + "3des-cbc", + "aes-128-cbc", + "aes-192-cbc", + "aes-256-cbc", + "null", + ], + }, + "spi": {"type": "int"}, + "passphrase": {"type": "str"}, + }, + }, + "nssa": { + "type": "dict", + "options": { + "translate": {"type": "bool"}, + "default_information_originate": { + "type": "dict", + "options": { + "metric_type": {"type": "int"}, + "metric": {"type": "int"}, + "nssa_only": {"type": "bool"}, + "set": {"type": "bool"}, + }, + }, + "nssa_only": {"type": "bool"}, + "set": {"type": "bool"}, + "no_summary": {"type": "bool"}, + }, + }, + "stub": { + "type": "dict", + "options": { + "summary_lsa": {"type": "bool"}, + "set": {"type": "bool"}, + }, + }, + "default_cost": {"type": "int"}, + "authentication": { + "type": "dict", + "options": { + "hidden_key": {"type": "bool"}, + "key": {"type": "str"}, + "algorithm": { + "type": "str", + "choices": ["md5", "sha1"], + }, + "encrypt_key": {"type": "bool"}, + "spi": {"type": "int"}, + "passphrase": {"type": "str"}, + }, + }, + }, + }, + "address_family": { + "elements": "dict", + "type": "list", + "options": { + "router_id": {"type": "str"}, + "distance": {"type": "int"}, + "redistribute": { + "elements": "dict", + "type": "list", + "options": { + "routes": { + "type": "str", + "choices": [ + "bgp", + "connected", + "static", + ], + }, + "route_map": {"type": "str"}, + }, + }, + "default_information": { + "type": "dict", + "options": { + "metric_type": {"type": "int"}, + "always": {"type": "bool"}, + "metric": {"type": "int"}, + "originate": {"type": "bool"}, + "route_map": {"type": "str"}, + }, + }, + "afi": { + "choices": ["ipv4", "ipv6"], + "type": "str", + }, + "fips_restrictions": {"type": "bool"}, + "default_metric": {"type": "int"}, + "maximum_paths": {"type": "int"}, + "adjacency": { + "type": "dict", + "options": { + "exchange_start": { + "type": "dict", + "options": { + "threshold": {"type": "int"} + }, + } + }, + }, + "max_metric": { + "type": "dict", + "options": { + "router_lsa": { + "type": "dict", + "options": { + "external_lsa": { + "type": "dict", + "options": { + "set": { + "type": "bool" + }, + "max_metric_value": { + "type": "int" + }, + }, + }, + "summary_lsa": { + "type": "dict", + "options": { + "set": { + "type": "bool" + }, + "max_metric_value": { + "type": "int" + }, + }, + }, + "set": {"type": "bool"}, + "on_startup": { + "type": "dict", + "options": { + "wait_for_bgp": { + "type": "bool" + }, + "wait_period": { + "type": "int" + }, + }, + }, + "include_stub": { + "type": "bool" + }, + }, + } + }, + }, + "log_adjacency_changes": { + "type": "dict", + "options": { + "set": {"type": "bool"}, + "detail": {"type": "bool"}, + }, + }, + "timers": { + "type": "dict", + "options": { + "throttle": { + "type": "dict", + "options": { + "max": {"type": "int"}, + "initial": {"type": "int"}, + "min": {"type": "int"}, + "spf": {"type": "bool"}, + "lsa": {"type": "bool"}, + }, + }, + "out_delay": {"type": "int"}, + "pacing": {"type": "int"}, + "lsa": {"type": "int"}, + }, + }, + "shutdown": {"type": "bool"}, + "auto_cost": { + "type": "dict", + "options": { + "reference_bandwidth": {"type": "int"} + }, + }, + "graceful_restart_helper": {"type": "bool"}, + "passive_interface": {"type": "bool"}, + "bfd": { + "type": "dict", + "options": { + "all_interfaces": {"type": "bool"} + }, + }, + "areas": { + "elements": "dict", + "type": "list", + "options": { + "ranges": { + "elements": "dict", + "type": "list", + "options": { + "subnet_mask": {"type": "str"}, + "advertise": {"type": "bool"}, + "cost": {"type": "int"}, + "subnet_address": { + "type": "str" + }, + "address": {"type": "str"}, + }, + }, + "area_id": {"type": "str"}, + "encryption": { + "type": "dict", + "options": { + "hidden_key": {"type": "bool"}, + "key": {"type": "str"}, + "algorithm": { + "type": "str", + "choices": ["sha1", "md5"], + }, + "encrypt_key": { + "type": "bool" + }, + "encryption": { + "type": "str", + "choices": [ + "3des-cbc", + "aes-128-cbc", + "aes-192-cbc", + "aes-256-cbc", + "null", + ], + }, + "spi": {"type": "int"}, + "passphrase": {"type": "str"}, + }, + }, + "nssa": { + "type": "dict", + "options": { + "translate": {"type": "bool"}, + "default_information_originate": { + "type": "dict", + "options": { + "metric_type": { + "type": "int" + }, + "metric": { + "type": "int" + }, + "nssa_only": { + "type": "bool" + }, + "set": { + "type": "bool" + }, + }, + }, + "nssa_only": {"type": "bool"}, + "set": {"type": "bool"}, + "no_summary": {"type": "bool"}, + }, + }, + "stub": { + "type": "dict", + "options": { + "summary_lsa": { + "type": "bool" + }, + "set": {"type": "bool"}, + }, + }, + "default_cost": {"type": "int"}, + "authentication": { + "type": "dict", + "options": { + "hidden_key": {"type": "bool"}, + "key": {"type": "str"}, + "algorithm": { + "type": "str", + "choices": ["md5", "sha1"], + }, + "encrypt_key": { + "type": "bool" + }, + "spi": {"type": "int"}, + "passphrase": {"type": "str"}, + }, + }, + }, + }, + "graceful_restart": { + "type": "dict", + "options": { + "grace_period": {"type": "int"}, + "set": {"type": "bool"}, + }, + }, + }, + }, + }, + } + }, + }, + } # pylint: disable=C0301 diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/static_routes/__init__.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/static_routes/__init__.py new file mode 100644 index 00000000..e69de29b --- /dev/null +++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/static_routes/__init__.py diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/static_routes/static_routes.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/static_routes/static_routes.py new file mode 100644 index 00000000..d28f8d7c --- /dev/null +++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/static_routes/static_routes.py @@ -0,0 +1,97 @@ +# +# -*- coding: utf-8 -*- +# Copyright 2019 Red Hat +# GNU General Public License v3.0+ +# (see COPYING or https://www.gnu.org/licenses/gpl-3.0.txt) + +############################################# +# WARNING # +############################################# +# +# This file is auto generated by the resource +# module builder playbook. +# +# Do not edit this file manually. +# +# Changes to this file will be over written +# by the resource module builder. +# +# Changes should be made in the model used to +# generate this file or in the resource module +# builder template. +# +############################################# +""" +The arg spec for the eos_static_routes module +""" + +from __future__ import absolute_import, division, print_function + +__metaclass__ = type + + +class Static_routesArgs(object): + """The arg spec for the eos_static_routes module + """ + + def __init__(self, **kwargs): + pass + + argument_spec = { + "config": { + "elements": "dict", + "options": { + "address_families": { + "elements": "dict", + "options": { + "afi": { + "choices": ["ipv4", "ipv6"], + "required": True, + "type": "str", + }, + "routes": { + "elements": "dict", + "options": { + "dest": {"required": True, "type": "str"}, + "next_hops": { + "elements": "dict", + "options": { + "admin_distance": {"type": "int"}, + "description": {"type": "str"}, + "forward_router_address": { + "type": "str" + }, + "interface": {"type": "str"}, + "nexthop_grp": {"type": "str"}, + "mpls_label": {"type": "int"}, + "tag": {"type": "int"}, + "track": {"type": "str"}, + "vrf": {"type": "str"}, + }, + "type": "list", + }, + }, + "type": "list", + }, + }, + "type": "list", + }, + "vrf": {"type": "str"}, + }, + "type": "list", + }, + "running_config": {"type": "str"}, + "state": { + "choices": [ + "deleted", + "merged", + "overridden", + "replaced", + "gathered", + "rendered", + "parsed", + ], + "default": "merged", + "type": "str", + }, + } # pylint: disable=C0301 diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/vlans/__init__.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/vlans/__init__.py new file mode 100644 index 00000000..e69de29b --- /dev/null +++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/vlans/__init__.py diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/vlans/vlans.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/vlans/vlans.py new file mode 100644 index 00000000..c4622093 --- /dev/null +++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/argspec/vlans/vlans.py @@ -0,0 +1,64 @@ +# -*- coding: utf-8 -*- +# Copyright 2019 Red Hat +# GNU General Public License v3.0+ +# (see COPYING or https://www.gnu.org/licenses/gpl-3.0.txt) + +############################################# +# WARNING # +############################################# +# +# This file is auto generated by the resource +# module builder playbook. +# +# Do not edit this file manually. +# +# Changes to this file will be over written +# by the resource module builder. +# +# Changes should be made in the model used to +# generate this file or in the resource module +# builder template. +# +############################################# + +""" +The arg spec for the eos_vlans module +""" + +from __future__ import absolute_import, division, print_function + +__metaclass__ = type + + +class VlansArgs(object): + """The arg spec for the eos_vlans module + """ + + def __init__(self, **kwargs): + pass + + argument_spec = { + "config": { + "elements": "dict", + "options": { + "vlan_id": {"required": True, "type": "int"}, + "name": {"type": "str"}, + "state": {"choices": ["active", "suspend"], "type": "str"}, + }, + "type": "list", + }, + "running_config": {"type": "str"}, + "state": { + "choices": [ + "merged", + "replaced", + "overridden", + "deleted", + "gathered", + "rendered", + "parsed", + ], + "default": "merged", + "type": "str", + }, + } diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/config/__init__.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/config/__init__.py new file mode 100644 index 00000000..e69de29b --- /dev/null +++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/config/__init__.py diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/config/acl_interfaces/__init__.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/config/acl_interfaces/__init__.py new file mode 100644 index 00000000..e69de29b --- /dev/null +++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/config/acl_interfaces/__init__.py diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/config/acl_interfaces/acl_interfaces.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/config/acl_interfaces/acl_interfaces.py new file mode 100644 index 00000000..0f9db622 --- /dev/null +++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/config/acl_interfaces/acl_interfaces.py @@ -0,0 +1,475 @@ +# +# -*- coding: utf-8 -*- +# Copyright 2019 Red Hat +# GNU General Public License v3.0+ +# (see COPYING or https://www.gnu.org/licenses/gpl-3.0.txt) +""" +The eos_acl_interfaces class +It is in this file where the current configuration (as dict) +is compared to the provided configuration (as dict) and the command set +necessary to bring the current configuration to it's desired end-state is +created +""" + +from __future__ import absolute_import, division, print_function + +__metaclass__ = type + +import itertools + +from ansible_collections.ansible.netcommon.plugins.module_utils.network.common.cfg.base import ( + ConfigBase, +) +from ansible_collections.ansible.netcommon.plugins.module_utils.network.common.utils import ( + to_list, + search_obj_in_list, +) +from ansible_collections.arista.eos.plugins.module_utils.network.eos.facts.facts import ( + Facts, +) + + +class Acl_interfaces(ConfigBase): + """ + The eos_acl_interfaces class + """ + + gather_subset = ["!all", "!min"] + + gather_network_resources = ["acl_interfaces"] + + def __init__(self, module): + super(Acl_interfaces, self).__init__(module) + + def get_acl_interfaces_facts(self, data=None): + """ Get the 'facts' (the current configuration) + + :rtype: A dictionary + :returns: The current configuration as a dictionary + """ + facts, _warnings = Facts(self._module).get_facts( + self.gather_subset, self.gather_network_resources, data=data + ) + acl_interfaces_facts = facts["ansible_network_resources"].get( + "acl_interfaces" + ) + if not acl_interfaces_facts: + return [] + return acl_interfaces_facts + + def execute_module(self): + """ Execute the module + + :rtype: A dictionary + :returns: The result from module execution + """ + result = {"changed": False} + warnings = list() + commands = list() + changed = False + + if self.state in self.ACTION_STATES: + existing_acl_interfaces_facts = self.get_acl_interfaces_facts() + else: + existing_acl_interfaces_facts = [] + if self.state in self.ACTION_STATES or self.state == "rendered": + commands.extend(self.set_config(existing_acl_interfaces_facts)) + if commands and self.state in self.ACTION_STATES: + if not self._module.check_mode: + self._connection.edit_config(commands) + changed = True + if changed: + result["changed"] = True + if self.state in self.ACTION_STATES: + result["commands"] = commands + if self.state in self.ACTION_STATES or self.state == "gathered": + changed_acl_interfaces_facts = self.get_acl_interfaces_facts() + elif self.state == "rendered": + result["rendered"] = commands + elif self.state == "parsed": + if not self._module.params["running_config"]: + self._module.fail_json( + msg="Value of running_config parameter must not be empty for state parsed" + ) + result["parsed"] = self.get_acl_interfaces_facts( + data=self._module.params["running_config"] + ) + else: + changed_acl_interfaces_facts = [] + if self.state in self.ACTION_STATES: + result["before"] = existing_acl_interfaces_facts + if result["changed"]: + result["after"] = changed_acl_interfaces_facts + elif self.state == "gathered": + result["gathered"] = changed_acl_interfaces_facts + + result["warnings"] = warnings + return result + + def set_config(self, existing_acl_interfaces_facts): + """ Collect the configuration from the args passed to the module, + collect the current configuration (as a dict from facts) + + :rtype: A list + :returns: the commands necessary to migrate the current configuration + to the desired configuration + """ + want = self._module.params["config"] + have = existing_acl_interfaces_facts + resp = self.set_state(want, have) + return to_list(resp) + + def set_state(self, want, have): + """ Select the appropriate function based on the state provided + + :param want: the desired configuration as a dictionary + :param have: the current configuration as a dictionary + :rtype: A list + :returns: the commands necessary to migrate the current configuration + to the desired configuration + """ + commands = [] + if self.state in ("merged", "replaced", "overridden") and not want: + self._module.fail_json( + msg="value of config parameter must not be empty for state {0}".format( + self.state + ) + ) + state = self._module.params["state"] + if state == "overridden": + commands = self._state_overridden(want, have) + elif state == "deleted": + commands = self._state_deleted(want, have) + elif state == "merged" or self.state == "rendered": + commands = self._state_merged(want, have) + elif state == "replaced": + commands = self._state_replaced(want, have) + return commands + + def _state_replaced(self, want, have): + """ The command generator when state is replaced + + :rtype: A list + :returns: the commands necessary to migrate the current configuration + to the desired configuration + """ + commandset = [] + want_interface = [] + for w in want: + commands = [] + diff_access_group = [] + want_interface.append(w["name"]) + obj_in_have = search_obj_in_list(w["name"], have, "name") + if not obj_in_have or "access_groups" not in obj_in_have.keys(): + commands.append(add_commands(w["access_groups"], w["name"])) + else: + if "access_groups" in obj_in_have.keys(): + obj = self.get_acl_diff(obj_in_have, w) + if obj[0]: + to_delete = { + "access_groups": [{"acls": obj[0], "afi": "ipv4"}] + } + commands.append(remove_commands(to_delete, w["name"])) + if obj[1]: + to_delete = { + "access_groups": [{"acls": obj[1], "afi": "ipv6"}] + } + commands.append(remove_commands(to_delete, w["name"])) + diff = self.get_acl_diff(w, obj_in_have) + if diff[0]: + diff_access_group.append( + {"afi": "ipv4", "acls": diff[0]} + ) + if diff[1]: + diff_access_group.append( + {"afi": "ipv6", "acls": diff[1]} + ) + if diff_access_group: + commands.append( + add_commands(diff_access_group, w["name"]) + ) + if commands: + intf_command = ["interface " + w["name"]] + commands = list(itertools.chain(*commands)) + commandset.append(intf_command) + commandset.append(commands) + + if commandset: + commandset = list(itertools.chain(*commandset)) + return commandset + + def _state_overridden(self, want, have): + """ The command generator when state is overridden + + :rtype: A list + :returns: the commands necessary to migrate the current configuration + to the desired configuration + """ + commandset = [] + want_interface = [] + for w in want: + commands = [] + diff_access_group = [] + want_interface.append(w["name"]) + obj_in_have = search_obj_in_list(w["name"], have, "name") + if not obj_in_have or "access_groups" not in obj_in_have.keys(): + commands.append(add_commands(w["access_groups"], w["name"])) + else: + if "access_groups" in obj_in_have.keys(): + obj = self.get_acl_diff(obj_in_have, w) + if obj[0]: + to_delete = { + "access_groups": [{"acls": obj[0], "afi": "ipv4"}] + } + commands.append(remove_commands(to_delete, w["name"])) + if obj[1]: + to_delete = { + "access_groups": [{"acls": obj[1], "afi": "ipv6"}] + } + commands.append(remove_commands(to_delete, w["name"])) + diff = self.get_acl_diff(w, obj_in_have) + if diff[0]: + diff_access_group.append( + {"afi": "ipv4", "acls": diff[0]} + ) + if diff[1]: + diff_access_group.append( + {"afi": "ipv6", "acls": diff[1]} + ) + if diff_access_group: + commands.append( + add_commands(diff_access_group, w["name"]) + ) + if commands: + intf_command = ["interface " + w["name"]] + commands = list(itertools.chain(*commands)) + commandset.append(intf_command) + commandset.append(commands) + for h in have: + commands = [] + if "access_groups" in h.keys() and h["access_groups"]: + if h["name"] not in want_interface: + for h_group in h["access_groups"]: + to_delete = {"access_groups": [h_group]} + commands.append(remove_commands(to_delete, h["name"])) + if commands: + intf_command = ["interface " + h["name"]] + commands = list(itertools.chain(*commands)) + commandset.append(intf_command) + commandset.append(commands) + + if commandset: + commandset = list(itertools.chain(*commandset)) + + return commandset + + def _state_merged(self, want, have): + """ The command generator when state is merged + + :rtype: A list + :returns: the commands necessary to merge the provided into + the current configuration + """ + commandset = [] + for w in want: + commands = [] + diff_access_group = [] + obj_in_have = search_obj_in_list(w["name"], have, "name") + if not obj_in_have: + commands = add_commands(w["access_groups"], w["name"]) + else: + if "access_groups" in obj_in_have.keys(): + diff = self.get_acl_diff(w, obj_in_have) + if diff[0]: + diff_access_group.append( + {"afi": "ipv4", "acls": diff[0]} + ) + if diff[1]: + diff_access_group.append( + {"afi": "ipv6", "acls": diff[1]} + ) + if diff_access_group: + commands = add_commands(diff_access_group, w["name"]) + else: + commands = add_commands(w["access_groups"], w["name"]) + if commands: + intf_command = ["interface " + w["name"]] + commandset.append(intf_command) + commandset.append(commands) + if commandset: + commandset = list(itertools.chain(*commandset)) + return commandset + + def _state_deleted(self, want, have): + """ The command generator when state is deleted + + :rtype: A list + :returns: the commands necessary to remove the current configuration + of the provided objects + """ + commandset = [] + for w in want: + commands = [] + intf_command = ["interface " + w["name"]] + obj_in_have = search_obj_in_list(w["name"], have, "name") + if "access_groups" not in w.keys() or not w["access_groups"]: + commands = remove_commands(obj_in_have, w["name"]) + if w["access_groups"]: + for w_grp in w["access_groups"]: + if "acls" not in w_grp.keys() or not w_grp["acls"]: + obj = self.get_acls_from_afi( + w["name"], w_grp["afi"], have + ) + to_delete = { + "access_groups": [ + {"acls": obj, "afi": w_grp["afi"]} + ] + } + commands = remove_commands(to_delete, w["name"]) + else: + if ( + "access_groups" not in obj_in_have.keys() + or not obj_in_have["access_groups"] + ): + continue + group = {"access_groups": [w_grp]} + obj = self.get_acl_diff(group, obj_in_have, True) + if obj[0]: + to_delete = { + "access_groups": [ + {"acls": obj[0], "afi": "ipv4"} + ] + } + commands.append( + remove_commands(to_delete, w["name"]) + ) + if obj[1]: + to_delete = { + "access_groups": [ + {"acls": obj[1], "afi": "ipv6"} + ] + } + commands.append( + remove_commands(to_delete, w["name"]) + ) + if commands: + commands = list(itertools.chain(*commands)) + if commands: + commandset.append(intf_command) + commandset.append(commands) + + if commandset: + commandset = list(itertools.chain(*commandset)) + return commandset + + def get_acl_diff(self, w, h, intersection=False): + diff_v4 = [] + diff_v6 = [] + w_acls_v4 = [] + w_acls_v6 = [] + h_acls_v4 = [] + h_acls_v6 = [] + for w_group in w["access_groups"]: + if w_group["afi"] == "ipv4": + w_acls_v4 = w_group["acls"] + if w_group["afi"] == "ipv6": + w_acls_v6 = w_group["acls"] + for h_group in h["access_groups"]: + if h_group["afi"] == "ipv4": + h_acls_v4 = h_group["acls"] + if h_group["afi"] == "ipv6": + h_acls_v6 = h_group["acls"] + for item in w_acls_v4: + match = list( + filter(lambda x: x["name"] == item["name"], h_acls_v4) + ) + if match: + if item["direction"] == match[0]["direction"]: + if intersection: + diff_v4.append(item) + else: + if not intersection: + diff_v4.append(item) + else: + if not intersection: + diff_v4.append(item) + for item in w_acls_v6: + match = list( + filter(lambda x: x["name"] == item["name"], h_acls_v6) + ) + if match: + if item["direction"] == match[0]["direction"]: + if intersection: + diff_v6.append(item) + else: + if not intersection: + diff_v6.append(item) + else: + if not intersection: + diff_v6.append(item) + return diff_v4, diff_v6 + + def get_acls_from_afi(self, interface, afi, have): + config = [] + for h in have: + if h["name"] == interface: + if "access_groups" not in h.keys() or not h["access_groups"]: + continue + if h["access_groups"]: + for h_grp in h["access_groups"]: + if h_grp["afi"] == afi: + config = h_grp["acls"] + return config + + +def add_commands(want, interface): + commands = [] + + for w in want: + # This module was verified on an ios device since vEOS doesnot support + # acl_interfaces cnfiguration. In ios, ipv6 acl is configured as + # traffic-filter and in eos it is access-group + + # a_cmd = "traffic-filter" if w['afi'] == 'ipv6' else "access-group" + a_cmd = "access-group" + afi = "ip" if w["afi"] == "ipv4" else w["afi"] + if "acls" in w.keys(): + for acl in w["acls"]: + commands.append( + afi + + " " + + a_cmd + + " " + + acl["name"] + + " " + + acl["direction"] + ) + return commands + + +def remove_commands(want, interface): + commands = [] + if "access_groups" not in want.keys() or not want["access_groups"]: + return commands + for w in want["access_groups"]: + # This module was verified on an ios device since vEOS doesnot support + # acl_interfaces cnfiguration. In ios, ipv6 acl is configured as + # traffic-filter and in eos it is access-group + + # a_cmd = "traffic-filter" if w['afi'] == 'ipv6' else "access-group" + a_cmd = "access-group" + + afi = "ip" if w["afi"] == "ipv4" else w["afi"] + if "acls" in w.keys(): + for acl in w["acls"]: + commands.append( + "no " + + afi + + " " + + a_cmd + + " " + + acl["name"] + + " " + + acl["direction"] + ) + return commands diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/config/acls/__init__.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/config/acls/__init__.py new file mode 100644 index 00000000..e69de29b --- /dev/null +++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/config/acls/__init__.py diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/config/acls/acls.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/config/acls/acls.py new file mode 100644 index 00000000..3d1d8216 --- /dev/null +++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/config/acls/acls.py @@ -0,0 +1,605 @@ +# +# -*- coding: utf-8 -*- +# Copyright 2019 Red Hat +# GNU General Public License v3.0+ +# (see COPYING or https://www.gnu.org/licenses/gpl-3.0.txt) +""" +The eos_acls class +It is in this file where the current configuration (as dict) +is compared to the provided configuration (as dict) and the command set +necessary to bring the current configuration to it's desired end-state is +created +""" + +from __future__ import absolute_import, division, print_function + +__metaclass__ = type + +import socket +import re +import itertools + +from ansible_collections.ansible.netcommon.plugins.module_utils.network.common.cfg.base import ( + ConfigBase, +) +from ansible_collections.ansible.netcommon.plugins.module_utils.network.common.utils import ( + to_list, +) +from ansible_collections.ansible.netcommon.plugins.module_utils.network.common.utils import ( + remove_empties, + dict_diff, +) +from ansible_collections.arista.eos.plugins.module_utils.network.eos.facts.facts import ( + Facts, +) + + +class Acls(ConfigBase): + """ + The eos_acls class + """ + + gather_subset = ["!all", "!min"] + + gather_network_resources = ["acls"] + + def __init__(self, module): + super(Acls, self).__init__(module) + + def get_acls_facts(self, data=None): + """ Get the 'facts' (the current configuration) + + :rtype: A dictionary + :returns: The current configuration as a dictionary + """ + facts, _warnings = Facts(self._module).get_facts( + self.gather_subset, self.gather_network_resources, data=data + ) + acls_facts = facts["ansible_network_resources"].get("acls") + if not acls_facts: + return [] + return acls_facts + + def execute_module(self): + """ Execute the module + + :rtype: A dictionary + :returns: The result from module execution + """ + result = {"changed": False} + warnings = list() + commands = list() + changed = False + + if self.state in self.ACTION_STATES: + existing_acls_facts = self.get_acls_facts() + else: + existing_acls_facts = [] + if self.state in self.ACTION_STATES or self.state == "rendered": + commands.extend(self.set_config(existing_acls_facts)) + if commands and self.state in self.ACTION_STATES: + if not self._module.check_mode: + self._connection.edit_config(commands) + changed = True + if changed: + result["changed"] = True + if self.state in self.ACTION_STATES: + result["commands"] = commands + if self.state in self.ACTION_STATES or self.state == "gathered": + changed_acls_facts = self.get_acls_facts() + elif self.state == "rendered": + commands = list(itertools.chain(*commands)) + result["rendered"] = commands + elif self.state == "parsed": + if not self._module.params["running_config"]: + self._module.fail_json( + msg="Value of running_config parameter must not be empty for state parsed" + ) + result["parsed"] = self.get_acls_facts( + data=self._module.params["running_config"] + ) + else: + changed_acls_facts = [] + if self.state in self.ACTION_STATES: + result["before"] = existing_acls_facts + if result["changed"]: + result["after"] = changed_acls_facts + elif self.state == "gathered": + result["gathered"] = changed_acls_facts + + result["warnings"] = warnings + return result + + def set_config(self, existing_acls_facts): + """ Collect the configuration from the args passed to the module, + collect the current configuration (as a dict from facts) + + :rtype: A list + :returns: the commands necessary to migrate the current configuration + to the desired configuration + """ + config = self._module.params.get("config") + want = [] + onbox_configs = [] + for h in existing_acls_facts: + have_configs = add_commands(remove_empties(h)) + onbox_configs.append(have_configs) + if config: + for w in config: + want.append(remove_empties(w)) + have = existing_acls_facts + resp = self.set_state(want, have) + if self.state == "merged": + to_config = self.compare_configs(onbox_configs, to_list(resp)) + else: + to_config = resp + return to_config + + def compare_configs(self, have, want): + commands = [] + want = list(itertools.chain(*want)) + have = list(itertools.chain(*have)) + h_index = 0 + config = list(want) + for w in want: + access_list = re.findall(r"(ip.*) access-list (.*)", w) + if access_list: + if w in have: + h_index = have.index(w) + else: + for num, h in enumerate(have, start=h_index + 1): + if "access-list" not in h: + seq_num = re.search(r"(\d+) (.*)", w) + if seq_num: + have_seq_num = re.search(r"(\d+) (.*)", h) + if seq_num.group(1) == have_seq_num.group( + 1 + ) and have_seq_num.group(2) != seq_num.group(2): + negate_cmd = "no " + seq_num.group(1) + config.insert(config.index(w), negate_cmd) + if w in h: + config.pop(config.index(w)) + break + for c in config: + access_list = re.findall(r"(ip.*) access-list (.*)", c) + if access_list: + acl_index = config.index(c) + else: + if config[acl_index] not in commands: + commands.append(config[acl_index]) + commands.append(c) + return commands + + def set_state(self, want, have): + """ Select the appropriate function based on the state provided + + :param want: the desired configuration as a dictionary + :param have: the current configuration as a dictionary + :rtype: A list + :returns: the commands necessary to migrate the current configuration + to the desired configuration + """ + commands = [] + if ( + self.state in ("merged", "replaced", "overridden", "rendered") + and not want + ): + self._module.fail_json( + msg="value of config parameter must not be empty for state {0}".format( + self.state + ) + ) + state = self._module.params["state"] + if state == "overridden": + commands = self._state_overridden(want, have) + elif state == "deleted": + commands = self._state_deleted(want, have) + elif state == "merged" or self.state == "rendered": + commands = self._state_merged(want, have) + elif state == "replaced": + commands = self._state_replaced(want, have) + return commands + + def _state_replaced(self, want, have): + """ The command generator when state is replaced + + :rtype: A list + :returns: the commands necessary to migrate the current configuration + to the desired configuration + """ + commands = [] + config_cmds = [] + remove_cmds = [] + diff = {} + for w in want: + afi = "ipv6" if w["afi"] == "ipv6" else "ipv4" + for acl in w["acls"]: + name = acl["name"] + want_ace = acl["aces"] + for h in have: + if h["afi"] == afi: + for h_acl in h["acls"]: + if h_acl["name"] == name: + h = {"afi": afi, "acls": [{"name": name}]} + for h_ace in h_acl["aces"]: + diff = get_ace_diff(h_ace, want_ace) + if diff: + h = { + "afi": afi, + "acls": [{"name": name, "aces": [h_ace]}], + } + remove_cmds.append(del_commands(h, have)) + for w_ace in want_ace: + w_diff = get_ace_diff(w_ace, h_acl["aces"]) + if w_diff: + w = [ + { + "afi": afi, + "acls": [ + {"name": name, "aces": [w_ace]} + ], + } + ] + config_cmds = set_commands(w, have) + config_cmds = list( + itertools.chain(*config_cmds) + ) + + if remove_cmds: + remove_cmds = list(itertools.chain(*remove_cmds)) + commands.append(remove_cmds) + if config_cmds: + commands.append(config_cmds) + commands = list(itertools.chain(*commands)) + commandset = [] + [commandset.append(cmd) for cmd in commands if cmd not in commandset] + return commandset + + def _state_overridden(self, want, have): + """ The command generator when state is overridden + + :rtype: A list + :returns: the commands necessary to migrate the current configuration + to the desired configuration + """ + commands = [] + ace_diff = {} + h_afi_list = [] + w_afi_list = [] + for h in have: + h_afi_list.append(h["afi"]) + for w in want: + w_afi_list.append(w["afi"]) + for hafi in h_afi_list: + if hafi not in w_afi_list: + h = {"afi": hafi} + remove_cmds = del_commands(h, have) + commands.append(remove_cmds) + for w in want: + w_names = [] + for h in have: + h_names = [] + if w["afi"] == h["afi"]: + for w_acl in w["acls"]: + w_names.append(w_acl["name"]) + for h_acl in h["acls"]: + h_names.append(h_acl["name"]) + if h_acl["name"] == w_acl["name"]: + for h_ace in h_acl["aces"]: + ace_diff = get_ace_diff( + h_ace, w_acl["aces"] + ) + if ace_diff: + h = { + "afi": h["afi"], + "acls": [ + { + "name": h_acl["name"], + "aces": [h_ace], + } + ], + } + remove_cmds = del_commands(h, have) + commands.append(remove_cmds) + for w_ace in w_acl["aces"]: + w_ace_diff = get_ace_diff( + w_ace, h_acl["aces"] + ) + if w_ace_diff: + w_diff = [ + { + "afi": w["afi"], + "acls": [ + { + "name": w_acl["name"], + "aces": [w_ace], + } + ], + } + ] + config_cmds = set_commands( + w_diff, have + ) + config_cmds = list( + itertools.chain(*config_cmds) + ) + commands.append(config_cmds) + for hname in h_names: + if hname not in w_names: + h = {"afi": h["afi"], "acls": [{"name": hname}]} + remove_cmds = del_commands(h, have) + if remove_cmds not in commands: + commands.append(remove_cmds) + + if commands: + commands = list(itertools.chain(*commands)) + commandset = [] + for c in commands: + access_list = re.findall(r"(ip.*) access-list (.*)", c) + if access_list and "no" in c: + commandset.append(c) + continue + if access_list: + acl_index = commands.index(c) + else: + if commands[acl_index] not in commandset: + commandset.append(commands[acl_index]) + commandset.append(c) + + return commandset + + def _state_merged(self, want, have): + """ The command generator when state is merged + + :rtype: A list + :returns: the commands necessary to merge the provided into + the current configuration + """ + return set_commands(want, have) + + def _state_deleted(self, want, have): + """ The command generator when state is deleted + + :rtype: A list + :returns: the commands necessary to remove the current configuration + of the provided objects + """ + commands = [] + if not want: + for h in have: + return_command = add_commands(h) + for command in return_command: + command = "no " + command + commands.append(command) + else: + for w in want: + return_command = del_commands(w, have, True) + if return_command[:5] == "Warn ": + self._module.warn(return_command[5:]) + return commands + else: + commands.append(return_command) + commands = list(itertools.chain(*commands)) + return commands + + +def set_commands(want, have): + commands = [] + for w in want: + wace_updated = [] + for h in have: + if w["afi"] == h["afi"]: + for wacl in w["acls"]: + for hacl in h["acls"]: + if wacl["name"] == hacl["name"]: + want_aces = wacl["aces"] + for wace in wacl["aces"]: + for hace in hacl["aces"]: + if ( + "sequence" in wace.keys() + and "sequence" in hace.keys() + ): + if ( + wace["sequence"] + == hace["sequence"] + ): + wace_updated = get_updated_ace( + wace, hace + ) + if wace_updated: + want_aces.pop( + want_aces.index(wace) + ) + want_aces.append(wace_updated) + return_command = add_commands(w) + commands.append(return_command) + return commands + + +def get_updated_ace(w, h): + # gives the ace to be updated in case of merge update. + if not dict_diff(w, h): + return + w_updated = w.copy() + for hkey in h.keys(): + if hkey not in w.keys(): + w_updated.update({hkey: h[hkey]}) + else: + w_updated.update({hkey: w[hkey]}) + return w_updated + + +def add_commands(want): + commandset = [] + protocol_name = { + "51": "ahp", + "47": "gre", + "1": "icmp", + "2": "igmp", + "4": "ip", + "89": "ospf", + "103": "pim", + "6": "tcp", + "17": "udp", + "112": "vrrp", + } + if not want: + return commandset + command = "" + afi = "ip" if want["afi"] == "ipv4" else "ipv6" + for acl in want["acls"]: + if "standard" in acl.keys() and acl["standard"]: + command = afi + " access-list standard " + acl["name"] + else: + command = afi + " access-list " + acl["name"] + commandset.append(command) + if "aces" not in acl.keys(): + continue + for ace in acl["aces"]: + command = "" + if "sequence" in ace.keys(): + command = str(ace["sequence"]) + if "remark" in ace.keys(): + command = command + " remark " + ace["remark"] + if "fragment_rules" in ace.keys() and ace["fragment_rules"]: + command = command + " fragment-rules" + if "grant" in ace.keys(): + command = command + " " + ace["grant"] + if "vlan" in ace.keys(): + command = command + " vlan " + ace["vlan"] + if "protocol" in ace.keys(): + protocol = ace["protocol"] + if protocol.isdigit(): + if protocol in protocol_name.keys(): + protocol = protocol_name[protocol] + command = command + " " + protocol + if "source" in ace.keys(): + if "any" in ace["source"].keys(): + command = command + " any" + elif "subnet_address" in ace["source"].keys(): + command = command + " " + ace["source"]["subnet_address"] + elif "host" in ace["source"].keys(): + command = command + " host " + ace["source"]["host"] + elif "address" in ace["source"].keys(): + command = ( + command + + " " + + ace["source"]["address"] + + " " + + ace["source"]["wildcard_bits"] + ) + if "port_protocol" in ace["source"].keys(): + for op, val in ace["source"]["port_protocol"].items(): + if val.isdigit(): + val = socket.getservbyport(int(val)) + command = command + " " + op + " " + val + if "destination" in ace.keys(): + if "any" in ace["destination"].keys(): + command = command + " any" + elif "subnet_address" in ace["destination"].keys(): + command = ( + command + " " + ace["destination"]["subnet_address"] + ) + elif "host" in ace["destination"].keys(): + command = command + " host " + ace["destination"]["host"] + elif "address" in ace["destination"].keys(): + command = ( + command + + " " + + ace["destination"]["address"] + + " " + + ace["destination"]["wildcard_bits"] + ) + if "port_protocol" in ace["destination"].keys(): + for op in ace["destination"]["port_protocol"].keys(): + command = ( + command + + " " + + op + + " " + + ace["destination"]["port_protocol"][op] + ) + if "protocol_options" in ace.keys(): + for proto in ace["protocol_options"].keys(): + if proto == "icmp" or proto == "icmpv6": + for icmp_msg in ace["protocol_options"][proto].keys(): + command = command + " " + icmp_msg + elif proto == "ip" or proto == "ipv6": + command = ( + command + + " nexthop-group " + + ace["protocol_options"][proto]["nexthop_group"] + ) + elif proto == "tcp": + for flag, val in ace["protocol_options"][proto][ + "flags" + ].items(): + if val: + command = command + " " + flag + if "hop_limit" in ace.keys(): + for op, val in ace["hop_limit"].items(): + command = command + " hop-limit " + op + " " + val + if "tracked" in ace.keys() and ace["tracked"]: + command = command + " tracked" + if "ttl" in ace.keys(): + for op, val in ace["ttl"].items(): + command = command + " ttl " + op + " " + str(val) + if "fragments" in ace.keys(): + command = command + " fragments" + if "log" in ace.keys(): + command = command + " log" + commandset.append(command.strip()) + return commandset + + +def del_commands(want, have, name_only=False): + commandset = [] + command = "" + have_command = [] + for h in have: + have_configs = add_commands(h) + have_command.append(have_configs) + have_command = list(itertools.chain(*have_command)) + afi = "ip" if want["afi"] == "ipv4" else "ipv6" + if "acls" not in want.keys(): + for have_cmd in have_command: + access_list = re.search(r"(ip.*)\s+access-list .*", have_cmd) + if access_list and access_list.group(1) == afi: + commandset.append("no " + have_cmd) + return commandset + + for acl in want["acls"]: + ace_present = True + if "standard" in acl.keys() and acl["standard"]: + command = afi + " access-list standard " + acl["name"] + else: + command = afi + " access-list " + acl["name"] + if "aces" not in acl.keys(): + ace_present = False + commandset.append("no " + command) + if ace_present: + if name_only: + msg = "Deleted operation allows deletion of access-list only and not the entries !!" + return "Warn " + msg + return_command = add_commands(want) + for cmd in return_command: + if "access-list" in cmd: + commandset.append(cmd) + continue + seq = re.search( + r"(\d+) (permit|deny|fragment-rules|remark) .*", cmd + ) + if seq: + commandset.append("no " + seq.group(1)) + else: + commandset.append("no " + cmd) + return commandset + + +def get_ace_diff(want_ace, have_ace): + # gives the diff of the aces passed. + for h_a in have_ace: + d = dict_diff(want_ace, h_a) + if not d: + break + return d diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/config/bgp_address_family/__init__.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/config/bgp_address_family/__init__.py new file mode 100644 index 00000000..e69de29b --- /dev/null +++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/config/bgp_address_family/__init__.py diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/config/bgp_address_family/bgp_address_family.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/config/bgp_address_family/bgp_address_family.py new file mode 100644 index 00000000..559cd624 --- /dev/null +++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/config/bgp_address_family/bgp_address_family.py @@ -0,0 +1,272 @@ +# +# -*- coding: utf-8 -*- +# Copyright 2021 Red Hat +# 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 + +""" +The eos_bgp_address_family config file. +It is in this file where the current configuration (as dict) +is compared to the provided configuration (as dict) and the command set +necessary to bring the current configuration to its desired end-state is +created. +""" + +from ansible.module_utils.six import iteritems +from ansible_collections.ansible.netcommon.plugins.module_utils.network.common.utils import ( + dict_merge, +) +from ansible_collections.ansible.netcommon.plugins.module_utils.network.common.resource_module import ( + ResourceModule, +) +from ansible_collections.arista.eos.plugins.module_utils.network.eos.facts.facts import ( + Facts, +) +from ansible_collections.arista.eos.plugins.module_utils.network.eos.rm_templates.bgp_address_family import ( + Bgp_afTemplate, +) +import re + + +class Bgp_af(ResourceModule): + """ + The eos_bgp_address_family config class + """ + + def __init__(self, module): + super(Bgp_af, self).__init__( + empty_fact_val={}, + facts_module=Facts(module), + module=module, + resource="bgp_address_family", + tmplt=Bgp_afTemplate(), + ) + self.parsers = [ + "router", + "address_family", + "bgp_params_additional_paths", + "bgp_params.nexthop_address_family", + "bgp_params.nexthop_unchanged", + "bgp_params.redistribute_internal", + "bgp_params.route", + "graceful_restart", + "neighbor.activate", + "neighbor.additional_paths", + "neighbor.default_originate", + "neighbor.graceful_restart", + "neighbor.next_hop_unchanged", + "neighbor.next_hop_address_family", + "neighbor.prefix_list", + "neighbor.route_map", + "neighbor.weight", + "neighbor.encapsulation", + "network", + "redistribute", + "route_target", + ] + + def execute_module(self): + """ Execute the module + + :rtype: A dictionary + :returns: The result from module execution + """ + if self.state not in ["parsed", "gathered"]: + self.generate_commands() + self.run_commands() + return self.result + + def generate_commands(self): + """ Generate configuration commands to send based on + want, have and desired state. + """ + wantd = {} + haved = {} + + if self.want: + wantd = {self.want["as_number"]: self.want} + if self.have: + haved = {self.have["as_number"]: self.have} + + # turn all lists of dicts into dicts prior to merge + for entry in wantd, haved: + self._bgp_af_list_to_dict(entry) + + # if state is merged, merge want onto have and then compare + if self.state == "merged": + wantd = dict_merge(haved, wantd) + if len(wantd.keys()) > 1: + self._module.fail_json( + msg="Only one bgp instance is allowed per device" + ) + wantd = {} + # if state is deleted, empty out wantd and set haved to wantd + if self.state == "deleted": + haved = { + k: v for k, v in iteritems(haved) if k in wantd or not wantd + } + for wk, wv in iteritems(wantd): + self._compare(want=wv, have=haved.pop(wk, {})) + + wantd = {} + + # remove superfluous config for overridden + if self.state == "overridden": + for k, have in iteritems(haved): + if k not in wantd: + self._compare(want={}, have=have) + for k, want in iteritems(wantd): + self._compare(want=want, have=haved.pop(k, {})) + + def _delete_af(self, want, have): + waf = want.get("address_family", {}) + haf = have.get("address_family", {}) + for hkey, entry in iteritems(haf): + if hkey in waf.keys(): + af_no_command = self._tmplt.render( + entry, "address_family", True + ).split("\n") + if re.search(r"\S+_\S+", hkey): + af_no_command[0] = af_no_command[0][3:] + af_no_command[1] = "no " + af_no_command[1] + for cmd in af_no_command: + self.commands.append(cmd) + else: + self.addcmd(entry, "address_family", True) + have = {} + + def _compare(self, want, have): + """Leverages the base class `compare()` method and + populates the list of commands to be run by comparing + the `want` and `have` data with the `parsers` defined + for the Bgp_af network resource. + """ + for name, entry in iteritems(want): + if name != "as_number": + if self.state == "deleted": + self._delete_af(want, have) + else: + self._compare_af({name: entry}, {name: have.get(name, {})}) + + if self.commands and "router bgp" not in self.commands[0]: + self.commands.insert( + 0, + self._tmplt.render( + {"as_number": want.get("as_number") or have["as_number"]}, + "router", + False, + ), + ) + + def _compare_af(self, want, have): + waf = want.get("address_family", {}) + haf = have.get("address_family", {}) + for name, entry in iteritems(waf): + begin = len(self.commands) + self._compare_lists(entry, have=haf.get(name, {})) + self._compare_neighbor(entry, have=haf.get(name, {})) + self.compare( + parsers=self.parsers, want=entry, have=haf.pop(name, {}) + ) + if len(self.commands) != begin: + af_command = self._tmplt.render( + entry, "address_family", False + ).split("\n") + for cmd in af_command: + self.commands.insert(begin, cmd) + self.commands.append("exit") + begin += 1 + for name, entry in iteritems(haf): + # skip superfluous configs for replaced + if self.state in ["replaced"]: + if name in waf.keys(): + self.addcmd(entry, "address_family", True) + else: + # overridden + # check if want has vrf or not + # if want doesnot have vrf, device's vrf config will not + # be touched. + vrf_present = False + for w_key in waf.keys(): + if re.search(r"\S+_\S+", w_key): + vrf_present = True + break + if vrf_present: + if re.search(r"\S+_\S+", name): + af_no_command = self._tmplt.render( + entry, "address_family", True + ).split("\n") + if name not in waf.keys(): + af_no_command[0] = af_no_command[0][3:] + af_no_command[1] = "no " + af_no_command[1] + for cmd in af_no_command: + self.commands.append(cmd) + else: + self.addcmd(entry, "address_family", True) + else: + if not re.search(r"\S+_\S+", name): + self.addcmd(entry, "address_family", True) + + def _compare_neighbor(self, want, have): + parsers = [ + "neighbor.activate", + "neighbor.additional_paths", + "neighbor.default_originate", + "neighbor.graceful_restart", + "neighbor.next_hop_unchanged", + "neighbor.next_hop_address_family", + "neighbor.prefix_list", + "neighbor.route_map", + "neighbor.weight", + "neighbor.encapsulation", + ] + wneigh = want.get("neighbor", {}) + hneigh = have.get("neighbor", {}) + for name, entry in iteritems(wneigh): + self.compare( + parsers=parsers, + want={"neighbor": entry}, + have={"neighbor": hneigh.pop(name, {})}, + ) + for name, entry in iteritems(hneigh): + self.compare(parsers=parsers, want={}, have={"neighbor": entry}) + + def _compare_lists(self, want, have): + for attrib in ["redistribute", "network"]: + wdict = want.pop(attrib, {}) + hdict = have.pop(attrib, {}) + for key, entry in iteritems(wdict): + if entry != hdict.pop(key, {}): + self.addcmd(entry, attrib, False) + # remove remaining items in have for replaced + for entry in hdict.values(): + self.addcmd(entry, attrib, True) + + def _bgp_af_list_to_dict(self, entry): + for name, proc in iteritems(entry): + if "address_family" in proc: + proc["address_family"] = { + entry["afi"] + "_" + entry.get("vrf", ""): entry + for entry in proc.get("address_family", []) + } + self._bgp_af_list_to_dict(proc["address_family"]) + + if "neighbor" in proc: + proc["neighbor"] = { + entry["peer"]: entry for entry in proc.get("neighbor", []) + } + if "network" in proc: + proc["network"] = { + entry["address"]: entry + for entry in proc.get("network", []) + } + if "redistribute" in proc: + proc["redistribute"] = { + entry["protocol"]: entry + for entry in proc.get("redistribute", []) + } diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/config/bgp_global/__init__.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/config/bgp_global/__init__.py new file mode 100644 index 00000000..e69de29b --- /dev/null +++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/config/bgp_global/__init__.py diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/config/bgp_global/bgp_global.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/config/bgp_global/bgp_global.py new file mode 100644 index 00000000..e2a3c973 --- /dev/null +++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/config/bgp_global/bgp_global.py @@ -0,0 +1,379 @@ +# +# -*- coding: utf-8 -*- +# Copyright 2020 Red Hat +# 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 + +""" +The eos_bgp_global config file. +It is in this file where the current configuration (as dict) +is compared to the provided configuration (as dict) and the command set +necessary to bring the current configuration to its desired end-state is +created. +""" + +from ansible.module_utils.six import iteritems +from ansible_collections.ansible.netcommon.plugins.module_utils.network.common.utils import ( + dict_merge, +) +from ansible_collections.ansible.netcommon.plugins.module_utils.network.common.resource_module import ( + ResourceModule, +) +from ansible_collections.arista.eos.plugins.module_utils.network.eos.facts.facts import ( + Facts, +) +from ansible_collections.arista.eos.plugins.module_utils.network.eos.rm_templates.bgp_global import ( + Bgp_globalTemplate, +) + + +class Bgp_global(ResourceModule): + """ + The eos_bgp_global config class + """ + + def __init__(self, module): + super(Bgp_global, self).__init__( + empty_fact_val={}, + facts_module=Facts(module), + module=module, + resource="bgp_global", + tmplt=Bgp_globalTemplate(), + ) + self.parsers = [ + "router", + "vrf", + "default_metric", + "distance", + "graceful_restart", + "graceful_restart_helper", + "acccess_group", + "maximum_paths", + "monitoring", + "route_target", + "router_id", + "shutdown", + "timers", + "ucmp_fec", + "ucmp_link_bandwidth", + "ucmp_mode", + "update", + "vlan", + "vlan_aware_bundle", + ] + + def execute_module(self): + """ Execute the module + + :rtype: A dictionary + :returns: The result from module execution + """ + if self.state not in ["parsed", "gathered"]: + self.generate_commands() + self.run_commands() + return self.result + + def generate_commands(self): + """ Generate configuration commands to send based on + want, have and desired state. + """ + wantd = {} + haved = {} + if ( + self.want.get("as_number") == self.have.get("as_number") + or not self.have + ): + if self.want: + wantd = {self.want["as_number"]: self.want} + if self.have: + haved = {self.have["as_number"]: self.have} + else: + self._module.fail_json( + msg="Only one bgp instance is allowed per device" + ) + + # turn all lists of dicts into dicts prior to merge + for entry in wantd, haved: + self._bgp_global_list_to_dict(entry) + + # if state is merged, merge want onto have and then compare + if self.state == "merged": + wantd = dict_merge(haved, wantd) + + # if state is deleted, empty out wantd and set haved to wantd + if self.state in ["deleted", "purged"]: + haved = { + k: v for k, v in iteritems(haved) if k in wantd or not wantd + } + wantd = {} + + if self.state == "deleted": + self._compare(want={}, have=self.have) + + if self.state == "purged": + for num, entry in iteritems(haved): + self.commands.append( + self._tmplt.render({"as_number": num}, "router", True) + ) + + for k, want in iteritems(wantd): + self._compare(want=want, have=haved.pop(k, {})) + + def _compare(self, want, have): + """Leverages the base class `compare()` method and + populates the list of commands to be run by comparing + the `want` and `have` data with the `parsers` defined + for the Bgp_global network resource. + """ + self._compare_vrfs(want, have) + self._compare_neighbor(want, have) + self._compare_lists(want, have) + self._compare_bgp_params(want, have) + for name, entry in iteritems(want): + if name != "as_number": + self.compare( + parsers=self.parsers, + want={name: entry}, + have={name: have.pop(name, {})}, + ) + for name, entry in iteritems(have): + if name != "as_number": + self.compare( + parsers=self.parsers, want={}, have={name: have.get(name)} + ) + + if self.commands and "router bgp" not in self.commands[0]: + self.commands.insert( + 0, self._tmplt.render(want or have, "router", False) + ) + + def _compare_bgp_params(self, want, have): + parsers = [ + "bgp_params_additional_paths", + "bgp_params_advertise_inactive", + "bgp_params_allowas_in", + "bgp_params_always_compare_med", + "bgp_params_asn", + "bgp_params_auto_local_addr", + "bgp_params_bestpath_as_path", + "bgp_params_bestpath_ecmp_fast", + "bgp_params_bestpath_med", + "bgp_params_bestpath_skip", + "bgp_params_tie_break", + "bgp_params_client_to_client", + "bgp_params_cluster_id", + "bgp_params_confederation", + "bgp_params_control_plane_filter", + "bgp_params_convergence", + "bgp_params_default", + "bgp_params.enforce_first_as", + "bgp_params.host_routes", + "bgp_params.labelled_unicast", + "bgp_params.listen_limit", + "bgp_params.listen_range", + "bgp_params.log_neighbor_changes", + "bgp_params.missing_policy", + "bgp_params.monitoring", + "bgp_params.nexthop_unchanged", + "bgp_params.redistribute_internal", + "bgp_params.route", + "bgp_params.route_reflector", + "bgp_params.transport", + ] + wbgp = want.pop("bgp_params", {}) + hbgp = have.pop("bgp_params", {}) + for name, entry in iteritems(wbgp): + self.compare( + parsers=parsers, + want={"bgp_params": {name: entry}}, + have={"bgp_params": {name: hbgp.pop(name, {})}}, + ) + for name, entry in iteritems(hbgp): + self.compare( + parsers=parsers, want={}, have={"bgp_params": {name: entry}} + ) + + def _compare_vrfs(self, want, have): + wvrf = want.pop("vrfs", {}) + hvrf = have.pop("vrfs", {}) + begin = len(self.commands) + for name, entry in iteritems(wvrf): + self._compare_neighbor(entry, hvrf.get(name, {})) + self._compare_lists(entry, hvrf.get(name, {})) + self._compare_bgp_params(entry, hvrf.get(name, {})) + for k, v in entry.items(): + if hvrf.get(name): + h = {k: hvrf[name].pop(k, {})} + else: + h = {} + if k != "vrf": + self.compare(parsers=self.parsers, want={k: v}, have=h) + + if len(self.commands) != begin: + self.commands.insert( + begin, self._tmplt.render({"vrf": name}, "vrf", False) + ) + self.commands.append("exit") + begin_negate = len(self.commands) + for name, entry in iteritems(hvrf): + if name not in wvrf.keys(): + if self._check_af(name): + self._module.fail_json( + msg="Use the _bgp_address_family module to delete the address_family under vrf, before replacing/deleting the vrf." + ) + else: + self.commands.append( + self._tmplt.render({"vrf": name}, "vrf", True) + ) + continue + self.compare(parsers=self.parsers, want={}, have=entry) + after_negate = len(self.commands) + if after_negate != begin_negate: + if "vrf " + name in self.commands: + index = self.commands.index("vrf " + name) + i = begin_negate + while i < after_negate: + cmd = self.commands.pop(i) + if cmd != "exit": + self.commands.insert(index + 1, cmd) + i += 1 + else: + self.commands.insert( + begin_negate, + self._tmplt.render({"vrf": name}, "vrf", False), + ) + self.commands.append("exit") + + def _get_config(self, connection): + return connection.get("show running-config | section bgp ") + + def _check_af(self, vrf): + af_present = False + if self._connection: + config_lines = self._get_config(self._connection).splitlines() + index = [i + 1 for i, el in enumerate(config_lines) if vrf in el] + if index: + # had to do this to escape flake8 and black errors + ind = index[0] + for line in config_lines[ind:]: + if "vrf" in line: + break + if "address-family" in line: + af_present = True + break + return af_present + + def _compare_neighbor(self, want, have): + parsers = [ + "neighbor.additional_paths", + "neighbor.allowas_in", + "neighbor.auto_local_addr", + "neighbor.default_originate", + "neighbor.description", + "neighbor.dont_capability_negotiate", + "neighbor.ebgp_multihop", + "neighbor.enforce_first_as", + "neighbor.export_localpref", + "neighbor.fall_over", + "neighbor.graceful_restart", + "neighbor.graceful_restart_helper", + "neighbor.idle_restart_timer", + "neighbor.import_localpref", + "neighbor.link_bandwidth", + "neighbor.local_as", + "neighbor.local_v6_addr", + "neighbor.maximum_accepted_routes", + "neighbor.maximum_received_routes", + "neighbor.metric_out", + "neighbor.monitoring", + "neighbor.next_hop_self", + "neighbor.next_hop_unchanged", + "neighbor.next_hop_v6_addr", + "neighbor.out_delay", + "neighbor.remote_as", + "neighbor.remove_private_as", + "neighbor.peer_group", + "neighbor.prefix_list", + "neighbor.route_map", + "neighbor.route_reflector_client", + "neighbor.route_to_peer", + "neighbor.send_community", + "neighbor.shutdown", + "neighbor.soft_reconfiguration", + "neighbor.transport", + "neighbor.timers", + "neighbor.ttl", + "neighbor.update_source", + "neighbor.weight", + ] + wneigh = want.pop("neighbor", {}) + hneigh = have.pop("neighbor", {}) + for name, entry in iteritems(wneigh): + for k, v in entry.items(): + peer = entry["peer"] + if hneigh.get(name): + h = {"peer": peer, k: hneigh[name].pop(k, {})} + else: + h = {} + self.compare( + parsers=parsers, + want={"neighbor": {"peer": peer, k: v}}, + have={"neighbor": h}, + ) + for name, entry in iteritems(hneigh): + if name not in wneigh.keys() and "peer_group" not in entry.keys(): + self.commands.append("no neighbor " + name) + continue + for k, v in entry.items(): + peer = entry["peer"] + self.compare( + parsers=parsers, + want={}, + have={"neighbor": {"peer": peer, k: v}}, + ) + + def _compare_lists(self, want, have): + for attrib in ["redistribute", "network", "aggregate_address"]: + wdict = want.pop(attrib, {}) + hdict = have.pop(attrib, {}) + for key, entry in iteritems(wdict): + if entry != hdict.pop(key, {}): + self.addcmd(entry, attrib, False) + # remove remaining items in have for replaced + for entry in hdict.values(): + self.addcmd(entry, attrib, True) + + def _bgp_global_list_to_dict(self, entry): + for name, proc in iteritems(entry): + if "neighbor" in proc: + proc["neighbor"] = { + entry["peer"]: entry for entry in proc.get("neighbor", []) + } + if "network" in proc: + proc["network"] = { + entry["address"]: entry + for entry in proc.get("network", []) + } + + if "aggregate_address" in proc: + proc["aggregate_address"] = { + entry["address"]: entry + for entry in proc.get("aggregate_address", []) + } + + if "redistribute" in proc: + proc["redistribute"] = { + entry["protocol"]: entry + for entry in proc.get("redistribute", []) + } + + if "vrfs" in proc: + proc["vrfs"] = { + entry["vrf"]: entry for entry in proc.get("vrfs", []) + } + self._bgp_global_list_to_dict(proc["vrfs"]) diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/config/interfaces/__init__.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/config/interfaces/__init__.py new file mode 100644 index 00000000..e69de29b --- /dev/null +++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/config/interfaces/__init__.py diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/config/interfaces/interfaces.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/config/interfaces/interfaces.py new file mode 100644 index 00000000..734ca311 --- /dev/null +++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/config/interfaces/interfaces.py @@ -0,0 +1,305 @@ +# -*- coding: utf-8 -*- +# Copyright 2019 Red Hat +# GNU General Public License v3.0+ +# (see COPYING or https://www.gnu.org/licenses/gpl-3.0.txt) +""" +The eos_interfaces class +It is in this file where the current configuration (as dict) +is compared to the provided configuration (as dict) and the command set +necessary to bring the current configuration to it's desired end-state is +created +""" + +from __future__ import absolute_import, division, print_function + +__metaclass__ = type + +import re + +from ansible_collections.ansible.netcommon.plugins.module_utils.network.common.cfg.base import ( + ConfigBase, +) +from ansible_collections.ansible.netcommon.plugins.module_utils.network.common.utils import ( + to_list, + dict_diff, + param_list_to_dict, +) +from ansible_collections.arista.eos.plugins.module_utils.network.eos.facts.facts import ( + Facts, +) +from ansible_collections.arista.eos.plugins.module_utils.network.eos.utils.utils import ( + normalize_interface, +) + + +class Interfaces(ConfigBase): + """ + The eos_interfaces class + """ + + gather_subset = ["!all", "!min"] + + gather_network_resources = ["interfaces"] + + def get_interfaces_facts(self, data=None): + """ Get the 'facts' (the current configuration) + + :rtype: A dictionary + :returns: The current configuration as a dictionary + """ + facts, _warnings = Facts(self._module).get_facts( + self.gather_subset, self.gather_network_resources, data=data + ) + interfaces_facts = facts["ansible_network_resources"].get("interfaces") + if not interfaces_facts: + return [] + return interfaces_facts + + def execute_module(self): + """ Execute the module + + :rtype: A dictionary + :returns: The result from module execution + """ + result = {"changed": False} + commands = list() + warnings = list() + + if self.state in self.ACTION_STATES: + existing_interfaces_facts = self.get_interfaces_facts() + else: + existing_interfaces_facts = [] + + if self.state in self.ACTION_STATES or self.state == "rendered": + commands.extend(self.set_config(existing_interfaces_facts)) + + if commands and self.state in self.ACTION_STATES: + if not self._module.check_mode: + self._connection.edit_config(commands) + result["changed"] = True + if self.state in self.ACTION_STATES: + result["commands"] = commands + + if self.state in self.ACTION_STATES or self.state == "gathered": + changed_interfaces_facts = self.get_interfaces_facts() + + elif self.state == "rendered": + result["rendered"] = commands + + elif self.state == "parsed": + running_config = self._module.params["running_config"] + if not running_config: + self._module.fail_json( + msg="value of running_config parameter must not be empty for state parsed" + ) + result["parsed"] = self.get_interfaces_facts(data=running_config) + + if self.state in self.ACTION_STATES: + result["before"] = existing_interfaces_facts + if result["changed"]: + result["after"] = changed_interfaces_facts + + elif self.state == "gathered": + result["gathered"] = changed_interfaces_facts + result["warnings"] = warnings + return result + + def set_config(self, existing_interfaces_facts): + """ Collect the configuration from the args passed to the module, + collect the current configuration (as a dict from facts) + + :rtype: A list + :returns: the commands necessary to migrate the current configuration + to the desired configuration + """ + want = self._module.params["config"] + have = existing_interfaces_facts + resp = self.set_state(want, have) + return to_list(resp) + + def set_state(self, want, have): + """ Select the appropriate function based on the state provided + + :param want: the desired configuration as a dictionary + :param have: the current configuration as a dictionary + :rtype: A list + :returns: the commands necessary to migrate the current configuration + to the desired configuration + """ + state = self._module.params["state"] + if ( + state in ("merged", "replaced", "overridden", "rendered") + and not want + ): + self._module.fail_json( + msg="value of config parameter must not be empty for state {0}".format( + state + ) + ) + want = param_list_to_dict(want) + have = param_list_to_dict(have) + commands = [] + if state == "overridden": + commands = self._state_overridden(want, have) + elif state == "deleted": + commands = self._state_deleted(want, have) + elif state == "merged" or state == "rendered": + commands = self._state_merged(want, have) + elif state == "replaced": + commands = self._state_replaced(want, have) + return commands + + @staticmethod + def _state_replaced(want, have): + """ The command generator when state is replaced + + :rtype: A list + :returns: the commands necessary to migrate the current configuration + to the desired configuration + """ + commands = [] + for key, desired in want.items(): + interface_name = normalize_interface(key) + if interface_name in have: + extant = have[interface_name] + else: + extant = dict() + + add_config = dict_diff(extant, desired) + del_config = dict_diff(desired, extant) + + if ( + "speed" in add_config.keys() + and "duplex" not in add_config.keys() + ): + add_config.update({"duplex": desired.get("duplex")}) + + commands.extend(generate_commands(key, add_config, del_config)) + + return commands + + @staticmethod + def _state_overridden(want, have): + """ The command generator when state is overridden + + :rtype: A list + :returns: the commands necessary to migrate the current configuration + to the desired configuration + """ + commands = [] + for key, extant in have.items(): + if key in want: + desired = want[key] + else: + desired = dict() + + add_config = dict_diff(extant, desired) + del_config = dict_diff(desired, extant) + + if ( + "speed" in add_config.keys() + and "duplex" not in add_config.keys() + ): + add_config.update({"duplex": desired.get("duplex")}) + + commands.extend(generate_commands(key, add_config, del_config)) + + return commands + + @staticmethod + def _state_merged(want, have): + """ The command generator when state is merged + + :rtype: A list + :returns: the commands necessary to merge the provided into + the current configuration + """ + commands = [] + for key, desired in want.items(): + interface_name = normalize_interface(key) + if interface_name in have: + extant = have[interface_name] + else: + extant = dict() + + add_config = dict_diff(extant, desired) + if ( + "speed" in add_config.keys() + and "duplex" not in add_config.keys() + ): + add_config.update({"duplex": desired.get("duplex")}) + commands.extend(generate_commands(key, add_config, {})) + + return commands + + @staticmethod + def _state_deleted(want, have): + """ The command generator when state is deleted + + :rtype: A list + :returns: the commands necessary to remove the current configuration + of the provided objects + """ + commands = [] + for key in want: + desired = dict() + if key in have: + extant = have[key] + else: + continue + + del_config = dict_diff(desired, extant) + + commands.extend(generate_commands(key, {}, del_config)) + + return commands + + +def generate_commands(interface, to_set, to_remove): + commands = [] + for key, value in to_set.items(): + if value is None: + continue + + if key == "enabled": + commands.append("{0}shutdown".format("no " if value else "")) + elif key == "speed": + if value == "auto": + commands.append("{0} {1}".format(key, value)) + else: + commands.append("speed {0}{1}".format(value, to_set["duplex"])) + elif key == "duplex": + # duplex is handled with speed + continue + elif key == "mode": + if not re.search(r"(M|m)anagement.*", interface): + if value == "layer3": + # switching from default (layer2) mode to layer3 + commands.append("no switchport") + else: + # setting to default (layer 2) mode + commands.append("switchport") + else: + commands.append("{0} {1}".format(key, value)) + + # Don't try to also remove the same key, if present in to_remove + to_remove.pop(key, None) + + for key in to_remove.keys(): + if key == "enabled": + commands.append("no shutdown") + elif key == "speed": + commands.append("speed auto") + elif key == "duplex": + # duplex is handled with speed + continue + elif key == "mode": + if not re.search(r"(M|m)anagement.*", interface): + commands.append("switchport") + else: + commands.append("no {0}".format(key)) + + if commands: + commands.insert(0, "interface {0}".format(interface)) + + return commands diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/config/l2_interfaces/__init__.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/config/l2_interfaces/__init__.py new file mode 100644 index 00000000..e69de29b --- /dev/null +++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/config/l2_interfaces/__init__.py diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/config/l2_interfaces/l2_interfaces.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/config/l2_interfaces/l2_interfaces.py new file mode 100644 index 00000000..6c7d2ab9 --- /dev/null +++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/config/l2_interfaces/l2_interfaces.py @@ -0,0 +1,361 @@ +# -*- coding: utf-8 -*- +# Copyright 2019 Red Hat +# GNU General Public License v3.0+ +# (see COPYING or https://www.gnu.org/licenses/gpl-3.0.txt) +""" +The eos_l2_interfaces class +It is in this file where the current configuration (as dict) +is compared to the provided configuration (as dict) and the command set +necessary to bring the current configuration to it's desired end-state is +created +""" + +from __future__ import absolute_import, division, print_function + +__metaclass__ = type + +from ansible_collections.ansible.netcommon.plugins.module_utils.network.common.cfg.base import ( + ConfigBase, +) +from ansible_collections.ansible.netcommon.plugins.module_utils.network.common.utils import ( + to_list, + param_list_to_dict, +) +from ansible_collections.arista.eos.plugins.module_utils.network.eos.facts.facts import ( + Facts, +) +from ansible_collections.arista.eos.plugins.module_utils.network.eos.utils.utils import ( + normalize_interface, + vlan_range_to_list, +) + + +class L2_interfaces(ConfigBase): + """ + The eos_l2_interfaces class + """ + + gather_subset = ["!all", "!min"] + + gather_network_resources = ["l2_interfaces"] + + def get_l2_interfaces_facts(self, data=None): + """ Get the 'facts' (the current configuration) + + :rtype: A dictionary + :returns: The current configuration as a dictionary + """ + facts, _warnings = Facts(self._module).get_facts( + self.gather_subset, self.gather_network_resources, data=data + ) + l2_interfaces_facts = facts["ansible_network_resources"].get( + "l2_interfaces" + ) + if not l2_interfaces_facts: + return [] + return l2_interfaces_facts + + def execute_module(self): + """ Execute the module + + :rtype: A dictionary + :returns: The result from module execution + """ + result = {"changed": False} + commands = list() + warnings = list() + + if self.state in self.ACTION_STATES: + existing_l2_interfaces_facts = self.get_l2_interfaces_facts() + else: + existing_l2_interfaces_facts = [] + + if self.state in self.ACTION_STATES or self.state == "rendered": + commands.extend(self.set_config(existing_l2_interfaces_facts)) + + if commands and self.state in self.ACTION_STATES: + if not self._module.check_mode: + self._connection.edit_config(commands) + result["changed"] = True + + if self.state in self.ACTION_STATES: + result["commands"] = commands + + if self.state in self.ACTION_STATES or self.state == "gathered": + changed_l2_interfaces_facts = self.get_l2_interfaces_facts() + + elif self.state == "rendered": + result["rendered"] = commands + + elif self.state == "parsed": + running_config = self._module.params["running_config"] + if not running_config: + self._module.fail_json( + msg="value of running_config parameter must not be empty for state parsed" + ) + result["parsed"] = self.get_l2_interfaces_facts( + data=running_config + ) + + if self.state in self.ACTION_STATES: + result["before"] = existing_l2_interfaces_facts + if result["changed"]: + result["after"] = changed_l2_interfaces_facts + + elif self.state == "gathered": + result["gathered"] = changed_l2_interfaces_facts + + result["warnings"] = warnings + return result + + def set_config(self, existing_l2_interfaces_facts): + """ Collect the configuration from the args passed to the module, + collect the current configuration (as a dict from facts) + + :rtype: A list + :returns: the commands necessary to migrate the current configuration + to the desired configuration + """ + want = self._module.params["config"] + have = existing_l2_interfaces_facts + resp = self.set_state(want, have) + return to_list(resp) + + def set_state(self, want, have): + """ Select the appropriate function based on the state provided + + :param want: the desired configuration as a dictionary + :param have: the current configuration as a dictionary + :rtype: A list + :returns: the commands necessary to migrate the current configuration + to the desired configuration + """ + state = self._module.params["state"] + if ( + state in ("merged", "replaced", "overridden", "rendered") + and not want + ): + self._module.fail_json( + msg="value of config parameter must not be empty for state {0}".format( + state + ) + ) + want = param_list_to_dict(want) + have = param_list_to_dict(have) + commands = [] + if state == "overridden": + commands = self._state_overridden(want, have) + elif state == "deleted": + commands = self._state_deleted(want, have) + elif state == "merged" or state == "rendered": + commands = self._state_merged(want, have) + elif state == "replaced": + commands = self._state_replaced(want, have) + return commands + + @staticmethod + def _state_replaced(want, have): + """ The command generator when state is replaced + + :rtype: A list + :returns: the commands necessary to migrate the current configuration + to the desired configuration + """ + commands = [] + for key, desired in want.items(): + interface_name = normalize_interface(key) + if interface_name in have: + extant = have[interface_name] + else: + extant = dict() + + intf_commands = set_interface(desired, extant) + intf_commands.extend(clear_interface(desired, extant)) + + if intf_commands: + commands.append("interface {0}".format(interface_name)) + commands.extend(intf_commands) + + return commands + + @staticmethod + def _state_overridden(want, have): + """ The command generator when state is overridden + + :rtype: A list + :returns: the commands necessary to migrate the current configuration + to the desired configuration + """ + commands = [] + for key, extant in have.items(): + if key in want: + desired = want[key] + else: + desired = dict() + + intf_commands = set_interface(desired, extant) + intf_commands.extend(clear_interface(desired, extant)) + + if intf_commands: + commands.append("interface {0}".format(key)) + commands.extend(intf_commands) + + return commands + + @staticmethod + def _state_merged(want, have): + """ The command generator when state is merged + + :rtype: A list + :returns: the commands necessary to merge the provided into + the current configuration + """ + commands = [] + for key, desired in want.items(): + interface_name = normalize_interface(key) + if interface_name in have: + extant = have[interface_name] + else: + extant = dict() + + intf_commands = set_interface(desired, extant) + + if intf_commands: + commands.append("interface {0}".format(interface_name)) + commands.extend(intf_commands) + + return commands + + @staticmethod + def _state_deleted(want, have): + """ The command generator when state is deleted + + :rtype: A list + :returns: the commands necessary to remove the current configuration + of the provided objects + """ + commands = [] + for key in want: + desired = dict() + if key in have: + extant = have[key] + else: + continue + intf_commands = clear_interface(desired, extant) + + if intf_commands: + commands.append("interface {0}".format(key)) + commands.extend(intf_commands) + + return commands + + +def set_interface(want, have): + commands = [] + + want_mode = want.get("mode") + if want_mode and want_mode != have.get("mode"): + commands.append("switchport mode {0}".format(want_mode)) + + wants_access = want.get("access") + if wants_access: + access_vlan = wants_access.get("vlan") + if access_vlan and access_vlan != have.get("access", {}).get("vlan"): + commands.append("switchport access vlan {0}".format(access_vlan)) + + wants_trunk = want.get("trunk") + if wants_trunk: + allowed_vlans = [] + has_allowed_vlans = {} + want_allowed_vlans = {} + has_trunk = have.get("trunk", {}) + native_vlan = wants_trunk.get("native_vlan") + if native_vlan and native_vlan != has_trunk.get("native_vlan"): + commands.append( + "switchport trunk native vlan {0}".format(native_vlan) + ) + for con in [want, have]: + expand_trunk_allowed_vlans(con) + want_allowed_vlans = want["trunk"].get("trunk_allowed_vlans") + if has_trunk: + has_allowed_vlans = has_trunk.get("trunk_allowed_vlans") + + if want_allowed_vlans and has_allowed_vlans: + allowed_vlans = list( + set(want_allowed_vlans.split(",")) + - set(has_allowed_vlans.split(",")) + ) + elif want_allowed_vlans: + allowed_vlans = want_allowed_vlans.split(",") + if allowed_vlans: + allowed_vlans.sort() + allowed_vlans = ",".join( + ["{0}".format(vlan) for vlan in allowed_vlans] + ) + if has_allowed_vlans: + commands.append( + "switchport trunk allowed vlan add {0}".format( + allowed_vlans + ) + ) + else: + commands.append( + "switchport trunk allowed vlan {0}".format(allowed_vlans) + ) + return commands + + +def expand_trunk_allowed_vlans(want): + if not want: + return None + if want.get("trunk"): + if "trunk_allowed_vlans" in want["trunk"]: + allowed_vlans = vlan_range_to_list( + want["trunk"]["trunk_allowed_vlans"] + ) + vlans_list = [str(num) for num in sorted(allowed_vlans)] + want["trunk"]["trunk_allowed_vlans"] = ",".join(vlans_list) + + +def clear_interface(want, have): + commands = [] + + if "mode" in have and want.get("mode") is None: + commands.append("no switchport mode") + + if "access" in have and not want.get("access"): + commands.append("no switchport access vlan") + + has_trunk = have.get("trunk") or {} + wants_trunk = want.get("trunk") or {} + if ( + "trunk_allowed_vlans" in has_trunk + and "trunk_allowed_vlans" not in wants_trunk + ): + commands.append("no switchport trunk allowed vlan") + if ( + "trunk_allowed_vlans" in has_trunk + and "trunk_allowed_vlans" in wants_trunk + ): + for con in [want, have]: + expand_trunk_allowed_vlans(con) + want_allowed_vlans = want["trunk"].get("trunk_allowed_vlans") + has_allowed_vlans = has_trunk.get("trunk_allowed_vlans") + allowed_vlans = list( + set(has_allowed_vlans.split(",")) + - set(want_allowed_vlans.split(",")) + ) + if allowed_vlans: + allowed_vlans = ",".join( + ["{0}".format(vlan) for vlan in allowed_vlans] + ) + + commands.append( + "switchport trunk allowed vlan remove {0}".format( + allowed_vlans + ) + ) + + if "native_vlan" in has_trunk and "native_vlan" not in wants_trunk: + commands.append("no switchport trunk native vlan") + return commands diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/config/l3_interfaces/__init__.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/config/l3_interfaces/__init__.py new file mode 100644 index 00000000..e69de29b --- /dev/null +++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/config/l3_interfaces/__init__.py diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/config/l3_interfaces/l3_interfaces.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/config/l3_interfaces/l3_interfaces.py new file mode 100644 index 00000000..453332ea --- /dev/null +++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/config/l3_interfaces/l3_interfaces.py @@ -0,0 +1,322 @@ +# -*- coding: utf-8 -*- +# Copyright 2019 Red Hat +# GNU General Public License v3.0+ +# (see COPYING or https://www.gnu.org/licenses/gpl-3.0.txt) +""" +The eos_l3_interfaces class +It is in this file where the current configuration (as dict) +is compared to the provided configuration (as dict) and the command set +necessary to bring the current configuration to it's desired end-state is +created +""" + +from __future__ import absolute_import, division, print_function + +__metaclass__ = type + +from ansible_collections.ansible.netcommon.plugins.module_utils.network.common.cfg.base import ( + ConfigBase, +) +from ansible_collections.ansible.netcommon.plugins.module_utils.network.common.utils import ( + to_list, + param_list_to_dict, +) +from ansible_collections.arista.eos.plugins.module_utils.network.eos.facts.facts import ( + Facts, +) +from ansible_collections.arista.eos.plugins.module_utils.network.eos.utils.utils import ( + normalize_interface, +) + + +class L3_interfaces(ConfigBase): + """ + The eos_l3_interfaces class + """ + + gather_subset = ["!all", "!min"] + + gather_network_resources = ["l3_interfaces"] + + def get_l3_interfaces_facts(self, data=None): + """ Get the 'facts' (the current configuration) + + :rtype: A dictionary + :returns: The current configuration as a dictionary + """ + facts, _warnings = Facts(self._module).get_facts( + self.gather_subset, self.gather_network_resources, data=data + ) + l3_interfaces_facts = facts["ansible_network_resources"].get( + "l3_interfaces" + ) + if not l3_interfaces_facts: + return [] + return l3_interfaces_facts + + def execute_module(self): + """ Execute the module + + :rtype: A dictionary + :returns: The result from module execution + """ + result = {"changed": False} + commands = list() + warnings = list() + + if self.state in self.ACTION_STATES: + existing_l3_interfaces_facts = self.get_l3_interfaces_facts() + else: + existing_l3_interfaces_facts = [] + + if self.state in self.ACTION_STATES or self.state == "rendered": + commands.extend(self.set_config(existing_l3_interfaces_facts)) + + if commands and self.state in self.ACTION_STATES: + if not self._module.check_mode: + self._connection.edit_config(commands) + result["changed"] = True + if self.state in self.ACTION_STATES: + result["commands"] = commands + + if self.state in self.ACTION_STATES or self.state == "gathered": + changed_l3_interfaces_facts = self.get_l3_interfaces_facts() + + elif self.state == "rendered": + result["rendered"] = commands + + elif self.state == "parsed": + running_config = self._module.params["running_config"] + if not running_config: + self._module.fail_json( + msg="value of running_config parameter must not be empty for state parsed" + ) + result["parsed"] = self.get_l3_interfaces_facts( + data=running_config + ) + + if self.state in self.ACTION_STATES: + result["before"] = existing_l3_interfaces_facts + if result["changed"]: + result["after"] = changed_l3_interfaces_facts + + elif self.state == "gathered": + result["gathered"] = changed_l3_interfaces_facts + + result["warnings"] = warnings + return result + + def set_config(self, existing_l3_interfaces_facts): + """ Collect the configuration from the args passed to the module, + collect the current configuration (as a dict from facts) + + :rtype: A list + :returns: the commands necessary to migrate the current configuration + to the desired configuration + """ + want = self._module.params["config"] + have = existing_l3_interfaces_facts + resp = self.set_state(want, have) + return to_list(resp) + + def set_state(self, want, have): + """ Select the appropriate function based on the state provided + + :param want: the desired configuration as a dictionary + :param have: the current configuration as a dictionary + :rtype: A list + :returns: the commands necessary to migrate the current configuration + to the desired configuration + """ + state = self._module.params["state"] + if ( + state in ("merged", "replaced", "overridden", "rendered") + and not want + ): + self._module.fail_json( + msg="value of config parameter must not be empty for state {0}".format( + state + ) + ) + want = param_list_to_dict(want) + have = param_list_to_dict(have) + commands = [] + if state == "overridden": + commands = self._state_overridden(want, have) + elif state == "deleted": + commands = self._state_deleted(want, have) + elif state == "merged" or state == "rendered": + commands = self._state_merged(want, have) + elif state == "replaced": + commands = self._state_replaced(want, have) + return commands + + @staticmethod + def _state_replaced(want, have): + """ The command generator when state is replaced + + :rtype: A list + :returns: the commands necessary to migrate the current configuration + to the desired configuration + """ + commands = [] + for key, desired in want.items(): + interface_name = normalize_interface(key) + if interface_name in have: + extant = have[interface_name] + else: + extant = dict() + intf_commands = set_interface(desired, extant) + intf_commands.extend(clear_interface(desired, extant)) + + if intf_commands: + commands.append("interface {0}".format(interface_name)) + commands.extend(intf_commands) + + return commands + + @staticmethod + def _state_overridden(want, have): + """ The command generator when state is overridden + + :rtype: A list + :returns: the commands necessary to migrate the current configuration + to the desired configuration + """ + commands = [] + for key, extant in have.items(): + if key in want: + desired = want[key] + else: + desired = dict() + if desired.get("ipv4"): + for ipv4 in desired["ipv4"]: + if ipv4["secondary"] is None: + del ipv4["secondary"] + + intf_commands = set_interface(desired, extant) + intf_commands.extend(clear_interface(desired, extant)) + + if intf_commands: + commands.append("interface {0}".format(key)) + commands.extend(intf_commands) + + return commands + + @staticmethod + def _state_merged(want, have): + """ The command generator when state is merged + + :rtype: A list + :returns: the commands necessary to merge the provided into + the current configuration + """ + commands = [] + for key, desired in want.items(): + interface_name = normalize_interface(key) + if interface_name in have: + extant = have[interface_name] + else: + extant = dict() + + intf_commands = set_interface(desired, extant) + if intf_commands: + commands.append("interface {0}".format(interface_name)) + commands.extend(intf_commands) + + return commands + + @staticmethod + def _state_deleted(want, have): + """ The command generator when state is deleted + + :rtype: A list + :returns: the commands necessary to remove the current configuration + of the provided objects + """ + commands = [] + for key in want: + desired = dict() + if key in have: + extant = have[key] + else: + continue + + intf_commands = clear_interface(desired, extant) + + if intf_commands: + commands.append("interface {0}".format(key)) + commands.extend(intf_commands) + + return commands + + +def set_interface(want, have): + commands = [] + + want_ipv4 = set( + tuple(sorted(address.items())) for address in want.get("ipv4") or [] + ) + have_ipv4 = set( + tuple(sorted(address.items())) for address in have.get("ipv4") or [] + ) + for address in want_ipv4 - have_ipv4: + address = dict(address) + if "secondary" in address and not address["secondary"]: + del address["secondary"] + if tuple(address.items()) in have_ipv4: + continue + + address_cmd = "ip address {0}".format(address["address"]) + if address.get("secondary"): + address_cmd += " secondary" + commands.append(address_cmd) + + want_ipv6 = set( + tuple(sorted(address.items())) for address in want.get("ipv6") or [] + ) + have_ipv6 = set( + tuple(sorted(address.items())) for address in have.get("ipv6") or [] + ) + for address in want_ipv6 - have_ipv6: + address = dict(address) + commands.append("ipv6 address {0}".format(address["address"])) + return commands + + +def clear_interface(want, have): + commands = [] + want_ipv4 = set( + tuple(sorted(address.items())) for address in want.get("ipv4") or [] + ) + have_ipv4 = set( + tuple(sorted(address.items())) for address in have.get("ipv4") or [] + ) + if not want_ipv4 and have_ipv4: + commands.append("no ip address") + else: + for address in have_ipv4 - want_ipv4: + address = dict(address) + if "secondary" not in address: + address["secondary"] = False + if tuple(address.items()) in want_ipv4: + continue + + if address.get("secondary"): + address_cmd = " {0} secondary".format(address["address"]) + commands.append(address_cmd) + + if "secondary" not in address: + # Removing non-secondary removes all other interfaces + break + + want_ipv6 = set( + tuple(sorted(address.items())) for address in want.get("ipv6") or [] + ) + have_ipv6 = set( + tuple(sorted(address.items())) for address in have.get("ipv6") or [] + ) + for address in have_ipv6 - want_ipv6: + address = dict(address) + commands.append("no ipv6 address {0}".format(address["address"])) + return commands diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/config/lacp/__init__.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/config/lacp/__init__.py new file mode 100644 index 00000000..e69de29b --- /dev/null +++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/config/lacp/__init__.py diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/config/lacp/lacp.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/config/lacp/lacp.py new file mode 100644 index 00000000..6d11ae3f --- /dev/null +++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/config/lacp/lacp.py @@ -0,0 +1,197 @@ +# -*- coding: utf-8 -*- +# Copyright 2019 Red Hat +# GNU General Public License v3.0+ +# (see COPYING or https://www.gnu.org/licenses/gpl-3.0.txt) +""" +The eos_lacp class +It is in this file where the current configuration (as dict) +is compared to the provided configuration (as dict) and the command set +necessary to bring the current configuration to it's desired end-state is +created +""" + +from __future__ import absolute_import, division, print_function + +__metaclass__ = type + +from ansible_collections.ansible.netcommon.plugins.module_utils.network.common.cfg.base import ( + ConfigBase, +) +from ansible_collections.ansible.netcommon.plugins.module_utils.network.common.utils import ( + to_list, + dict_diff, +) +from ansible_collections.arista.eos.plugins.module_utils.network.eos.facts.facts import ( + Facts, +) + + +class Lacp(ConfigBase): + """ + The eos_lacp class + """ + + gather_subset = ["!all", "!min"] + + gather_network_resources = ["lacp"] + + def get_lacp_facts(self, data=None): + """ Get the 'facts' (the current configuration) + + :rtype: A dictionary + :returns: The current configuration as a dictionary + """ + facts, _warnings = Facts(self._module).get_facts( + self.gather_subset, self.gather_network_resources, data=data + ) + lacp_facts = facts["ansible_network_resources"].get("lacp") + if not lacp_facts: + return {} + return lacp_facts + + def execute_module(self): + """ Execute the module + + :rtype: A dictionary + :returns: The result from module execution + """ + result = {"changed": False} + warnings = list() + commands = list() + + if self.state in self.ACTION_STATES: + existing_lacp_facts = self.get_lacp_facts() + else: + existing_lacp_facts = {} + + if self.state in self.ACTION_STATES or self.state == "rendered": + commands.extend(self.set_config(existing_lacp_facts)) + + if commands and self.state in self.ACTION_STATES: + if not self._module.check_mode: + self._connection.edit_config(commands) + result["changed"] = True + if self.state in self.ACTION_STATES: + result["commands"] = commands + + if self.state in self.ACTION_STATES or self.state == "gathered": + changed_lacp_facts = self.get_lacp_facts() + elif self.state == "rendered": + result["rendered"] = commands + elif self.state == "parsed": + running_config = self._module.params["running_config"] + if not running_config: + self._module.fail_json( + msg="value of running_config parameter must not be empty for state parsed" + ) + result["parsed"] = self.get_lacp_facts(data=running_config) + + if self.state in self.ACTION_STATES: + result["before"] = existing_lacp_facts + if result["changed"]: + result["after"] = changed_lacp_facts + + elif self.state == "gathered": + result["gathered"] = changed_lacp_facts + + result["warnings"] = warnings + return result + + def set_config(self, existing_lacp_facts): + """ Collect the configuration from the args passed to the module, + collect the current configuration (as a dict from facts) + + :rtype: A list + :returns: the commands necessary to migrate the current configuration + to the desired configuration + """ + want = self._module.params["config"] or {} + have = existing_lacp_facts + resp = self.set_state(want, have) + return to_list(resp) + + def set_state(self, want, have): + """ Select the appropriate function based on the state provided + + :param want: the desired configuration as a dictionary + :param have: the current configuration as a dictionary + :rtype: A list + :returns: the commands necessary to migrate the current configuration + to the desired configuration + """ + state = self._module.params["state"] + if state in ("merged", "replaced", "rendered") and not want: + self._module.fail_json( + msg="value of config parameter must not be empty for state {0}".format( + state + ) + ) + if state == "deleted": + commands = self._state_deleted(want, have) + elif state == "merged" or state == "rendered": + commands = self._state_merged(want, have) + elif state == "replaced": + commands = self._state_replaced(want, have) + return commands + + @staticmethod + def _state_replaced(want, have): + """ The command generator when state is replaced + + :rtype: A list + :returns: the commands necessary to migrate the current configuration + to the desired configuration + """ + commands = [] + to_set = dict_diff(have, want) + if "system" in to_set: + system = to_set["system"] + if "priority" in system: + commands.append( + "lacp system-priority {0}".format(system["priority"]) + ) + + to_del = dict_diff(want, have) + if "system" in to_del: + system = to_del["system"] + system_set = to_set.get("system", {}) + if "priority" in system and "priority" not in system_set: + commands.append("no lacp system-priority") + + return commands + + @staticmethod + def _state_merged(want, have): + """ The command generator when state is merged + + :rtype: A list + :returns: the commands necessary to merge the provided into + the current configuration + """ + commands = [] + to_set = dict_diff(have, want) + if "system" in to_set: + system = to_set["system"] + if "priority" in system: + commands.append( + "lacp system-priority {0}".format(system["priority"]) + ) + + return commands + + @staticmethod + def _state_deleted(want, have): + """ The command generator when state is deleted + + :rtype: A list + :returns: the commands necessary to remove the current configuration + of the provided objects + """ + commands = [] + to_del = dict_diff(want, have) + if "system" in to_del: + system = to_del["system"] + if "priority" in system: + commands.append("no lacp system-priority") + + return commands diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/config/lacp_interfaces/__init__.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/config/lacp_interfaces/__init__.py new file mode 100644 index 00000000..e69de29b --- /dev/null +++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/config/lacp_interfaces/__init__.py diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/config/lacp_interfaces/lacp_interfaces.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/config/lacp_interfaces/lacp_interfaces.py new file mode 100644 index 00000000..3f013824 --- /dev/null +++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/config/lacp_interfaces/lacp_interfaces.py @@ -0,0 +1,254 @@ +# -*- coding: utf-8 -*- +# Copyright 2019 Red Hat +# GNU General Public License v3.0+ +# (see COPYING or https://www.gnu.org/licenses/gpl-3.0.txt) +""" +The eos_lacp_interfaces class +It is in this file where the current configuration (as dict) +is compared to the provided configuration (as dict) and the command set +necessary to bring the current configuration to it's desired end-state is +created +""" + +from __future__ import absolute_import, division, print_function + +__metaclass__ = type + +from ansible_collections.ansible.netcommon.plugins.module_utils.network.common.cfg.base import ( + ConfigBase, +) +from ansible_collections.ansible.netcommon.plugins.module_utils.network.common.utils import ( + to_list, + dict_diff, + param_list_to_dict, +) +from ansible_collections.arista.eos.plugins.module_utils.network.eos.facts.facts import ( + Facts, +) +from ansible_collections.arista.eos.plugins.module_utils.network.eos.utils.utils import ( + normalize_interface, +) + + +class Lacp_interfaces(ConfigBase): + """ + The eos_lacp_interfaces class + """ + + gather_subset = ["!all", "!min"] + + gather_network_resources = ["lacp_interfaces"] + + def get_lacp_interfaces_facts(self, data=None): + """ Get the 'facts' (the current configuration) + + :rtype: A dictionary + :returns: The current configuration as a dictionary + """ + facts, _warnings = Facts(self._module).get_facts( + self.gather_subset, self.gather_network_resources, data=data + ) + lacp_interfaces_facts = facts["ansible_network_resources"].get( + "lacp_interfaces" + ) + if not lacp_interfaces_facts: + return [] + return lacp_interfaces_facts + + def execute_module(self): + """ Execute the module + + :rtype: A dictionary + :returns: The result from module execution + """ + result = {"changed": False} + warnings = list() + commands = list() + + if self.state in self.ACTION_STATES: + existing_lacp_interfaces_facts = self.get_lacp_interfaces_facts() + else: + existing_lacp_interfaces_facts = {} + + if self.state in self.ACTION_STATES or self.state == "rendered": + commands.extend(self.set_config(existing_lacp_interfaces_facts)) + + if commands and self.state in self.ACTION_STATES: + if not self._module.check_mode: + self._connection.edit_config(commands) + result["changed"] = True + if self.state in self.ACTION_STATES: + result["commands"] = commands + + if self.state in self.ACTION_STATES or self.state == "gathered": + changed_lacp_interfaces_facts = self.get_lacp_interfaces_facts() + elif self.state == "rendered": + result["rendered"] = commands + elif self.state == "parsed": + running_config = self._module.params["running_config"] + if not running_config: + self._module.fail_json( + msg="value of running_config parameter must not be empty for state parsed" + ) + result["parsed"] = self.get_lacp_interfaces_facts( + data=running_config + ) + if self.state in self.ACTION_STATES: + result["before"] = existing_lacp_interfaces_facts + if result["changed"]: + result["after"] = changed_lacp_interfaces_facts + elif self.state == "gathered": + result["gathered"] = changed_lacp_interfaces_facts + + result["warnings"] = warnings + return result + + def set_config(self, existing_lacp_interfaces_facts): + """ Collect the configuration from the args passed to the module, + collect the current configuration (as a dict from facts) + + :rtype: A list + :returns: the commands necessary to migrate the current configuration + to the desired configuration + """ + want = self._module.params["config"] + have = existing_lacp_interfaces_facts + resp = self.set_state(want, have) + return to_list(resp) + + def set_state(self, want, have): + """ Select the appropriate function based on the state provided + + :param want: the desired configuration as a dictionary + :param have: the current configuration as a dictionary + :rtype: A list + :returns: the commands necessary to migrate the current configuration + to the desired configuration + """ + state = self._module.params["state"] + if ( + state in ("merged", "replaced", "overridden", "rendered") + and not want + ): + self._module.fail_json( + msg="value of config parameter must not be empty for state {0}".format( + state + ) + ) + want = param_list_to_dict(want) + have = param_list_to_dict(have) + if state == "overridden": + commands = self._state_overridden(want, have) + elif state == "deleted": + commands = self._state_deleted(want, have) + elif state == "merged" or state == "rendered": + commands = self._state_merged(want, have) + elif state == "replaced": + commands = self._state_replaced(want, have) + return commands + + @staticmethod + def _state_replaced(want, have): + """ The command generator when state is replaced + + :rtype: A list + :returns: the commands necessary to migrate the current configuration + to the desired configuration + """ + commands = [] + for key, desired in want.items(): + interface_name = normalize_interface(key) + if interface_name in have: + extant = have[interface_name] + else: + extant = dict() + + add_config = dict_diff(extant, desired) + del_config = dict_diff(desired, extant) + + commands.extend(generate_commands(key, add_config, del_config)) + + return commands + + @staticmethod + def _state_overridden(want, have): + """ The command generator when state is overridden + + :rtype: A list + :returns: the commands necessary to migrate the current configuration + to the desired configuration + """ + commands = [] + for key, extant in have.items(): + if key in want: + desired = want[key] + else: + desired = dict() + + add_config = dict_diff(extant, desired) + del_config = dict_diff(desired, extant) + + commands.extend(generate_commands(key, add_config, del_config)) + + return commands + + @staticmethod + def _state_merged(want, have): + """ The command generator when state is merged + + :rtype: A list + :returns: the commands necessary to merge the provided into + the current configuration + """ + commands = [] + for key, desired in want.items(): + interface_name = normalize_interface(key) + if interface_name in have: + extant = have[interface_name] + else: + extant = dict() + + add_config = dict_diff(extant, desired) + + commands.extend(generate_commands(key, add_config, {})) + + return commands + + @staticmethod + def _state_deleted(want, have): + """ The command generator when state is deleted + + :rtype: A list + :returns: the commands necessary to remove the current configuration + of the provided objects + """ + commands = [] + for key in want: + desired = dict() + if key in have: + extant = have[key] + else: + continue + + del_config = dict_diff(desired, extant) + + commands.extend(generate_commands(key, {}, del_config)) + + return commands + + +def generate_commands(interface, to_set, to_remove): + commands = [] + for key in to_remove.keys(): + commands.append("no lacp {0}".format(key.replace("_", "-"))) + + for key, value in to_set.items(): + if value is None: + continue + + commands.append("lacp {0} {1}".format(key.replace("_", "-"), value)) + + if commands: + commands.insert(0, "interface {0}".format(interface)) + + return commands diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/config/lag_interfaces/__init__.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/config/lag_interfaces/__init__.py new file mode 100644 index 00000000..e69de29b --- /dev/null +++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/config/lag_interfaces/__init__.py diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/config/lag_interfaces/lag_interfaces.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/config/lag_interfaces/lag_interfaces.py new file mode 100644 index 00000000..87faf91f --- /dev/null +++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/config/lag_interfaces/lag_interfaces.py @@ -0,0 +1,271 @@ +# -*- coding: utf-8 -*- +# Copyright 2019 Red Hat +# GNU General Public License v3.0+ +# (see COPYING or https://www.gnu.org/licenses/gpl-3.0.txt) +""" +The eos_lag_interfaces class +It is in this file where the current configuration (as dict) +is compared to the provided configuration (as dict) and the command set +necessary to bring the current configuration to it's desired end-state is +created +""" + +from __future__ import absolute_import, division, print_function + +import re + +__metaclass__ = type + +from ansible_collections.ansible.netcommon.plugins.module_utils.network.common.utils import ( + to_list, + dict_diff, +) + +from ansible_collections.ansible.netcommon.plugins.module_utils.network.common.cfg.base import ( + ConfigBase, +) +from ansible_collections.arista.eos.plugins.module_utils.network.eos.facts.facts import ( + Facts, +) +from ansible_collections.arista.eos.plugins.module_utils.network.eos.utils.utils import ( + normalize_interface, +) + + +class Lag_interfaces(ConfigBase): + """ + The eos_lag_interfaces class + """ + + gather_subset = ["!all", "!min"] + + gather_network_resources = ["lag_interfaces"] + + def get_lag_interfaces_facts(self, data=None): + """ Get the 'facts' (the current configuration) + + :rtype: A dictionary + :returns: The current configuration as a dictionary + """ + facts, _warnings = Facts(self._module).get_facts( + self.gather_subset, self.gather_network_resources, data=data + ) + lag_interfaces_facts = facts["ansible_network_resources"].get( + "lag_interfaces" + ) + if not lag_interfaces_facts: + return [] + return lag_interfaces_facts + + def execute_module(self): + """ Execute the module + + :rtype: A dictionary + :returns: The result from module execution + """ + result = {"changed": False} + commands = list() + warnings = list() + + if self.state in self.ACTION_STATES: + existing_lag_interfaces_facts = self.get_lag_interfaces_facts() + else: + existing_lag_interfaces_facts = {} + + if self.state in self.ACTION_STATES or self.state == "rendered": + commands.extend(self.set_config(existing_lag_interfaces_facts)) + + if commands and self.state in self.ACTION_STATES: + if not self._module.check_mode: + self._connection.edit_config(commands) + result["changed"] = True + if self.state in self.ACTION_STATES: + result["commands"] = commands + + if self.state in self.ACTION_STATES or self.state == "gathered": + changed_lag_interfaces_facts = self.get_lag_interfaces_facts() + elif self.state == "rendered": + result["rendered"] = commands + elif self.state == "parsed": + running_config = self._module.params["running_config"] + if not running_config: + self._module.fail_json( + msg="value of running_config parameter must not be empty for state parsed" + ) + result["parsed"] = self.get_lag_interfaces_facts( + data=running_config + ) + + if self.state in self.ACTION_STATES: + result["before"] = existing_lag_interfaces_facts + if result["changed"]: + result["after"] = changed_lag_interfaces_facts + elif self.state == "gathered": + result["gathered"] = changed_lag_interfaces_facts + + result["warnings"] = warnings + return result + + def set_config(self, existing_lag_interfaces_facts): + """ Collect the configuration from the args passed to the module, + collect the current configuration (as a dict from facts) + + :rtype: A list + :returns: the commands necessary to migrate the current configuration + to the desired configuration + """ + want = self._module.params["config"] + have = existing_lag_interfaces_facts + resp = self.set_state(want, have) + return to_list(resp) + + def set_state(self, want, have): + """ Select the appropriate function based on the state provided + + :param want: the desired configuration as a dictionary + :param have: the current configuration as a dictionary + :rtype: A list + :returns: the commands necessary to migrate the current configuration + to the desired configuration + """ + state = self._module.params["state"] + if ( + state in ("merged", "replaced", "overridden", "rendered") + and not want + ): + self._module.fail_json( + msg="value of config parameter must not be empty for state {0}".format( + state + ) + ) + if state == "overridden": + commands = self._state_overridden(want, have) + elif state == "deleted": + commands = self._state_deleted(want, have) + elif state == "merged" or state == "rendered": + commands = self._state_merged(want, have) + elif state == "replaced": + commands = self._state_replaced(want, have) + return commands + + @staticmethod + def _state_replaced(want, have): + """ The command generator when state is replaced + :rtype: A list + :returns: the commands necessary to migrate the current configuration + to the desired configuration + """ + commands = [] + for interface in want: + interface_name = normalize_interface(interface["name"]) + for extant in have: + if extant["name"] == interface_name: + break + else: + extant = dict(name=interface_name) + + commands.extend(set_config(interface, extant)) + commands.extend(remove_config(interface, extant)) + + return commands + + @staticmethod + def _state_overridden(want, have): + """ The command generator when state is overridden + :rtype: A list + :returns: the commands necessary to migrate the current configuration + to the desired configuration + """ + commands = [] + for extant in have: + for interface in want: + if normalize_interface(interface["name"]) == extant["name"]: + break + else: + interface = dict(name=extant["name"]) + commands.extend(remove_config(interface, extant)) + + for interface in want: + interface_name = normalize_interface(interface["name"]) + for extant in have: + if extant["name"] == interface_name: + break + else: + extant = dict(name=interface_name) + commands.extend(set_config(interface, extant)) + + return commands + + @staticmethod + def _state_merged(want, have): + """ The command generator when state is merged + :rtype: A list + :returns: the commands necessary to merge the provided into + the current configuration + """ + commands = [] + for interface in want: + interface_name = normalize_interface(interface["name"]) + for extant in have: + if extant["name"] == interface_name: + break + else: + extant = dict(name=interface_name) + + commands.extend(set_config(interface, extant)) + + return commands + + @staticmethod + def _state_deleted(want, have): + """ The command generator when state is deleted + :rtype: A list + :returns: the commands necessary to remove the current configuration + of the provided objects + """ + commands = [] + for interface in want: + interface_name = normalize_interface(interface["name"]) + for extant in have: + if extant["name"] == interface_name: + break + else: + extant = dict(name=interface_name) + + # Clearing all args, send empty dictionary + interface = dict(name=interface_name) + commands.extend(remove_config(interface, extant)) + + return commands + + +def set_config(want, have): + commands = [] + to_set = dict_diff(have, want) + for member in to_set.get("members", []): + channel_id = re.search(r"\d.*", want["name"]) + if channel_id: + commands.extend( + [ + "interface {0}".format(member["member"]), + "channel-group {0} mode {1}".format( + channel_id.group(0), member["mode"] + ), + ] + ) + + return commands + + +def remove_config(want, have): + commands = [] + if not want.get("members"): + return ["no interface {0}".format(want["name"])] + + to_remove = dict_diff(want, have) + for member in to_remove.get("members", []): + commands.extend( + ["interface {0}".format(member["member"]), "no channel-group"] + ) + + return commands diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/config/lldp_global/__init__.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/config/lldp_global/__init__.py new file mode 100644 index 00000000..e69de29b --- /dev/null +++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/config/lldp_global/__init__.py diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/config/lldp_global/lldp_global.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/config/lldp_global/lldp_global.py new file mode 100644 index 00000000..caf1c409 --- /dev/null +++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/config/lldp_global/lldp_global.py @@ -0,0 +1,203 @@ +# +# -*- coding: utf-8 -*- +# Copyright 2019 Red Hat +# GNU General Public License v3.0+ +# (see COPYING or https://www.gnu.org/licenses/gpl-3.0.txt) +""" +The eos_lldp_global class +It is in this file where the current configuration (as dict) +is compared to the provided configuration (as dict) and the command set +necessary to bring the current configuration to it's desired end-state is +created +""" + +from __future__ import absolute_import, division, print_function + +__metaclass__ = type + +from ansible_collections.ansible.netcommon.plugins.module_utils.network.common.cfg.base import ( + ConfigBase, +) +from ansible_collections.ansible.netcommon.plugins.module_utils.network.common.utils import ( + dict_diff, + to_list, +) +from ansible_collections.arista.eos.plugins.module_utils.network.eos.facts.facts import ( + Facts, +) + + +class Lldp_global(ConfigBase): + """ + The eos_lldp_global class + """ + + gather_subset = ["!all", "!min"] + + gather_network_resources = ["lldp_global"] + + def __init__(self, module): + super(Lldp_global, self).__init__(module) + + def get_lldp_global_facts(self, data=None): + """ Get the 'facts' (the current configuration) + + :rtype: A dictionary + :returns: The current configuration as a dictionary + """ + facts, _warnings = Facts(self._module).get_facts( + self.gather_subset, self.gather_network_resources, data=data + ) + lldp_global_facts = facts["ansible_network_resources"].get( + "lldp_global" + ) + if not lldp_global_facts: + return {} + return lldp_global_facts + + def execute_module(self): + """ Execute the module + + :rtype: A dictionary + :returns: The result from module execution + """ + result = {"changed": False} + warnings = list() + commands = list() + + if self.state in self.ACTION_STATES: + existing_lldp_global_facts = self.get_lldp_global_facts() + else: + existing_lldp_global_facts = {} + + if self.state in self.ACTION_STATES or self.state == "rendered": + commands.extend(self.set_config(existing_lldp_global_facts)) + + if commands and self.state in self.ACTION_STATES: + if not self._module.check_mode: + self._connection.edit_config(commands) + result["changed"] = True + if self.state in self.ACTION_STATES: + result["commands"] = commands + + if self.state in self.ACTION_STATES or self.state == "gathered": + changed_lldp_global_facts = self.get_lldp_global_facts() + elif self.state == "rendered": + result["rendered"] = commands + elif self.state == "parsed": + running_config = self._module.params["running_config"] + if not running_config: + self._module.fail_json( + msg="value of running_config parameter must not be empty for state parsed" + ) + result["parsed"] = self.get_lldp_global_facts(data=running_config) + + if self.state in self.ACTION_STATES: + result["before"] = existing_lldp_global_facts + if result["changed"]: + result["after"] = changed_lldp_global_facts + + elif self.state == "gathered": + result["gathered"] = changed_lldp_global_facts + + result["warnings"] = warnings + return result + + def set_config(self, existing_lldp_global_facts): + """ Collect the configuration from the args passed to the module, + collect the current configuration (as a dict from facts) + + :rtype: A list + :returns: the commands necessary to migrate the current configuration + to the desired configuration + """ + want = self._module.params["config"] or {} + have = existing_lldp_global_facts + resp = self.set_state(want, have) + return to_list(resp) + + def set_state(self, want, have): + """ Select the appropriate function based on the state provided + + :param want: the desired configuration as a dictionary + :param have: the current configuration as a dictionary + :rtype: A list + :returns: the commands necessary to migrate the current configuration + to the desired configuration + """ + state = self._module.params["state"] + if ( + state in ("merged", "replaced", "overridden", "rendered") + and not want + ): + self._module.fail_json( + msg="value of config parameter must not be empty for state {0}".format( + state + ) + ) + if state == "deleted": + commands = state_deleted(want, have) + elif state == "merged" or state == "rendered": + commands = state_merged(want, have) + elif state == "replaced": + commands = state_replaced(want, have) + return commands + + +def state_replaced(want, have): + """ The command generator when state is replaced + + :rtype: A list + :returns: the commands necessary to migrate the current configuration + to the desired configuration + """ + commands = set() + # merged and deleted are likely to emit duplicate tlv-select commands + commands.update(state_merged(want, have)) + commands.update(state_deleted(want, have)) + + return list(commands) + + +def state_merged(want, have): + """ The command generator when state is merged + + :rtype: A list + :returns: the commands necessary to merge the provided into + the current configuration + """ + commands = [] + to_set = dict_diff(have, want) + tlv_options = to_set.pop("tlv_select", {}) + for key, value in to_set.items(): + commands.append("lldp {0} {1}".format(key, value)) + for key, value in tlv_options.items(): + device_option = key.replace("_", "-") + if value is True: + commands.append("lldp tlv-select {0}".format(device_option)) + elif value is False: + commands.append("no lldp tlv-select {0}".format(device_option)) + + return commands + + +def state_deleted(want, have): + """ The command generator when state is deleted + + :rtype: A list + :returns: the commands necessary to remove the current configuration + of the provided objects + """ + commands = [] + to_remove = dict_diff(want, have) + tlv_options = to_remove.pop("tlv_select", {}) + for key in to_remove: + commands.append("no lldp {0}".format(key)) + for key, value in tlv_options.items(): + device_option = key.replace("_", "-") + if value is False: + commands.append("lldp tlv-select {0}".format(device_option)) + elif value is True: + commands.append("no lldp tlv-select {0}".format(device_option)) + + return commands diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/config/lldp_interfaces/__init__.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/config/lldp_interfaces/__init__.py new file mode 100644 index 00000000..e69de29b --- /dev/null +++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/config/lldp_interfaces/__init__.py diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/config/lldp_interfaces/lldp_interfaces.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/config/lldp_interfaces/lldp_interfaces.py new file mode 100644 index 00000000..1a7354eb --- /dev/null +++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/config/lldp_interfaces/lldp_interfaces.py @@ -0,0 +1,260 @@ +# -*- coding: utf-8 -*- +# Copyright 2019 Red Hat +# GNU General Public License v3.0+ +# (see COPYING or https://www.gnu.org/licenses/gpl-3.0.txt) +""" +The eos_lldp_interfaces class +It is in this file where the current configuration (as dict) +is compared to the provided configuration (as dict) and the command set +necessary to bring the current configuration to it's desired end-state is +created +""" + +from __future__ import absolute_import, division, print_function + +__metaclass__ = type + +from ansible_collections.ansible.netcommon.plugins.module_utils.network.common.cfg.base import ( + ConfigBase, +) +from ansible_collections.ansible.netcommon.plugins.module_utils.network.common.utils import ( + to_list, + dict_diff, + param_list_to_dict, +) +from ansible_collections.arista.eos.plugins.module_utils.network.eos.facts.facts import ( + Facts, +) +from ansible_collections.arista.eos.plugins.module_utils.network.eos.utils.utils import ( + normalize_interface, +) + + +class Lldp_interfaces(ConfigBase): + """ + The eos_lldp_interfaces class + """ + + gather_subset = ["!all", "!min"] + + gather_network_resources = ["lldp_interfaces"] + + def get_lldp_interfaces_facts(self, data=None): + """ Get the 'facts' (the current configuration) + + :rtype: A dictionary + :returns: The current configuration as a dictionary + """ + facts, _warnings = Facts(self._module).get_facts( + self.gather_subset, self.gather_network_resources, data=data + ) + lldp_interfaces_facts = facts["ansible_network_resources"].get( + "lldp_interfaces" + ) + if not lldp_interfaces_facts: + return [] + return lldp_interfaces_facts + + def execute_module(self): + """ Execute the module + + :rtype: A dictionary + :returns: The result from module execution + """ + result = {"changed": False} + warnings = list() + commands = list() + + if self.state in self.ACTION_STATES: + existing_lldp_interfaces_facts = self.get_lldp_interfaces_facts() + else: + existing_lldp_interfaces_facts = [] + + if self.state in self.ACTION_STATES or self.state == "rendered": + commands.extend(self.set_config(existing_lldp_interfaces_facts)) + + if commands and self.state in self.ACTION_STATES: + if not self._module.check_mode: + self._connection.edit_config(commands) + result["changed"] = True + if self.state in self.ACTION_STATES: + result["commands"] = commands + + if self.state in self.ACTION_STATES or self.state == "gathered": + changed_lldp_interfaces_facts = self.get_lldp_interfaces_facts() + elif self.state == "rendered": + result["rendered"] = commands + elif self.state == "parsed": + running_config = self._module.params["running_config"] + if not running_config: + self._module.fail_json( + msg="value of running_config parameter must not be empty for state parsed" + ) + result["parsed"] = self.get_lldp_interfaces_facts( + data=running_config + ) + + if self.state in self.ACTION_STATES: + result["before"] = existing_lldp_interfaces_facts + if result["changed"]: + result["after"] = changed_lldp_interfaces_facts + + elif self.state == "gathered": + result["gathered"] = changed_lldp_interfaces_facts + + result["warnings"] = warnings + return result + + def set_config(self, existing_lldp_interfaces_facts): + """ Collect the configuration from the args passed to the module, + collect the current configuration (as a dict from facts) + + :rtype: A list + :returns: the commands necessary to migrate the current configuration + to the desired configuration + """ + want = self._module.params["config"] + have = existing_lldp_interfaces_facts + resp = self.set_state(want, have) + return to_list(resp) + + def set_state(self, want, have): + """ Select the appropriate function based on the state provided + + :param want: the desired configuration as a dictionary + :param have: the current configuration as a dictionary + :rtype: A list + :returns: the commands necessary to migrate the current configuration + to the desired configuration + """ + state = self._module.params["state"] + if ( + state in ("merged", "replaced", "overridden", "rendered") + and not want + ): + self._module.fail_json( + msg="value of config parameter must not be empty for state {0}".format( + state + ) + ) + want = param_list_to_dict(want, remove_key=False) + have = param_list_to_dict(have, remove_key=False) + if state == "overridden": + commands = self._state_overridden(want, have) + elif state == "deleted": + commands = self._state_deleted(want, have) + elif state == "merged" or state == "rendered": + commands = self._state_merged(want, have) + elif state == "replaced": + commands = self._state_replaced(want, have) + return commands + + @staticmethod + def _state_replaced(want, have): + """ The command generator when state is replaced + + :rtype: A list + :returns: the commands necessary to migrate the current configuration + to the desired configuration + """ + commands = [] + for key, desired in want.items(): + interface_name = normalize_interface(key) + if interface_name in have: + extant = have[interface_name] + else: + extant = dict(name=interface_name) + + add_config = dict_diff(extant, desired) + del_config = dict_diff(desired, extant) + + commands.extend( + generate_commands(interface_name, add_config, del_config) + ) + + return commands + + @staticmethod + def _state_overridden(want, have): + """ The command generator when state is overridden + + :rtype: A list + :returns: the commands necessary to migrate the current configuration + to the desired configuration + """ + commands = [] + for key, extant in have.items(): + if key in want: + desired = want[key] + else: + desired = dict(name=key) + + add_config = dict_diff(extant, desired) + del_config = dict_diff(desired, extant) + + commands.extend(generate_commands(key, add_config, del_config)) + + return commands + + @staticmethod + def _state_merged(want, have): + """ The command generator when state is merged + + :rtype: A list + :returns: the commands necessary to merge the provided into + the current configuration + """ + commands = [] + for key, desired in want.items(): + interface_name = normalize_interface(key) + if interface_name in have: + extant = have[interface_name] + else: + extant = dict(name=interface_name) + + add_config = dict_diff(extant, desired) + + commands.extend(generate_commands(interface_name, add_config, {})) + + return commands + + @staticmethod + def _state_deleted(want, have): + """ The command generator when state is deleted + + :rtype: A list + :returns: the commands necessary to remove the current configuration + of the provided objects + """ + commands = [] + for key in want.keys(): + interface_name = normalize_interface(key) + desired = dict(name=interface_name) + if interface_name in have: + extant = have[interface_name] + else: + continue + + del_config = dict_diff(desired, extant) + + commands.extend(generate_commands(interface_name, {}, del_config)) + + return commands + + +def generate_commands(name, to_set, to_remove): + commands = [] + for key, value in to_set.items(): + if value is None: + continue + + prefix = "" if value else "no " + commands.append("{0}lldp {1}".format(prefix, key)) + + for key in to_remove: + commands.append("lldp {0}".format(key)) + + if commands: + commands.insert(0, "interface {0}".format(name)) + + return commands diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/config/ospf_interfaces/__init__.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/config/ospf_interfaces/__init__.py new file mode 100644 index 00000000..e69de29b --- /dev/null +++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/config/ospf_interfaces/__init__.py diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/config/ospf_interfaces/ospf_interfaces.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/config/ospf_interfaces/ospf_interfaces.py new file mode 100644 index 00000000..766f0571 --- /dev/null +++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/config/ospf_interfaces/ospf_interfaces.py @@ -0,0 +1,202 @@ +# +# -*- coding: utf-8 -*- +# Copyright 2020 Red Hat +# 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 + +""" +The eos_ospf_interfaces config file. +It is in this file where the current configuration (as dict) +is compared to the provided configuration (as dict) and the command set +necessary to bring the current configuration to its desired end-state is +created. +""" +from ansible.module_utils.six import iteritems +from ansible_collections.ansible.netcommon.plugins.module_utils.network.common.utils import ( + dict_merge, +) +from ansible_collections.ansible.netcommon.plugins.module_utils.network.common.resource_module import ( + ResourceModule, +) +from ansible_collections.arista.eos.plugins.module_utils.network.eos.facts.facts import ( + Facts, +) +from ansible_collections.arista.eos.plugins.module_utils.network.eos.rm_templates.ospf_interfaces import ( + Ospf_interfacesTemplate, +) + + +class Ospf_interfaces(ResourceModule): + """ + The eos_ospf_interfaces config class + """ + + def __init__(self, module): + super(Ospf_interfaces, self).__init__( + empty_fact_val={}, + facts_module=Facts(module), + module=module, + resource="ospf_interfaces", + tmplt=Ospf_interfacesTemplate(), + ) + self.parsers = [ + "interfaces", + "area", + "authentication_v2", + "authentication_v3", + "authentication_key", + "deadinterval", + "encryption", + "hellointerval", + "bfd", + "cost", + "ip_params_area", + "ip_params_bfd", + "ip_params_cost", + "ip_params_dead_interval", + "ip_params_hello_interval", + "ip_params_mtu_ignore", + "ip_params_network", + "ip_params_priority", + "ip_params_passive_interface", + "ip_params_retransmit_interval", + "ip_params_transmit_delay", + "mtu_ignore", + "network", + "priority", + "passive_interface", + "retransmit_interval", + "transmit_delay", + "message_digest_key", + ] + + def execute_module(self): + """ Execute the module + + :rtype: A dictionary + :returns: The result from module execution + """ + if self.state not in ["parsed", "gathered"]: + self.generate_commands() + self.run_commands() + return self.result + + def generate_commands(self): + """ Generate configuration commands to send based on + want, have and desired state. + """ + + # convert list of dicts to dicts of dicts + wantd = {entry["name"]: entry for entry in self.want} + haved = {entry["name"]: entry for entry in self.have} + + # turn all lists of dicts into dicts prior to merge + for entry in wantd, haved: + self._ospf_int_list_to_dict(entry) + + # if state is merged, merge want onto have and then compare + if self.state == "merged": + wantd = dict_merge(haved, wantd) + + # if state is deleted, empty out wantd and set haved to wantd + if self.state == "deleted": + haved = { + k: v for k, v in iteritems(haved) if k in wantd or not wantd + } + for k, have in iteritems(haved): + self._compare(want={}, have=have) + wantd = {} + + # remove superfluous config for overridden + if self.state == "overridden": + for k, have in iteritems(haved): + if k not in wantd: + self._compare(want={}, have=have) + + for k, want in iteritems(wantd): + self._compare(want=want, have=haved.pop(k, {})) + + def _compare(self, want, have): + """Leverages the base class `compare()` method and + populates the list of commands to be run by comparing + the `want` and `have` data with the `parsers` defined + for the Ospf_interfaces network resource. + """ + begin = len(self.commands) + self._compare_addr_family(want=want, have=have) + if len(self.commands) != begin: + + tmp = want or have + tmp.pop("address_family", {}) + self.commands.insert( + begin, self._tmplt.render(tmp, "interfaces", False) + ) + + def _compare_addr_family(self, want, have): + wdict = want.get("address_family", {}) + hdict = have.get("address_family", {}) + for afi in ["ipv4", "ipv6"]: + w_family = wdict.pop(afi, {}) + h_family = hdict.pop(afi, {}) + for k in w_family.keys(): + if k == "afi": + continue + w = {"afi": afi, k: w_family[k]} + h = {"afi": afi, k: h_family.pop(k, {})} + if k == "ip_params": + self._compare_ip_params(want=w, have=h) + self.compare(parsers=self.parsers, want=w, have=h) + for k in h_family.keys(): + if k in ["afi"]: + continue + w = {"afi": afi, k: None} + h = {"afi": afi, k: h_family[k]} + if k == "ip_params": + w = {"afi": afi, k: {}} + self._compare_ip_params(want=w, have=h) + self.compare(parsers=self.parsers, want=w, have=h) + + def _compare_ip_params(self, want, have): + w_params = want.get("ip_params", {}) + h_params = have.get("ip_params", {}) + for afi in ["ipv4", "ipv6"]: + w_p = w_params.pop(afi, {}) + h_p = h_params.pop(afi, {}) + for k, params in iteritems(w_p): + if k == "afi": + continue + w = {"afi": afi, k: params} + h = {"afi": afi, k: h_p.pop(k, None)} + self.compare( + parsers=self.parsers, + want={"ip_params": w}, + have={"ip_params": h}, + ) + for k, params in iteritems(h_p): + if k == "afi": + continue + w = {"afi": afi, k: None} + h = {"afi": afi, k: params} + self.compare( + parsers=self.parsers, + want={"ip_params": w}, + have={"ip_params": h}, + ) + + def _ospf_int_list_to_dict(self, entry): + for name, family in iteritems(entry): + if family.get("ip_params"): + family["ip_params"] = { + entry["afi"]: entry for entry in family["ip_params"] + } + if "address_family" in family: + family["address_family"] = { + entry["afi"]: entry + for entry in family.get("address_family", []) + } + self._ospf_int_list_to_dict(family["address_family"]) diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/config/ospfv2/__init__.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/config/ospfv2/__init__.py new file mode 100644 index 00000000..e69de29b --- /dev/null +++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/config/ospfv2/__init__.py diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/config/ospfv2/ospfv2.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/config/ospfv2/ospfv2.py new file mode 100644 index 00000000..5fe8836c --- /dev/null +++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/config/ospfv2/ospfv2.py @@ -0,0 +1,796 @@ +# +# -*- coding: utf-8 -*- +# Copyright 2019 Red Hat +# GNU General Public License v3.0+ +# (see COPYING or https://www.gnu.org/licenses/gpl-3.0.txt) +""" +The eos_ospfv2 class +It is in this file where the current configuration (as dict) +is compared to the provided configuration (as dict) and the command set +necessary to bring the current configuration to it's desired end-state is +created +""" + +from __future__ import absolute_import, division, print_function + +__metaclass__ = type + +import re + +from ansible_collections.ansible.netcommon.plugins.module_utils.network.common.cfg.base import ( + ConfigBase, +) +from ansible_collections.ansible.netcommon.plugins.module_utils.network.common.utils import ( + to_list, + dict_diff, + remove_empties, +) + +from ansible_collections.arista.eos.plugins.module_utils.network.eos.facts.facts import ( + Facts, +) + + +class Ospfv2(ConfigBase): + """ + The eos_ospfv2 class + """ + + gather_subset = ["!all", "!min"] + + gather_network_resources = ["ospfv2"] + + def __init__(self, module): + super(Ospfv2, self).__init__(module) + + def get_ospfv2_facts(self, data=None): + """ Get the 'facts' (the current configuration) + + :rtype: A dictionary + :returns: The current configuration as a dictionary + """ + facts, _warnings = Facts(self._module).get_facts( + self.gather_subset, self.gather_network_resources, data=data + ) + + ospfv2_facts = facts["ansible_network_resources"].get("ospfv2") + if not ospfv2_facts: + return [] + return ospfv2_facts + + def execute_module(self): + """ Execute the module + + :rtype: A dictionary + :returns: The result from module execution + """ + result = {"changed": False} + warnings = list() + commands = list() + + if self.state in self.ACTION_STATES: + existing_ospfv2_facts = self.get_ospfv2_facts() + else: + existing_ospfv2_facts = [] + if self.state in self.ACTION_STATES or self.state == "rendered": + commands.extend(self.set_config(existing_ospfv2_facts)) + if commands and self.state in self.ACTION_STATES: + if not self._module.check_mode: + self._connection.edit_config(commands) + result["changed"] = True + if self.state in self.ACTION_STATES: + result["commands"] = commands + if self.state in self.ACTION_STATES or self.state == "gathered": + changed_ospfv2_facts = self.get_ospfv2_facts() + elif self.state == "rendered": + result["rendered"] = commands + elif self.state == "parsed": + if not self._module.params["running_config"]: + self._module.fail_json( + msg="value of running_config parameter must not be empty for state parsed" + ) + result["parsed"] = self.get_ospfv2_facts( + data=self._module.params["running_config"] + ) + else: + changed_ospfv2_facts = self.get_ospfv2_facts() + if self.state in self.ACTION_STATES: + result["before"] = existing_ospfv2_facts + if result["changed"]: + result["after"] = changed_ospfv2_facts + elif self.state == "gathered": + result["gathered"] = changed_ospfv2_facts + + result["warnings"] = warnings + return result + + def set_config(self, existing_ospfv2_facts): + """ Collect the configuration from the args passed to the module, + collect the current configuration (as a dict from facts) + + :rtype: A list + :returns: the commands necessary to migrate the current configuration + to the desired configuration + """ + want = self._module.params["config"] + have = existing_ospfv2_facts + resp = self.set_state(want, have) + return to_list(resp) + + def set_state(self, want, have): + """ Select the appropriate function based on the state provided + + :param want: the desired configuration as a dictionary + :param have: the current configuration as a dictionary + :rtype: A list + :returns: the commands necessary to migrate the current configuration + to the desired configuration + """ + commands = [] + if ( + self.state in ("merged", "replaced", "overridden", "rendered") + and not want + ): + self._module.fail_json( + msg="value of config parameter must not be empty for state {0}".format( + self.state + ) + ) + if self.state == "overridden": + commands = self._state_overridden(want, have) + elif self.state == "deleted": + commands = self._state_deleted(want, have) + elif self.state == "merged" or self.state == "rendered": + commands = self._state_merged(want, have) + elif self.state == "replaced": + commands = self._state_replaced(want, have) + return commands + + def _state_replaced(self, want, have): + """ The command generator when state is replaced + + :rtype: A list + :returns: the commands necessary to migrate the current configuration + to the desired configuration + """ + commands = [] + for w in want["processes"]: + del_cmds = w.copy() + add_cmds = {} + for h in have["processes"]: + if h["process_id"] != w["process_id"]: + continue + if w.get("vrf"): + if w["vrf"] != h["vrf"]: + self._module.fail_json( + msg="Value of vrf and process_id does not match the config present in the device" + ) + break + del_instance_list = self.compare_dicts(h, w) + if del_instance_list: + del_cmds = {"processes": del_instance_list} + add_instance_list = self.compare_dicts(w, h) + if add_instance_list: + add_cmds = {"processes": add_instance_list} + + return_command = self.del_commands(del_cmds, have) + for command in return_command: + if "exit" not in command: + commands.append(command) + return_command = self.add_commands(add_cmds, have) + for command in return_command: + if "router ospf" in command: + if command not in commands: + commands.append(command) + else: + commands.append(command) + commandset = [] + if commands: + commandset.append(commands[0]) + for cmd in commands[1::]: + if "router ospf" in cmd and commandset[-1] != "exit": + commandset.append("exit") + commandset.append(cmd) + if commandset[-1] != "exit": + commandset.append("exit") + return commandset + + def _state_overridden(self, want, have): + """ The command generator when state is overridden + + :rtype: A list + :returns: the commands necessary to migrate the current configuration + to the desired configuration + """ + commands = [] + for h in have["processes"]: + present = False + for w in want["processes"]: + if h["process_id"] == w["process_id"]: + present = True + break + if not present: + commands.append("no router ospf " + str(h["process_id"])) + replace_cmds = self._state_replaced(want, have) + for cmd in replace_cmds: + commands.append(cmd) + return commands + + def _state_merged(self, want, have): + """ The command generator when state is merged + + :rtype: A list + :returns: the commands necessary to merge the provided into + the current configuration + """ + + return self.set_commands(want, have) + + def _state_deleted(self, want, have): + """ The command generator when state is deleted + + :rtype: A list + :returns: the commands necessary to remove the current configuration + of the provided objects + """ + commands = [] + return_command = self.del_commands(want, have) + if return_command: + for cmd in return_command: + if "no exit" in cmd: + cmd = "exit" + commands.append(cmd) + return commands + + def set_commands(self, want, have): + commands = [] + instance_list = [] + for w in want["processes"]: + present = False + c = [] + if have and not have.get("processes"): + instance_list = want["processes"] + break + if have: + for h in have["processes"]: + if w["process_id"] == h["process_id"]: + if w.get("vrf"): + if w["vrf"] != h["vrf"]: + self._module.fail_json( + msg="Value of vrf and process_id does not match the config present in the device" + ) + continue + present = True + c = self.compare_dicts(w, h) + break + if c: + instance_list.append(c[0]) + if not present: + if w["vrf"] in _get_vrf_list(have): + self._module.fail_json( + msg="Value of vrf and process_id does not match the config present in the device" + ) + instance_list.append(w) + instance_dict = {"processes": instance_list} + return_command = self.add_commands(instance_dict, have) + for command in return_command: + commands.append(command) + return commands + + def compare_dicts(self, want_inst, have_inst): + want_dict = remove_empties(want_inst) + have = have_inst + ospf_list = [] + return_ospf_dict = {} + for w_key in want_dict.keys(): + if not have.get(w_key): + return_ospf_dict.update({w_key: want_dict[w_key]}) + elif ( + isinstance(want_dict[w_key], str) + or isinstance(want_dict[w_key], bool) + or isinstance(want_dict[w_key], int) + ): + if want_dict[w_key] != have[w_key]: + return_ospf_dict.update({w_key: want_dict[w_key]}) + elif isinstance(want_dict[w_key], dict): + diff = dict_diff(have.get(w_key, {}), want_dict[w_key]) + if diff: + return_ospf_dict.update({w_key: diff}) + elif isinstance(want_dict[w_key], list): + if have.get(w_key): + compare_list = self.compare_ospf_list( + want_dict[w_key], have.get(w_key), w_key + ) + if compare_list: + return_ospf_dict.update({w_key: compare_list}) + else: + if want_dict[w_key] != have.get(w_key): + return_ospf_dict.update({w_key: want_dict[w_key]}) + + if return_ospf_dict: + if want_dict.get("vrf"): + return_ospf_dict.update( + { + "process_id": want_dict["process_id"], + "vrf": want_dict["vrf"], + } + ) + else: + return_ospf_dict.update( + {"process_id": want_dict["process_id"]} + ) + ospf_list.append(return_ospf_dict) + return ospf_list + + def compare_ospf_list(self, w_list, h_list, l_key): + return_list = [] + for w in w_list: + present = False + for h in h_list: + diff = dict_diff(h, w) + if not diff: + present = True + break + if not present: + return_list.append(w) + return return_list + + def add_commands(self, want, have): + commands = [] + if not want: + return commands + for ospf_params in want["processes"]: + commands.append(_get_router_command(ospf_params)) + if ospf_params.get("traffic_engineering"): + commands.append("traffic-engineering") + if ospf_params.get("adjacency"): + threshold = ospf_params["adjacency"]["exchange_start"][ + "threshold" + ] + commands.append( + "adjacency exchange-start threshold " + str(threshold) + ) + if ospf_params.get("areas"): + command_list = _parse_areas(ospf_params["areas"]) + for c in command_list: + commands.append(c) + if ospf_params.get("auto_cost"): + commands.append( + "auto-cose reference-bandwidth " + ospf_params["auto_cost"] + ) + if ospf_params.get("bfd"): + commands.append("bfd default") + if ospf_params.get("default_information"): + commands.append( + _parse_default_information( + ospf_params["default_information"] + ) + ) + if ospf_params.get("default_metric"): + commands.append( + "default-metric" + " " + str(ospf_params["default_metric"]) + ) + if ospf_params.get("distance"): + for k, v in ospf_params["distance"].items(): + if v: + k = re.sub(r"_", "-", k) + commands.append("distance ospf " + k + " " + str(v)) + if ospf_params.get("distribute_list"): + commands.append( + "distribute-list " + + ospf_params["distribute_list"].keys()[0] + + " " + + ospf_params["distribute_list"].values()[0] + + " in" + ) + if ospf_params.get("dn_bit_ignore"): + commands.append("dn-bit-ignore") + if ospf_params.get("graceful_restart"): + if ospf_params["graceful_restart"].get("set"): + commands.append("graceful-restart") + else: + commands.append( + "graceful-restart grace-period " + + str( + ospf_params["graceful_restart"].get("grace_period") + ) + ) + if ospf_params.get("graceful_restart_helper"): + commands.append("graceful-restart-helper") + if ospf_params.get("log_adjacency_changes"): + cmd = "log-adjacency-changes" + if ospf_params["log_adjacency_changes"].get("detail"): + cmd = cmd + " detail" + commands.append(cmd) + if ospf_params.get("max_lsa"): + commands.append(_parse_max_lsa(ospf_params["max_lsa"])) + if ospf_params.get("max_metric"): + commands.append(_parse_max_metric(ospf_params["max_metric"])) + if ospf_params.get("maximum_paths"): + commands.append( + "maximum-paths " + str(ospf_params["maximum_paths"]) + ) + if ospf_params.get("mpls_ldp"): + commands.append("mpls ldp sync default") + if ospf_params.get("networks"): + command_list = _parse_networks(ospf_params["networks"]) + for c in command_list: + commands.append(c) + if ospf_params.get("passive_interface"): + if "interface_list" in ospf_params["passive_interface"].keys(): + commands.append( + "passive-interface " + + ospf_params["passive_interface"]["interface_list"] + ) + else: + commands.append("passive-interface default") + if ospf_params.get("point_to_point"): + commands.append("point-to-point routes") + if ospf_params.get("redistribute"): + command_list = _parse_redistribute(ospf_params["redistribute"]) + for c in command_list: + commands.append(c) + if ospf_params.get("retransmission_threshold"): + commands.append( + "retransmission-threshold lsa " + + str(ospf_params["retransmission_threshold"]) + ) + if ospf_params.get("rfc1583compatibility"): + commands.append("compatible rfc1583") + if ospf_params.get("router_id"): + commands.append("router-id " + ospf_params.get("router_id")) + if ospf_params.get("summary_address"): + commands.append( + _parse_summary_address(ospf_params["summary_address"]) + ) + if ospf_params.get("timers"): + command_list = _parse_timers(ospf_params["timers"]) + for c in command_list: + commands.append(c) + commands.append("exit") + commandset = [] + for command in commands: + commandset.append(command.strip()) + return commandset + + def del_commands(self, want, have): + commands = [] + other_commands = 0 + want = remove_empties(want) + if want.get("processes"): + for w_inst in want["processes"]: + router_context = 0 + d_cmds = [] + instance_list = [] + if have.get("processes"): + for h_inst in have["processes"]: + if h_inst["process_id"] == w_inst["process_id"]: + if w_inst.get("vrf") and w_inst.get( + "vrf" + ) == h_inst.get("vrf"): + if list(w_inst.keys()) == [ + "process_id", + "vrf", + ]: + commands.append( + "no router ospf " + + str(w_inst["process_id"]) + + " vrf " + + w_inst["vrf"] + ) + router_context = 1 + if len(w_inst.keys()) == 1 and list( + w_inst.keys() + ) == ["process_id"]: + commands.append( + "no router ospf " + + str(w_inst["process_id"]) + ) + router_context = 1 + if not router_context: + instance_list = self.compare_dicts( + w_inst, h_inst + ) + if not instance_list: + del_want = {"processes": [w_inst]} + d_cmds = self.add_commands(del_want, have) + for cmd in d_cmds: + if "router ospf" in cmd: + other_commands = 0 + if cmd not in commands: + commands.append(cmd) + else: + cmd = "no " + cmd + if cmd not in commands: + commands.append(cmd) + other_commands += 1 + if ( + not other_commands + and len(commands) == 1 + and not router_context + ): + if ( + "no" not in commands[0] + and "router ospf" in commands[0] + ): + commands[0] = "no " + commands[0] + return commands + + +def _get_router_command(inst): + command = "" + if inst.get("vrf") and inst.get("vrf") != "default": + command = ( + "router ospf " + str(inst["process_id"]) + " vrf " + inst["vrf"] + ) + else: + command = "router ospf " + str(inst["process_id"]) + return command + + +def _get_vrf_list(want): + vrf_list = [] + if not want: + return vrf_list + for w in want["processes"]: + if w.get("vrf"): + vrf_list.append(w["vrf"]) + return vrf_list + + +def _parse_areas(areas): + command = [] + for area in areas: + area_cmd = "area " + area["area_id"] + if area.get("default_cost"): + command.append( + area_cmd + " default-cost " + str(area.get("default_cost")) + ) + elif area.get("filter"): + command.append( + area_cmd + " " + _parse_areas_filter(area["filter"]) + ) + elif area.get("not_so_stubby"): + command.append( + area_cmd + + " " + + _parse_areas_filter_notsostubby(area["not_so_stubby"]) + ) + elif area.get("nssa"): + command.append( + area_cmd + " " + _parse_areas_filter_nssa(area["nssa"]) + ) + elif area.get("range"): + command.append(area_cmd + " " + _parse_areas_range(area["range"])) + return command + + +def _parse_areas_filter(filter_dict): + filter_cmd = "filter " + if filter_dict.get("prefix_list"): + filter_cmd = filter_cmd + filter_dict.get("filter") + elif filter_dict.get("address"): + filter_cmd = filter_cmd + filter_dict.get("address") + else: + filter_cmd = ( + filter_cmd + + filter_dict.get("subnet_address") + + " " + + filter_dict.get("subnet_mask") + ) + return filter_cmd + + +def _parse_areas_filter_notsostubby(nss_dict): + nss_cmd = "not-so-stubby " + if nss_dict.get("default_information_originate"): + nss_cmd = nss_cmd + "default-information-originate " + for def_keys in nss_dict["default_information_originate"].keys(): + if ( + def_keys == "nssa_only" + and nss_dict["default_information_originate"]["nssa_only"] + ): + nss_cmd = nss_cmd + " nssa-only " + elif nss_dict["default_information_originate"].get(def_keys): + nss_cmd = ( + nss_cmd + + def_keys + + " " + + nss_dict["default_information_originate"][def_keys] + ) + elif "lsa" in nss_dict.keys() and nss_dict.get("lsa"): + nss_cmd = nss_cmd + " lsa type-7 convert type-5" + elif "no_summary" in nss_dict.keys() and nss_dict.get("no_summary"): + nss_cmd = nss_cmd + " no-summary" + elif "nssa_only" in nss_dict.keys() and nss_dict.get("nssa_only"): + nss_cmd = nss_cmd + " nssa-only" + return nss_cmd + + +def _parse_areas_filter_nssa(nss_dict): + nss_cmd = "nssa " + if nss_dict.get("default_information_originate"): + nss_cmd = nss_cmd + "default-information-originate " + for def_keys in nss_dict["default_information_originate"].keys(): + if ( + def_keys == "nssa_only" + and nss_dict["default_information_originate"]["nssa_only"] + ): + nss_cmd = nss_cmd + " nssa-only " + elif nss_dict["default_information_originate"].get(def_keys): + nss_cmd = ( + nss_cmd + + def_keys + + " " + + nss_dict["default_information_originate"][def_keys] + ) + elif "no_summary" in nss_dict.keys() and nss_dict.get("no_summary"): + nss_cmd = nss_cmd + " no-summary" + elif "nssa_only" in nss_dict.keys() and nss_dict.get("nssa_only"): + nss_cmd = nss_cmd + " nssa-only" + return nss_cmd + + +def _parse_areas_range(range_dict): + range_cmd = " range " + if range_dict.get("address"): + range_cmd = range_cmd + range_dict["address"] + if range_dict.get("subnet_address"): + range_cmd = ( + range_cmd + + range_dict["subnet_address"] + + " " + + range_dict["subnet_mask"] + ) + if range_dict.get("advertise") is not None: + if range_dict["advertise"]: + range_cmd = range_cmd + " advertise " + else: + range_cmd = range_cmd + " not-advertise " + if range_dict.get("cost"): + range_cmd = range_cmd + " cost " + str(range_dict["cost"]) + return range_cmd + + +def _parse_default_information(default_dict): + def_cmd = "default-information originate" + for def_key in sorted(default_dict.keys()): + if def_key == "always": + if default_dict.get(def_key): + def_cmd = def_cmd + " " + def_key + elif def_key in ["metric", "metric_type", "route_map"]: + if default_dict.get(def_key): + k = re.sub(r"_", "-", def_key) + def_cmd = def_cmd + " " + k + " " + str(default_dict[def_key]) + return def_cmd + + +def _parse_max_lsa(max_lsa_dict): + max_lsa_cmd = "max-lsa " + if max_lsa_dict.get("count"): + max_lsa_cmd = max_lsa_cmd + " " + str(max_lsa_dict["count"]) + if max_lsa_dict.get("threshold"): + max_lsa_cmd = max_lsa_cmd + " " + str(max_lsa_dict["threshold"]) + for lsa_key, lsa_val in sorted(max_lsa_dict.items()): + if lsa_key == "warning" and lsa_val: + max_lsa_cmd = max_lsa_cmd + " warning-only" + elif lsa_key in ["ignore_count", "reset_time", "ignore_time"]: + if lsa_val: + k = re.sub(r"_", "-", lsa_key) + max_lsa_cmd = max_lsa_cmd + " " + k + " " + str(lsa_val) + " " + return max_lsa_cmd + + +def _parse_max_metric(max_metric_dict): + metric_cmd = "max-metric router-lsa " + for k, v in max_metric_dict["router_lsa"].items(): + if not v: + continue + if k == "include_stub" and v: + metric_cmd = metric_cmd + " include-stub" + elif k == "on_startup": + metric_cmd = metric_cmd + " on-startup " + str(v["wait_period"]) + elif k in ["summary_lsa", "external_lsa"]: + k = re.sub(r"_", "-", k) + if v.get("set"): + metric_cmd = metric_cmd + " " + k + else: + metric_cmd = ( + metric_cmd + " " + k + " " + str(v.get("max_metric_value")) + ) + return metric_cmd + + +def _parse_networks(net_list): + network_cmd = [] + for net_dict in net_list: + net_cmd = "network " + if net_dict.get("prefix"): + net_cmd = net_cmd + net_dict.get("prefix") + else: + net_cmd = ( + net_cmd + + net_dict.get("network_address") + + " " + + net_dict.get("mask") + ) + if net_dict.get("area"): + net_cmd = net_cmd + " area " + net_dict.get("area") + network_cmd.append(net_cmd) + return network_cmd + + +def _parse_redistribute(r_list): + rcmd_list = [] + for r_dict in r_list: + r_cmd = "redistribute " + r_cmd = r_cmd + r_dict["routes"] + if r_dict.get("isis_level"): + k = re.sub(r"_", "-", r_dict["isis_level"]) + r_cmd = r_cmd + " " + k + if r_dict.get("route_map"): + r_cmd = r_cmd + " route-map " + r_dict["route_map"] + rcmd_list.append(r_cmd) + return rcmd_list + + +def _parse_summary_address(addr_dict): + sum_cmd = "summary-address " + if addr_dict.get("prefix"): + sum_cmd = sum_cmd + addr_dict.get("prefix") + else: + sum_cmd = ( + sum_cmd + addr_dict.get("address") + " " + addr_dict.get("mask") + ) + if "attribute_map" in addr_dict.keys(): + sum_cmd = sum_cmd + " attribute-map " + addr_dict["attribute_map"] + elif addr_dict.get("not_advertise"): + sum_cmd = sum_cmd + " not-advertise " + elif "tag" in addr_dict.keys(): + sum_cmd = sum_cmd + " tag " + addr_dict["tag"] + return sum_cmd + + +def _parse_timers(timers_list): + timers_cmd = [] + for t_dict in timers_list: + t_cmd = "timers " + for t_key in t_dict.keys(): + if not t_dict.get(t_key): + break + if t_key == "lsa": + if t_dict["lsa"].get("rx"): + t_cmd = ( + t_cmd + + "lsa rx min interval " + + str(t_dict["lsa"]["rx"]["min_interval"]) + ) + else: + t_cmd = ( + t_cmd + + "lsa tx delay initial " + + str(t_dict["lsa"]["tx"]["delay"]["initial"]) + + " " + + str(t_dict["lsa"]["tx"]["delay"]["min"]) + + " " + + str(t_dict["lsa"]["tx"]["delay"]["max"]) + ) + elif t_key == "out_delay": + t_cmd = t_cmd + " out-delay " + str(t_dict["out_delay"]) + elif t_key == "pacing": + t_cmd = t_cmd + " pacing flood " + str(t_dict["pacing"]) + elif t_key == "spf": + if "seconds" in t_dict["spf"].keys(): + t_cmd = t_cmd + " spf " + str(t_dict["spf"]["seconds"]) + else: + t_cmd = ( + t_cmd + + " spf delay initial " + + str(t_dict["spf"]["initial"]) + + " " + + str(t_dict["spf"]["max"]) + + " " + + str(t_dict["spf"]["min"]) + ) + timers_cmd.append(t_cmd) + return timers_cmd diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/config/ospfv3/__init__.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/config/ospfv3/__init__.py new file mode 100644 index 00000000..e69de29b --- /dev/null +++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/config/ospfv3/__init__.py diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/config/ospfv3/ospfv3.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/config/ospfv3/ospfv3.py new file mode 100644 index 00000000..a0430737 --- /dev/null +++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/config/ospfv3/ospfv3.py @@ -0,0 +1,285 @@ +# +# -*- coding: utf-8 -*- +# Copyright 2020 Red Hat +# 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 + +""" +The eos_ospfv3 config file. +It is in this file where the current configuration (as dict) +is compared to the provided configuration (as dict) and the command set +necessary to bring the current configuration to its desired end-state is +created. +""" + +import re +from ansible.module_utils.six import iteritems +from ansible_collections.ansible.netcommon.plugins.module_utils.network.common.utils import ( + dict_merge, +) +from ansible_collections.ansible.netcommon.plugins.module_utils.network.common.resource_module import ( + ResourceModule, +) +from ansible_collections.arista.eos.plugins.module_utils.network.eos.facts.facts import ( + Facts, +) +from ansible_collections.arista.eos.plugins.module_utils.network.eos.rm_templates.ospfv3 import ( + Ospfv3Template, +) +from ansible_collections.ansible.netcommon.plugins.module_utils.network.common.utils import ( + get_from_dict, +) + + +class Ospfv3(ResourceModule): + """ + The eos_ospfv3 config class + """ + + def __init__(self, module): + super(Ospfv3, self).__init__( + empty_fact_val={}, + facts_module=Facts(module), + module=module, + resource="ospfv3", + tmplt=Ospfv3Template(), + ) + self.parsers = [ + "vrf", + "address_family", + "adjacency", + "auto_cost", + "area.default_cost", + "area.authentication", + "area.encryption", + "area.nssa", + "area.ranges", + "area.stub", + "bfd", + "default_information", + "default_metric", + "distance", + "fips_restrictions", + "graceful_restart", + "graceful_restart_period", + "graceful_restart_helper", + "log_adjacency_changes", + "max_metric", + "maximum_paths", + "passive_interface", + "redistribute", + "router_id", + "shutdown", + "timers.lsa", + "timers.out_delay", + "timers.pacing", + "timers.throttle.lsa", + "timers.throttle.spf", + ] + + def execute_module(self): + """ Execute the module + + :rtype: A dictionary + :returns: The result from module execution + """ + if self.state not in ["parsed", "gathered"]: + self.generate_commands() + self.run_commands() + return self.result + + def generate_commands(self): + """ Generate configuration commands to send based on + want, have and desired state. + """ + + wantd = { + entry["vrf"]: entry for entry in self.want.get("processes", []) + } + haved = { + entry["vrf"]: entry for entry in self.have.get("processes", []) + } + + # turn all lists of dicts into dicts prior to merge + for entry in wantd, haved: + self._ospf_list_to_dict(entry) + # if state is merged, merge want onto have and then compare + if self.state == "merged": + wantd = dict_merge(haved, wantd) + + # if state is deleted, empty out wantd and set haved to wantd + if self.state == "deleted": + haved = { + k: v for k, v in iteritems(haved) if k in wantd or not wantd + } + wantd = {} + + # remove superfluous config for overridden and deleted + if self.state in ["overridden", "deleted"]: + for k, have in iteritems(haved): + if k not in wantd and have.get("vrf") == k: + self.commands.append(self._tmplt.render(have, "vrf", True)) + + for k, want in iteritems(wantd): + self._compare(want=want, have=haved.pop(k, {})) + + def _compare(self, want, have): + """Leverages the base class `compare()` method and + populates the list of commands to be run by comparing + the `want` and `have` data with the `parsers` defined + for the Ospfv3 network resource. + """ + begin = len(self.commands) + self._af_compare(want=want, have=have) + self._global_compare(want=want, have=have) + + if len(self.commands) != begin or (not have and want): + self.commands.insert( + begin, self._tmplt.render(want or have, "vrf", False) + ) + self.commands.append("exit") + + def _global_compare(self, want, have): + for name, entry in iteritems(want): + if name in ["vrf", "address_family"]: + continue + if not isinstance(entry, dict) and name != "areas": + self.compare( + parsers=self.parsers, + want={name: entry}, + have={name: have.pop(name, None)}, + ) + else: + if name == "areas" and entry: + self._areas_compare( + want={name: entry}, have={name: have.get(name, {})} + ) + else: + # passing dict without vrf, inorder to avoid no router ospfv3 command + h = {i: have[i] for i in have if i != "vrf"} + self.compare( + parsers=self.parsers, + want={name: entry}, + have={name: h.pop(name, {})}, + ) + # remove remaining items in have for replaced + for name, entry in iteritems(have): + if name in ["vrf", "address_family"]: + continue + if not isinstance(entry, dict): + self.compare( + parsers=self.parsers, + want={name: want.pop(name, None)}, + have={name: entry}, + ) + else: + # passing dict without vrf, inorder to avoid no router ospfv3 command + # w = {i: want[i] for i in want if i != "vrf"} + self.compare( + parsers=self.parsers, + want={name: want.pop(name, {})}, + have={name: entry}, + ) + + def _af_compare(self, want, have): + wafs = want.get("address_family", {}) + hafs = have.get("address_family", {}) + for name, entry in iteritems(wafs): + begin = len(self.commands) + self._compare_lists(want=entry, have=hafs.get(name, {})) + self._areas_compare(want=entry, have=hafs.get(name, {})) + self.compare( + parsers=self.parsers, want=entry, have=hafs.pop(name, {}) + ) + if ( + len(self.commands) != begin + and "afi" in entry + and entry["afi"] != "router" + ): + self._rotate_commands(begin=begin) + self.commands.insert( + begin, self._tmplt.render(entry, "address_family", False) + ) + self.commands.append("exit") + for name, entry in iteritems(hafs): + self.addcmd(entry, "address_family", True) + + def _rotate_commands(self, begin=0): + # move negate commands to beginning + for cmd in self.commands[begin::]: + negate = re.match(r"^no .*", cmd) + if negate: + self.commands.insert( + begin, self.commands.pop(self.commands.index(cmd)) + ) + begin += 1 + + def _areas_compare(self, want, have): + wareas = want.get("areas", {}) + hareas = have.get("areas", {}) + for name, entry in iteritems(wareas): + self._area_compare(want=entry, have=hareas.pop(name, {})) + for name, entry in iteritems(hareas): + self._area_compare(want={}, have=entry) + + def _area_compare(self, want, have): + parsers = [ + "area.default_cost", + "area.encryption", + "area.authentication", + "area.nssa", + "area.stub", + ] + self.compare(parsers=parsers, want=want, have=have) + self._area_compare_lists(want=want, have=have) + + def _area_compare_lists(self, want, have): + for attrib in ["ranges"]: + wdict = want.get(attrib, {}) + hdict = have.get(attrib, {}) + for key, entry in iteritems(wdict): + if entry != hdict.pop(key, {}): + entry["area_id"] = want["area_id"] + self.addcmd(entry, "area.{0}".format(attrib), False) + # remove remaining items in have for replaced + for entry in hdict.values(): + entry["area_id"] = have["area_id"] + self.addcmd(entry, "area.{0}".format(attrib), True) + + def _compare_lists(self, want, have): + for attrib in ["redistribute"]: + wdict = get_from_dict(want, attrib) or {} + hdict = get_from_dict(have, attrib) or {} + for key, entry in iteritems(wdict): + if entry != hdict.pop(key, {}): + self.addcmd(entry, attrib, False) + # remove remaining items in have for replaced + for entry in hdict.values(): + self.addcmd(entry, attrib, True) + + def _ospf_list_to_dict(self, entry): + for name, proc in iteritems(entry): + for area in proc.get("areas", []): + if "ranges" in area: + area["ranges"] = { + entry["address"]: entry + for entry in area.get("ranges", []) + } + proc["areas"] = { + entry["area_id"]: entry for entry in proc.get("areas", []) + } + proc["redistribute"] = { + entry["routes"]: entry + for entry in proc.get("redistribute", []) + } + if "address_family" in proc: + proc["address_family"] = { + entry["afi"]: entry + for entry in proc.get("address_family", []) + } + self._ospf_list_to_dict(proc["address_family"]) diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/config/static_routes/__init__.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/config/static_routes/__init__.py new file mode 100644 index 00000000..e69de29b --- /dev/null +++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/config/static_routes/__init__.py diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/config/static_routes/static_routes.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/config/static_routes/static_routes.py new file mode 100644 index 00000000..5dd28b11 --- /dev/null +++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/config/static_routes/static_routes.py @@ -0,0 +1,364 @@ +# +# -*- coding: utf-8 -*- +# Copyright 2019 Red Hat +# GNU General Public License v3.0+ +# (see COPYING or https://www.gnu.org/licenses/gpl-3.0.txt) +""" +The eos_static_routes class +It is in this file where the current configuration (as dict) +is compared to the provided configuration (as dict) and the command set +necessary to bring the current configuration to it's desired end-state is +created +""" + +from __future__ import absolute_import, division, print_function + +__metaclass__ = type + +import re +from ansible_collections.ansible.netcommon.plugins.module_utils.network.common.cfg.base import ( + ConfigBase, +) +from ansible_collections.ansible.netcommon.plugins.module_utils.network.common.utils import ( + remove_empties, +) +from ansible_collections.arista.eos.plugins.module_utils.network.eos.facts.facts import ( + Facts, +) + + +class Static_routes(ConfigBase): + """ + The eos_static_routes class + """ + + gather_subset = ["!all", "!min"] + + gather_network_resources = ["static_routes"] + + def __init__(self, module): + super(Static_routes, self).__init__(module) + + def get_static_routes_facts(self, data=None): + """ Get the 'facts' (the current configuration) + + :rtype: A dictionary + :returns: The current configuration as a dictionary + """ + facts, _warnings = Facts(self._module).get_facts( + self.gather_subset, self.gather_network_resources, data=data + ) + static_routes_facts = facts["ansible_network_resources"].get( + "static_routes" + ) + if not static_routes_facts: + return [] + return static_routes_facts + + def execute_module(self): + """ Execute the module + + :rtype: A dictionary + :returns: The result from module execution + """ + result = {"changed": False} + warnings = list() + commands = list() + if self.state in self.ACTION_STATES: + existing_static_routes_facts = self.get_static_routes_facts() + else: + existing_static_routes_facts = [] + + if self.state in self.ACTION_STATES or self.state == "rendered": + commands.extend(self.set_config(existing_static_routes_facts)) + + if commands and self.state in self.ACTION_STATES: + if not self._module.check_mode: + self._connection.edit_config(commands) + result["changed"] = True + if self.state in self.ACTION_STATES: + result["commands"] = commands + if self.state in self.ACTION_STATES or self.state == "gathered": + changed_static_routes_facts = self.get_static_routes_facts() + elif self.state == "rendered": + result["rendered"] = commands + elif self.state == "parsed": + if not self._module.params["running_config"]: + self._module.fail_json( + msg="Value of running_config parameter must not be empty for state parsed" + ) + result["parsed"] = self.get_static_routes_facts( + data=self._module.params["running_config"] + ) + else: + changed_static_routes_facts = [] + + if self.state in self.ACTION_STATES: + result["before"] = existing_static_routes_facts + if result["changed"]: + result["after"] = changed_static_routes_facts + elif self.state == "gathered": + result["gathered"] = changed_static_routes_facts + + result["warnings"] = warnings + return result + + def set_config(self, existing_static_routes_facts): + """ Collect the configuration from the args passed to the module, + collect the current configuration (as a dict from facts) + + :rtype: A list + :returns: the commands necessary to migrate the current configuration + to the desired configuration + """ + commands = [] + onbox_configs = [] + for h in existing_static_routes_facts: + return_command = add_commands(h) + for command in return_command: + onbox_configs.append(command) + config = self._module.params.get("config") + want = [] + if config: + for w in config: + want.append(remove_empties(w)) + have = existing_static_routes_facts + resp = self.set_state(want, have) + for want_config in resp: + if want_config not in onbox_configs: + commands.append(want_config) + return commands + + def set_state(self, want, have): + """ Select the appropriate function based on the state provided + + :param want: the desired configuration as a dictionary + :param have: the current configuration as a dictionary + :rtype: A list + :returns: the commands necessary to migrate the current configuration + to the desired configuration + """ + commands = [] + if self.state in ("merged", "replaced", "overridden") and not want: + self._module.fail_json( + msg="value of config parameter must not be empty for state {0}".format( + self.state + ) + ) + state = self._module.params["state"] + if state == "overridden": + commands = self._state_overridden(want, have) + elif state == "deleted": + commands = self._state_deleted(want, have) + elif state == "merged" or self.state == "rendered": + commands = self._state_merged(want, have) + elif state == "replaced": + commands = self._state_replaced(want, have) + return commands + + @staticmethod + def _state_replaced(want, have): + """ The command generator when state is replaced + + :rtype: A list + :returns: the commands necessary to migrate the current configuration + to the desired configuration + """ + commands = [] + haveconfigs = [] + vrf = get_vrf(want) + dest = get_dest(want) + for h in have: + return_command = add_commands(h) + for command in return_command: + for d in dest: + if d in command: + if vrf is None: + if "vrf" not in command: + haveconfigs.append(command) + else: + if vrf in command: + haveconfigs.append(command) + wantconfigs = set_commands(want, have) + + removeconfigs = list(set(haveconfigs) - set(wantconfigs)) + for command in removeconfigs: + commands.append("no " + command) + for wantcmd in wantconfigs: + commands.append(wantcmd) + return commands + + @staticmethod + def _state_overridden(want, have): + """ The command generator when state is overridden + + :rtype: A list + :returns: the commands necessary to migrate the current configuration + to the desired configuration + """ + commands = [] + haveconfigs = [] + for h in have: + return_command = add_commands(h) + for command in return_command: + haveconfigs.append(command) + wantconfigs = set_commands(want, have) + idempotentconfigs = list(set(haveconfigs) - set(wantconfigs)) + if not idempotentconfigs: + return idempotentconfigs + removeconfigs = list(set(haveconfigs) - set(wantconfigs)) + for command in removeconfigs: + commands.append("no " + command) + for wantcmd in wantconfigs: + commands.append(wantcmd) + return commands + + @staticmethod + def _state_merged(want, have): + """ The command generator when state is merged + + :rtype: A list + :returns: the commands necessary to merge the provided into + the current configuration + """ + return set_commands(want, have) + + @staticmethod + def _state_deleted(want, have): + """ The command generator when state is deleted + + :rtype: A list + :returns: the commands necessary to remove the current configuration + of the provided objects + """ + commands = [] + if not want: + for h in have: + return_command = add_commands(h) + for command in return_command: + command = "no " + command + commands.append(command) + else: + for w in want: + return_command = del_commands(w, have) + for command in return_command: + commands.append(command) + return commands + + +def set_commands(want, have): + commands = [] + for w in want: + return_command = add_commands(w) + for command in return_command: + commands.append(command) + return commands + + +def add_commands(want): + commandset = [] + if not want: + return commandset + vrf = ( + want["vrf"] + if "vrf" in want.keys() and want["vrf"] is not None + else None + ) + for address_family in want["address_families"]: + for route in address_family["routes"]: + for next_hop in route["next_hops"]: + commands = [] + if address_family["afi"] == "ipv4": + commands.append("ip route") + else: + commands.append("ipv6 route") + if vrf: + commands.append(" vrf " + vrf) + if not re.search(r"/", route["dest"]): + mask = route["dest"].split()[1] + cidr = get_net_size(mask) + commands.append( + " " + route["dest"].split()[0] + "/" + cidr + ) + else: + commands.append(" " + route["dest"]) + if "interface" in next_hop.keys(): + commands.append(" " + next_hop["interface"]) + if "nexthop_grp" in next_hop.keys(): + commands.append( + " Nexthop-Group" + " " + next_hop["nexthop_grp"] + ) + if "forward_router_address" in next_hop.keys(): + commands.append(" " + next_hop["forward_router_address"]) + if "mpls_label" in next_hop.keys(): + commands.append(" label " + str(next_hop["mpls_label"])) + if "track" in next_hop.keys(): + commands.append(" track " + next_hop["track"]) + if "admin_distance" in next_hop.keys(): + commands.append(" " + str(next_hop["admin_distance"])) + if "description" in next_hop.keys(): + commands.append(" name " + str(next_hop["description"])) + if "tag" in next_hop.keys(): + commands.append(" tag " + str(next_hop["tag"])) + + config_commands = "".join(commands) + commandset.append(config_commands) + return commandset + + +def del_commands(want, have): + commandset = [] + haveconfigs = [] + for h in have: + return_command = add_commands(h) + for command in return_command: + command = "no " + command + haveconfigs.append(command) + if want is None or "address_families" not in want.keys(): + commandset = haveconfigs + if "address_families" not in want.keys() and "vrf" in want.keys(): + commandset = [] + for command in haveconfigs: + if want["vrf"] in command: + commandset.append(command) + elif ( + want is not None + and "vrf" not in want.keys() + and "address_families" not in want.keys() + ): + commandset = [] + for command in haveconfigs: + if "vrf" not in command: + commandset.append(command) + + elif want["address_families"]: + for address_family in want["address_families"]: + for command in haveconfigs: + afi = "ip " if address_family["afi"] == "ipv4" else "ipv6" + if afi in command: + commandset.append(command) + return commandset + + +def get_net_size(netmask): + binary_str = "" + netmask = netmask.split(".") + for octet in netmask: + binary_str += bin(int(octet))[2:].zfill(8) + return str(len(binary_str.rstrip("0"))) + + +def get_vrf(config): + vrf = "" + for c in config: + vrf = c["vrf"] if "vrf" in c.keys() and c["vrf"] else None + return vrf + + +def get_dest(config): + dest = [] + for c in config: + for address_family in c["address_families"]: + for route in address_family["routes"]: + dest.append(route["dest"]) + return dest diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/config/vlans/__init__.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/config/vlans/__init__.py new file mode 100644 index 00000000..e69de29b --- /dev/null +++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/config/vlans/__init__.py diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/config/vlans/vlans.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/config/vlans/vlans.py new file mode 100644 index 00000000..9e0fedf5 --- /dev/null +++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/config/vlans/vlans.py @@ -0,0 +1,258 @@ +# -*- coding: utf-8 -*- +# Copyright 2019 Red Hat +# GNU General Public License v3.0+ +# (see COPYING or https://www.gnu.org/licenses/gpl-3.0.txt) +""" +The eos_vlans class +It is in this file where the current configuration (as dict) +is compared to the provided configuration (as dict) and the command set +necessary to bring the current configuration to it's desired end-state is +created +""" + +from __future__ import absolute_import, division, print_function + +__metaclass__ = type + +from ansible_collections.ansible.netcommon.plugins.module_utils.network.common.cfg.base import ( + ConfigBase, +) +from ansible_collections.ansible.netcommon.plugins.module_utils.network.common.utils import ( + to_list, + dict_diff, + param_list_to_dict, +) +from ansible_collections.arista.eos.plugins.module_utils.network.eos.facts.facts import ( + Facts, +) + + +class Vlans(ConfigBase): + """ + The eos_vlans class + """ + + gather_subset = ["!all", "!min"] + + gather_network_resources = ["vlans"] + + def get_vlans_facts(self, data=None): + """ Get the 'facts' (the current configuration) + + :rtype: A dictionary + :returns: The current configuration as a dictionary + """ + facts, _warnings = Facts(self._module).get_facts( + self.gather_subset, self.gather_network_resources, data=data + ) + vlans_facts = facts["ansible_network_resources"].get("vlans") + if not vlans_facts: + return [] + return vlans_facts + + def execute_module(self): + """ Execute the module + + :rtype: A dictionary + :returns: The result from module execution + """ + result = {"changed": False} + warnings = list() + commands = list() + + if self.state in self.ACTION_STATES: + existing_vlans_facts = self.get_vlans_facts() + else: + existing_vlans_facts = {} + + if self.state in self.ACTION_STATES or self.state == "rendered": + commands.extend(self.set_config(existing_vlans_facts)) + + if commands and self.state in self.ACTION_STATES: + if not self._module.check_mode: + self._connection.edit_config(commands) + result["changed"] = True + if self.state in self.ACTION_STATES: + result["commands"] = commands + + if self.state in self.ACTION_STATES or self.state == "gathered": + changed_vlans_facts = self.get_vlans_facts() + elif self.state == "rendered": + result["rendered"] = commands + elif self.state == "parsed": + running_config = self._module.params["running_config"] + if not running_config: + self._module.fail_json( + msg="value of running_config parameter must not be empty for state parsed" + ) + result["parsed"] = self.get_vlans_facts(data=running_config) + + if self.state in self.ACTION_STATES: + result["before"] = existing_vlans_facts + if result["changed"]: + result["after"] = changed_vlans_facts + elif self.state == "gathered": + result["gathered"] = changed_vlans_facts + result["warnings"] = warnings + return result + + def set_config(self, existing_vlans_facts): + """ Collect the configuration from the args passed to the module, + collect the current configuration (as a dict from facts) + + :rtype: A list + :returns: the commands necessary to migrate the current configuration + to the desired configuration + """ + want = self._module.params["config"] + have = existing_vlans_facts + resp = self.set_state(want, have) + return to_list(resp) + + def set_state(self, want, have): + """ Select the appropriate function based on the state provided + + :param want: the desired configuration as a dictionary + :param have: the current configuration as a dictionary + :rtype: A list + :returns: the commands necessary to migrate the current configuration + to the desired configuration + """ + state = self._module.params["state"] + if ( + state in ("merged", "replaced", "overridden", "rendered") + and not want + ): + self._module.fail_json( + msg="value of config parameter must not be empty for state {0}".format( + state + ) + ) + want = param_list_to_dict(want, "vlan_id", remove_key=False) + have = param_list_to_dict(have, "vlan_id", remove_key=False) + if state == "overridden": + commands = self._state_overridden(want, have) + elif state == "deleted": + commands = self._state_deleted(want, have) + elif state == "merged" or state == "rendered": + commands = self._state_merged(want, have) + elif state == "replaced": + commands = self._state_replaced(want, have) + return commands + + @staticmethod + def _state_replaced(want, have): + """ The command generator when state is replaced + + :rtype: A list + :returns: the commands necessary to migrate the current configuration + to the desired configuration + """ + commands = [] + for vlan_id, desired in want.items(): + if vlan_id in have: + extant = have[vlan_id] + else: + extant = dict() + + add_config = dict_diff(extant, desired) + del_config = dict_diff(desired, extant) + + commands.extend(generate_commands(vlan_id, add_config, del_config)) + + return commands + + @staticmethod + def _state_overridden(want, have): + """ The command generator when state is overridden + + :rtype: A list + :returns: the commands necessary to migrate the current configuration + to the desired configuration + """ + commands = [] + for vlan_id, extant in have.items(): + if vlan_id in want: + desired = want[vlan_id] + else: + desired = dict() + + add_config = dict_diff(extant, desired) + del_config = dict_diff(desired, extant) + + commands.extend(generate_commands(vlan_id, add_config, del_config)) + + # Handle vlans not already in config + new_vlans = [vlan_id for vlan_id in want if vlan_id not in have] + for vlan_id in new_vlans: + desired = want[vlan_id] + extant = dict(vlan_id=vlan_id) + add_config = dict_diff(extant, desired) + + commands.extend(generate_commands(vlan_id, add_config, {})) + + return commands + + @staticmethod + def _state_merged(want, have): + """ The command generator when state is merged + + :rtype: A list + :returns: the commands necessary to merge the provided into + the current configuration + """ + commands = [] + for vlan_id, desired in want.items(): + if vlan_id in have: + extant = have[vlan_id] + else: + extant = dict() + + add_config = dict_diff(extant, desired) + + commands.extend(generate_commands(vlan_id, add_config, {})) + + return commands + + @staticmethod + def _state_deleted(want, have): + """ The command generator when state is deleted + + :rtype: A list + :returns: the commands necessary to remove the current configuration + of the provided objects + """ + commands = [] + for vlan_id in want: + desired = dict() + if vlan_id in have: + extant = have[vlan_id] + else: + continue + + del_config = dict_diff(desired, extant) + + commands.extend(generate_commands(vlan_id, {}, del_config)) + + return commands + + +def generate_commands(vlan_id, to_set, to_remove): + commands = [] + if "vlan_id" in to_remove: + return ["no vlan {0}".format(vlan_id)] + + for key in to_remove: + if key in to_set.keys(): + continue + commands.append("no {0}".format(key)) + + for key, value in to_set.items(): + if key == "vlan_id" or value is None: + continue + + commands.append("{0} {1}".format(key, value)) + + if commands: + commands.insert(0, "vlan {0}".format(vlan_id)) + return commands diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/eos.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/eos.py new file mode 100644 index 00000000..609cec5f --- /dev/null +++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/eos.py @@ -0,0 +1,751 @@ +# +# 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. +# +# (c) 2017 Red Hat, Inc. +# +# Redistribution and use in source and binary forms, with or without modification, +# are permitted provided that the following conditions are met: +# +# * 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. +# +# 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. +# +from __future__ import absolute_import, division, print_function + +__metaclass__ = type +import json +import os +import time + +from ansible.module_utils._text import to_text +from ansible.module_utils.basic import env_fallback +from ansible.module_utils.connection import Connection, ConnectionError +from ansible_collections.ansible.netcommon.plugins.module_utils.network.common.config import ( + NetworkConfig, + dumps, +) +from ansible_collections.ansible.netcommon.plugins.module_utils.network.common.utils import ( + to_list, + ComplexList, +) +from ansible.module_utils.urls import fetch_url + +_DEVICE_CONNECTION = None + +eos_provider_spec = { + "host": dict(), + "port": dict(type="int"), + "username": dict(fallback=(env_fallback, ["ANSIBLE_NET_USERNAME"])), + "password": dict( + fallback=(env_fallback, ["ANSIBLE_NET_PASSWORD"]), no_log=True + ), + "ssh_keyfile": dict( + fallback=(env_fallback, ["ANSIBLE_NET_SSH_KEYFILE"]), type="path" + ), + "authorize": dict( + fallback=(env_fallback, ["ANSIBLE_NET_AUTHORIZE"]), + type="bool", + default=False, + ), + "auth_pass": dict( + no_log=True, fallback=(env_fallback, ["ANSIBLE_NET_AUTH_PASS"]) + ), + "use_ssl": dict(default=True, type="bool"), + "use_proxy": dict(default=True, type="bool"), + "validate_certs": dict(default=True, type="bool"), + "timeout": dict(type="int"), + "transport": dict(default="cli", choices=["cli", "eapi"]), +} +eos_argument_spec = { + "provider": dict( + type="dict", + options=eos_provider_spec, + removed_at_date="2022-06-01", + removed_from_collection="arista.eos", + ) +} + + +def get_provider_argspec(): + return eos_provider_spec + + +def get_connection(module): + global _DEVICE_CONNECTION + if not _DEVICE_CONNECTION: + if is_local_eapi(module): + conn = LocalEapi(module) + else: + connection_proxy = Connection(module._socket_path) + cap = json.loads(connection_proxy.get_capabilities()) + if cap["network_api"] == "cliconf": + conn = Cli(module) + elif cap["network_api"] == "eapi": + conn = HttpApi(module) + _DEVICE_CONNECTION = conn + return _DEVICE_CONNECTION + + +class Cli: + def __init__(self, module): + self._module = module + self._device_configs = {} + self._session_support = None + self._connection = None + + @property + def supports_sessions(self): + if self._session_support is None: + self._session_support = self._get_connection().supports_sessions() + return self._session_support + + def _get_connection(self): + if self._connection: + return self._connection + self._connection = Connection(self._module._socket_path) + + return self._connection + + def get_config(self, flags=None): + """Retrieves the current config from the device or cache + """ + flags = [] if flags is None else flags + + cmd = "show running-config " + cmd += " ".join(flags) + cmd = cmd.strip() + + try: + return self._device_configs[cmd] + except KeyError: + conn = self._get_connection() + try: + out = conn.get_config(flags=flags) + except ConnectionError as exc: + self._module.fail_json( + msg=to_text(exc, errors="surrogate_then_replace") + ) + + cfg = to_text(out, errors="surrogate_then_replace").strip() + self._device_configs[cmd] = cfg + return cfg + + def run_commands(self, commands, check_rc=True): + """Run list of commands on remote device and return results + """ + connection = self._get_connection() + try: + response = connection.run_commands( + commands=commands, check_rc=check_rc + ) + except ConnectionError as exc: + self._module.fail_json( + msg=to_text(exc, errors="surrogate_then_replace") + ) + return response + + def load_config(self, commands, commit=False, replace=False): + """Loads the config commands onto the remote device + """ + conn = self._get_connection() + try: + response = conn.edit_config(commands, commit, replace) + except ConnectionError as exc: + message = getattr(exc, "err", to_text(exc)) + if ( + "check mode is not supported without configuration session" + in message + ): + self._module.warn( + "EOS can not check config without config session" + ) + response = {"changed": True} + else: + self._module.fail_json( + msg="%s" % message, + data=to_text(message, errors="surrogate_then_replace"), + ) + + return response + + def get_diff( + self, + candidate=None, + running=None, + diff_match="line", + diff_ignore_lines=None, + path=None, + diff_replace="line", + ): + conn = self._get_connection() + try: + diff = conn.get_diff( + candidate=candidate, + running=running, + diff_match=diff_match, + diff_ignore_lines=diff_ignore_lines, + path=path, + diff_replace=diff_replace, + ) + except ConnectionError as exc: + self._module.fail_json( + msg=to_text(exc, errors="surrogate_then_replace") + ) + return diff + + def get_capabilities(self): + """Returns platform info of the remove device + """ + if hasattr(self._module, "_capabilities"): + return self._module._capabilities + + connection = self._get_connection() + try: + capabilities = connection.get_capabilities() + except ConnectionError as exc: + self._module.fail_json( + msg=to_text(exc, errors="surrogate_then_replace") + ) + self._module._capabilities = json.loads(capabilities) + return self._module._capabilities + + +class LocalEapi: + def __init__(self, module): + self._module = module + self._enable = None + self._session_support = None + self._device_configs = {} + + provider = module.params.get("provider") or {} + host = provider.get("host") + port = provider.get("port") + + self._module.params["url_username"] = provider.get("username") + self._module.params["url_password"] = provider.get("password") + + if provider.get("use_ssl"): + proto = "https" + else: + proto = "http" + + module.params["validate_certs"] = provider.get("validate_certs") + + self._url = "%s://%s:%s/command-api" % (proto, host, port) + + if provider.get("auth_pass"): + self._enable = { + "cmd": "enable", + "input": provider.get("auth_pass"), + } + else: + self._enable = "enable" + + @property + def supports_sessions(self): + if self._session_support is None: + response = self.send_request(["show configuration sessions"]) + self._session_support = "error" not in response + return self._session_support + + def _request_builder(self, commands, output, reqid=None): + params = dict(version=1, cmds=commands, format=output) + return dict(jsonrpc="2.0", id=reqid, method="runCmds", params=params) + + def send_request(self, commands, output="text"): + commands = to_list(commands) + + if self._enable: + commands.insert(0, self._enable) + + body = self._request_builder(commands, output) + data = self._module.jsonify(body) + + headers = {"Content-Type": "application/json-rpc"} + timeout = self._module.params["provider"]["timeout"] + use_proxy = self._module.params["provider"]["use_proxy"] + + response, headers = fetch_url( + self._module, + self._url, + data=data, + headers=headers, + method="POST", + timeout=timeout, + use_proxy=use_proxy, + ) + + if headers["status"] != 200: + self._module.fail_json(**headers) + + try: + data = response.read() + response = self._module.from_json( + to_text(data, errors="surrogate_then_replace") + ) + except ValueError: + self._module.fail_json( + msg="unable to load response from device", data=data + ) + + if self._enable and "result" in response: + response["result"].pop(0) + + return response + + def run_commands(self, commands, check_rc=True): + """Runs list of commands on remote device and returns results + """ + output = None + queue = list() + responses = list() + + def _send(commands, output): + response = self.send_request(commands, output=output) + if "error" in response: + err = response["error"] + self._module.fail_json(msg=err["message"], code=err["code"]) + return response["result"] + + for item in to_list(commands): + if is_json(item["command"]): + item["command"] = str(item["command"]).replace("| json", "") + item["output"] = "json" + + if output and output != item["output"]: + responses.extend(_send(queue, output)) + queue = list() + + output = item["output"] or "json" + queue.append(item["command"]) + + if queue: + responses.extend(_send(queue, output)) + + for index, item in enumerate(commands): + try: + responses[index] = responses[index]["output"].strip() + except KeyError: + pass + + return responses + + def get_config(self, flags=None): + """Retrieves the current config from the device or cache + """ + flags = [] if flags is None else flags + + cmd = "show running-config " + cmd += " ".join(flags) + cmd = cmd.strip() + + try: + return self._device_configs[cmd] + except KeyError: + out = self.send_request(cmd) + cfg = str(out["result"][0]["output"]).strip() + self._device_configs[cmd] = cfg + return cfg + + def configure(self, commands): + """Sends the ordered set of commands to the device + """ + cmds = ["configure terminal"] + cmds.extend(commands) + + responses = self.send_request(commands) + if "error" in responses: + err = responses["error"] + self._module.fail_json(msg=err["message"], code=err["code"]) + + return responses[1:] + + def load_config(self, config, commit=False, replace=False): + """Loads the configuration onto the remote devices + + If the device doesn't support configuration sessions, this will + fallback to using configure() to load the commands. If that happens, + there will be no returned diff or session values + """ + use_session = os.getenv("ANSIBLE_EOS_USE_SESSIONS", True) + try: + use_session = int(use_session) + except ValueError: + pass + + if not all((bool(use_session), self.supports_sessions)): + if commit: + return self.configure(config) + else: + self._module.warn( + "EOS can not check config without config session" + ) + result = {"changed": True} + return result + + session = "ansible_%s" % int(time.time()) + result = {"session": session} + commands = ["configure session %s" % session] + + if replace: + commands.append("rollback clean-config") + + commands.extend(config) + + response = self.send_request(commands) + if "error" in response: + commands = ["configure session %s" % session, "abort"] + self.send_request(commands) + err = response["error"] + error_text = [] + for data in err["data"]: + error_text.extend(data.get("errors", [])) + error_text = "\n".join(error_text) or err["message"] + self._module.fail_json(msg=error_text, code=err["code"]) + + commands = [ + "configure session %s" % session, + "show session-config diffs", + ] + if commit: + commands.append("commit") + else: + commands.append("abort") + + response = self.send_request(commands, output="text") + diff = response["result"][1]["output"] + if len(diff) > 0: + result["diff"] = diff + + return result + + # get_diff added here to support connection=local and transport=eapi scenario + def get_diff( + self, + candidate, + running=None, + diff_match="line", + diff_ignore_lines=None, + path=None, + diff_replace="line", + ): + diff = {} + + # prepare candidate configuration + candidate_obj = NetworkConfig(indent=3) + candidate_obj.load(candidate) + + if running and diff_match != "none" and diff_replace != "config": + # running configuration + running_obj = NetworkConfig( + indent=3, contents=running, ignore_lines=diff_ignore_lines + ) + configdiffobjs = candidate_obj.difference( + running_obj, path=path, match=diff_match, replace=diff_replace + ) + + else: + configdiffobjs = candidate_obj.items + + configdiff = ( + dumps(configdiffobjs, "commands") if configdiffobjs else "" + ) + diff["config_diff"] = configdiff if configdiffobjs else {} + return diff + + def get_capabilities(self): + # Implement the bare minimum to support eos_facts + return dict(device_info=dict(network_os="eos"), network_api="eapi") + + +class HttpApi: + def __init__(self, module): + self._module = module + self._device_configs = {} + self._session_support = None + self._connection_obj = None + + @property + def _connection(self): + if not self._connection_obj: + self._connection_obj = Connection(self._module._socket_path) + + return self._connection_obj + + @property + def supports_sessions(self): + if self._session_support is None: + self._session_support = self._connection.supports_sessions() + return self._session_support + + def run_commands(self, commands, check_rc=True): + """Runs list of commands on remote device and returns results + """ + output = None + queue = list() + responses = list() + + def run_queue(queue, output): + try: + response = to_list( + self._connection.send_request(queue, output=output) + ) + except ConnectionError as exc: + if check_rc: + raise + return to_list(to_text(exc)) + + if output == "json": + response = [json.loads(item) for item in response] + return response + + for item in to_list(commands): + cmd_output = "text" + if isinstance(item, dict): + command = item["command"] + if "output" in item: + cmd_output = item["output"] + else: + command = item + + # Emulate '| json' from CLI + if is_json(command): + command = command.rsplit("|", 1)[0] + cmd_output = "json" + + if output and output != cmd_output: + responses.extend(run_queue(queue, output)) + queue = list() + + output = cmd_output + queue.append(command) + + if queue: + responses.extend(run_queue(queue, output)) + + return responses + + def get_config(self, flags=None): + """Retrieves the current config from the device or cache + """ + flags = [] if flags is None else flags + + cmd = "show running-config " + cmd += " ".join(flags) + cmd = cmd.strip() + + try: + return self._device_configs[cmd] + except KeyError: + try: + out = self._connection.send_request(cmd) + except ConnectionError as exc: + self._module.fail_json( + msg=to_text(exc, errors="surrogate_then_replace") + ) + + cfg = to_text(out).strip() + self._device_configs[cmd] = cfg + return cfg + + def get_diff( + self, + candidate=None, + running=None, + diff_match="line", + diff_ignore_lines=None, + path=None, + diff_replace="line", + ): + diff = {} + + # prepare candidate configuration + candidate_obj = NetworkConfig(indent=3) + candidate_obj.load(candidate) + + if running and diff_match != "none" and diff_replace != "config": + # running configuration + running_obj = NetworkConfig( + indent=3, contents=running, ignore_lines=diff_ignore_lines + ) + configdiffobjs = candidate_obj.difference( + running_obj, path=path, match=diff_match, replace=diff_replace + ) + + else: + configdiffobjs = candidate_obj.items + + diff["config_diff"] = ( + dumps(configdiffobjs, "commands") if configdiffobjs else {} + ) + return diff + + def load_config(self, config, commit=False, replace=False): + """Loads the configuration onto the remote devices + + If the device doesn't support configuration sessions, this will + fallback to using configure() to load the commands. If that happens, + there will be no returned diff or session values + """ + return self.edit_config(config, commit, replace) + + def edit_config(self, config, commit=False, replace=False): + """Loads the configuration onto the remote devices + + If the device doesn't support configuration sessions, this will + fallback to using configure() to load the commands. If that happens, + there will be no returned diff or session values + """ + session = "ansible_%s" % int(time.time()) + result = {"session": session} + banner_cmd = None + banner_input = [] + + commands = ["configure session %s" % session] + if replace: + commands.append("rollback clean-config") + + for command in config: + if command.startswith("banner"): + banner_cmd = command + banner_input = [] + elif banner_cmd: + if command == "EOF": + command = { + "cmd": banner_cmd, + "input": "\n".join(banner_input), + } + banner_cmd = None + commands.append(command) + else: + banner_input.append(command) + continue + else: + commands.append(command) + + try: + response = self._connection.send_request(commands) + except Exception: + commands = ["configure session %s" % session, "abort"] + response = self._connection.send_request(commands, output="text") + raise + + commands = [ + "configure session %s" % session, + "show session-config diffs", + ] + if commit: + commands.append("commit") + else: + commands.append("abort") + + response = self._connection.send_request(commands, output="text") + diff = response[1].strip() + if diff: + result["diff"] = diff + + return result + + def get_capabilities(self): + """Returns platform info of the remove device + """ + try: + capabilities = self._connection.get_capabilities() + except ConnectionError as exc: + self._module.fail_json( + msg=to_text(exc, errors="surrogate_then_replace") + ) + + return json.loads(capabilities) + + +def is_json(cmd): + return to_text(cmd, errors="surrogate_then_replace").endswith("| json") + + +def is_local_eapi(module): + provider = module.params.get("provider") + if provider: + return provider.get("transport") == "eapi" + return False + + +def to_command(module, commands): + if is_local_eapi(module): + default_output = "json" + else: + default_output = "text" + + transform = ComplexList( + dict( + command=dict(key=True), + output=dict(default=default_output), + prompt=dict(type="list"), + answer=dict(type="list"), + newline=dict(type="bool", default=True), + sendonly=dict(type="bool", default=False), + check_all=dict(type="bool", default=False), + ), + module, + ) + + return transform(to_list(commands)) + + +def get_config(module, flags=None): + flags = None if flags is None else flags + + conn = get_connection(module) + return conn.get_config(flags) + + +def run_commands(module, commands, check_rc=True): + conn = get_connection(module) + return conn.run_commands(to_command(module, commands), check_rc=check_rc) + + +def load_config(module, config, commit=False, replace=False): + conn = get_connection(module) + return conn.load_config(config, commit, replace) + + +def get_diff( + self, + candidate=None, + running=None, + diff_match="line", + diff_ignore_lines=None, + path=None, + diff_replace="line", +): + conn = self.get_connection() + return conn.get_diff( + candidate=candidate, + running=running, + diff_match=diff_match, + diff_ignore_lines=diff_ignore_lines, + path=path, + diff_replace=diff_replace, + ) + + +def get_capabilities(module): + conn = get_connection(module) + return conn.get_capabilities() diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/__init__.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/__init__.py new file mode 100644 index 00000000..e69de29b --- /dev/null +++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/__init__.py diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/acl_interfaces/__init__.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/acl_interfaces/__init__.py new file mode 100644 index 00000000..e69de29b --- /dev/null +++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/acl_interfaces/__init__.py diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/acl_interfaces/acl_interfaces.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/acl_interfaces/acl_interfaces.py new file mode 100644 index 00000000..4157bf61 --- /dev/null +++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/acl_interfaces/acl_interfaces.py @@ -0,0 +1,146 @@ +# +# -*- coding: utf-8 -*- +# Copyright 2019 Red Hat +# GNU General Public License v3.0+ +# (see COPYING or https://www.gnu.org/licenses/gpl-3.0.txt) +""" +The eos acl_interfaces fact class +It is in this file the configuration is collected from the device +for a given resource, parsed, and the facts tree is populated +based on the configuration. +""" + +from __future__ import absolute_import, division, print_function + +__metaclass__ = type + + +import re +from copy import deepcopy + +from ansible_collections.ansible.netcommon.plugins.module_utils.network.common import ( + utils, +) +from ansible_collections.arista.eos.plugins.module_utils.network.eos.argspec.acl_interfaces.acl_interfaces import ( + Acl_interfacesArgs, +) + + +class Acl_interfacesFacts(object): + """ The eos acl_interfaces fact class + """ + + def __init__(self, module, subspec="config", options="options"): + self._module = module + self.argument_spec = Acl_interfacesArgs.argument_spec + spec = deepcopy(self.argument_spec) + if subspec: + if options: + facts_argument_spec = spec[subspec][options] + else: + facts_argument_spec = spec[subspec] + else: + facts_argument_spec = spec + + self.generated_spec = utils.generate_dict(facts_argument_spec) + + def get_device_data(self, connection): + return connection.get( + "show running-config | include interface | access-group | traffic-filter" + ) + + def populate_facts(self, connection, ansible_facts, data=None): + """ Populate the facts for acl_interfaces + :param connection: the device connection + :param ansible_facts: Facts dictionary + :param data: previously collected conf + :rtype: dictionary + :returns: facts + """ + + if not data: + data = self.get_device_data(connection) + # split the config into instances of the resource + resource_delim = "interface" + find_pattern = r"(?:^|\n)%s.*?(?=(?:^|\n)%s|$)" % ( + resource_delim, + resource_delim, + ) + resources = [ + p.strip() for p in re.findall(find_pattern, data, re.DOTALL) + ] + objs = [] + for resource in resources: + if resource: + obj = self.render_config(self.generated_spec, resource) + if obj: + objs.append(obj) + + ansible_facts["ansible_network_resources"].pop("acl_interfaces", None) + facts = {} + if objs: + params = utils.validate_config( + self.argument_spec, {"config": objs} + ) + facts["acl_interfaces"] = [ + utils.remove_empties(cfg) for cfg in params["config"] + ] + + ansible_facts["ansible_network_resources"].update(facts) + return ansible_facts + + def render_config(self, spec, conf): + """ + Render config as dictionary structure and delete keys + from spec for null values + + :param spec: The facts tree, generated from the argspec + :param conf: The configuration + :rtype: dictionary + :returns: The generated config + """ + config = deepcopy(spec) + access_group_list = [] + access_group_v6_list = [] + acls_list = [] + group_list = [] + group_dict = {} + config["name"] = utils.parse_conf_arg(conf, "interface") + conf_lines = conf.split("\n") + for line in conf_lines: + if config["name"] in line: + continue + access_group = utils.parse_conf_arg(line, "ip access-group") + # This module was verified on an ios device since vEOS doesnot support + # acl_interfaces cnfiguration. In ios, ipv6 acl is configured as + # traffic-filter and in eos it is access-group + + # access_group_v6 = utils.parse_conf_arg(line, 'ipv6 traffic-filter') + access_group_v6 = utils.parse_conf_arg(line, "ipv6 access-group") + if access_group: + access_group_list.append(access_group) + if access_group_v6: + access_group_v6_list.append(access_group_v6) + if access_group_list: + for acl in access_group_list: + a_name = acl.split()[0] + a_dir = acl.split()[1] + acls_dict = {"name": a_name, "direction": a_dir} + acls_list.append(acls_dict) + group_dict = {"afi": "ipv4", "acls": acls_list} + group_list.append(group_dict) + acls_list = [] + if group_list: + config["access_groups"] = group_list + if access_group_v6_list: + for acl in access_group_v6_list: + a_name = acl.split()[0] + a_dir = acl.split()[1] + acls_dict = {"name": a_name, "direction": a_dir} + acls_list.append(acls_dict) + group_dict = {"acls": acls_list, "afi": "ipv6"} + group_list.append(group_dict) + acls_list = [] + if group_list: + config["access_groups"] = group_list + return utils.remove_empties(config) diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/acls/__init__.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/acls/__init__.py new file mode 100644 index 00000000..e69de29b --- /dev/null +++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/acls/__init__.py diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/acls/acls.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/acls/acls.py new file mode 100644 index 00000000..e6282303 --- /dev/null +++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/acls/acls.py @@ -0,0 +1,387 @@ +# +# -*- coding: utf-8 -*- +# Copyright 2019 Red Hat +# GNU General Public License v3.0+ +# (see COPYING or https://www.gnu.org/licenses/gpl-3.0.txt) +""" +The eos acls fact class +It is in this file the configuration is collected from the device +for a given resource, parsed, and the facts tree is populated +based on the configuration. +""" + +from __future__ import absolute_import, division, print_function + +__metaclass__ = type + +import re +from copy import deepcopy + +from ansible_collections.ansible.netcommon.plugins.module_utils.network.common import ( + utils, +) +from ansible_collections.arista.eos.plugins.module_utils.network.eos.argspec.acls.acls import ( + AclsArgs, +) + + +class AclsFacts(object): + """ The eos acls fact class + """ + + def __init__(self, module, subspec="config", options="options"): + self._module = module + self.argument_spec = AclsArgs.argument_spec + spec = deepcopy(self.argument_spec) + if subspec: + if options: + facts_argument_spec = spec[subspec][options] + else: + facts_argument_spec = spec[subspec] + else: + facts_argument_spec = spec + + self.generated_spec = utils.generate_dict(facts_argument_spec) + + def get_device_data(self, connection): + return connection.get("show running-config | section access-list") + + def populate_facts(self, connection, ansible_facts, data=None): + """ Populate the facts for acls + :param connection: the device connection + :param ansible_facts: Facts dictionary + :param data: previously collected conf + :rtype: dictionary + :returns: facts + """ + if not data: + data = self.get_device_data(connection) + + # split the config into instances of the resource + find_pattern = r"(?:^|\n)(?:ip|ipv6) access\-list.*?(?=(?:^|\n)(?:ip|ipv6) access\-list|$)" + resources = [] + for p in re.findall(find_pattern, data, re.DOTALL): + resources.append(p) + + objs = [] + ipv4list = [] + ipv6list = [] + for resource in resources: + if "ipv6" in resource: + ipv6list.append(resource) + else: + ipv4list.append(resource) + ipv4list = ["\n".join(ipv4list)] + ipv6list = ["\n".join(ipv6list)] + for resource in ipv4list: + if resource: + obj = self.render_config(self.generated_spec, resource) + if obj: + objs.append(obj) + for resource in ipv6list: + if resource: + obj = self.render_config(self.generated_spec, resource) + if obj: + objs.append(obj) + + ansible_facts["ansible_network_resources"].pop("acls", None) + facts = {} + if objs: + facts["acls"] = [] + params = utils.validate_config( + self.argument_spec, {"config": objs} + ) + for cfg in params["config"]: + facts["acls"].append(utils.remove_empties(cfg)) + + ansible_facts["ansible_network_resources"].update(facts) + return ansible_facts + + def render_config(self, spec, conf): + """ + Render config as dictionary structure and delete keys + from spec for null values + + :param spec: The facts tree, generated from the argspec + :param conf: The configuration + :rtype: dictionary + :returns: The generated config + """ + config = deepcopy(spec) + afi_list = [] + acls_list = [] + name_dict = {} + standard = 0 + operator = ["eq", "lt", "neq", "range", "gt"] + flags = ["ack", "established", "fin", "psh", "rst", "syn", "urg"] + others = ["hop_limit", "log", "ttl", "fragments", "tracked"] + for dev_config in conf.split("\n"): + ace_dict = {} + if not dev_config: + continue + if dev_config == "!": + continue + dev_config = dev_config.strip() + matches = re.findall(r"(ip.*?) access-list (.*)", dev_config) + if matches: + afi = "ipv4" if matches[0][0] == "ip" else "ipv6" + ace_list = [] + if bool(name_dict): + acls_list.append(name_dict.copy()) + name_dict = {} + if afi not in afi_list: + afi_list.append(afi) + config.update({"afi": afi}) + if "standard" in matches[0][1]: + standard = 1 + name = matches[0][1].split() + name_dict.update({"name": name[1]}) + name_dict.update({"standard": True}) + else: + name_dict.update({"name": matches[0][1]}) + else: + source_dict = {} + dest_dict = {} + dev_config = re.sub("-", "_", dev_config) + dev_config_remainder = dev_config.split() + if "fragment_rules" in dev_config: + ace_dict.update({"sequence": dev_config_remainder.pop(0)}) + ace_dict.update({"fragment_rules": True}) + if "remark" in dev_config: + ace_dict.update({"sequence": dev_config_remainder.pop(0)}) + ace_dict.update( + {"remark": " ".join(dev_config_remainder[1:])} + ) + seq = re.search(r"\d+ (permit|deny) .*", dev_config) + if seq: + ace_dict.update({"sequence": dev_config_remainder.pop(0)}) + ace_dict.update({"grant": dev_config_remainder.pop(0)}) + if ( + dev_config_remainder + and dev_config_remainder[0] == "vlan" + ): + vlan_str = "" + dev_config_remainder.pop(0) + if ( + dev_config_remainder + and dev_config_remainder[0] == "inner" + ): + vlan_str = dev_config_remainder.pop(0) + " " + vlan_str = ( + dev_config_remainder.pop(0) + + " " + + dev_config_remainder.pop(0) + ) + ace_dict.update({"vlan": vlan_str}) + if not standard: + protocol = dev_config_remainder[0] + ace_dict.update( + {"protocol": dev_config_remainder.pop(0)} + ) + src_prefix = re.search(r"/", dev_config_remainder[0]) + src_address = re.search( + r"[a-z\d:\.]+", dev_config_remainder[0] + ) + if ( + dev_config_remainder + and dev_config_remainder[0] == "host" + ): + source_dict.update( + {"host": dev_config_remainder.pop(1)} + ) + dev_config_remainder.pop(0) + elif ( + dev_config_remainder + and dev_config_remainder[0] == "any" + ): + source_dict.update({"any": True}) + dev_config_remainder.pop(0) + elif src_prefix: + source_dict.update( + {"subnet_address": dev_config_remainder.pop(0)} + ) + elif src_address: + source_dict.update( + {"address": dev_config_remainder.pop(0)} + ) + source_dict.update( + {"wildcard_bits": dev_config_remainder.pop(0)} + ) + if dev_config_remainder: + if ( + dev_config_remainder + and dev_config_remainder[0] in operator + ): + port_dict = {} + src_port = "" + src_opr = dev_config_remainder.pop(0) + portlist = dev_config_remainder[:] + for config_remainder in portlist: + addr = re.search(r"[\.\:]", config_remainder) + if ( + config_remainder == "any" + or config_remainder == "host" + or addr + ): + break + src_port = src_port + " " + config_remainder + dev_config_remainder.pop(0) + src_port = src_port.strip() + port_dict.update({src_opr: src_port}) + source_dict.update({"port_protocol": port_dict}) + ace_dict.update({"source": source_dict}) + if not dev_config_remainder or standard: + if ( + dev_config_remainder + and "log" in dev_config_remainder + ): + ace_dict.update({"log": True}) + if bool(ace_dict): + ace_list.append(ace_dict.copy()) + if len(ace_list): + name_dict = name_dict.copy() + name_dict.update({"aces": ace_list[:]}) + # acls_list.append(name_dict) + continue + dest_prefix = re.search(r"/", dev_config_remainder[0]) + dest_address = re.search( + r"[a-z\d:\.]+", dev_config_remainder[0] + ) + if ( + dev_config_remainder + and dev_config_remainder[0] == "host" + ): + dest_dict.update({"host": dev_config_remainder.pop(1)}) + dev_config_remainder.pop(0) + elif ( + dev_config_remainder + and dev_config_remainder[0] == "any" + ): + dest_dict.update({"any": True}) + dev_config_remainder.pop(0) + elif dest_prefix: + dest_dict.update( + {"subnet_address": dev_config_remainder.pop(0)} + ) + elif dest_address: + dest_dict.update( + {"address": dev_config_remainder.pop(0)} + ) + dest_dict.update( + {"wildcard_bits": dev_config_remainder.pop(0)} + ) + if dev_config_remainder: + if dev_config_remainder[0] in operator: + port_dict = {} + dest_port = "" + dest_opr = dev_config_remainder.pop(0) + portlist = dev_config_remainder[:] + for config_remainder in portlist: + if ( + config_remainder in operator + or config_remainder in others + ): + break + dest_port = dest_port + " " + config_remainder + dev_config_remainder.pop(0) + dest_port = dest_port.strip() + port_dict.update({dest_opr: dest_port}) + dest_dict.update({"port_protocol": port_dict}) + ace_dict.update({"destination": dest_dict}) + protocol_option_dict = {} + tcp_dict = {} + icmp_dict = {} + ip_dict = {} + if not dev_config_remainder: + if bool(ace_dict): + ace_list.append(ace_dict.copy()) + if len(ace_list): + name_dict = name_dict.copy() + name_dict.update({"aces": ace_list[:]}) + # acls_list.append(name_dict) + continue + if protocol == "tcp" or "6": + protocol = "tcp" + flags_dict = {} + if ( + dev_config_remainder + and dev_config_remainder[0] in flags + ): + flaglist = dev_config_remainder[:] + for config_remainder in flaglist: + if config_remainder not in flags: + break + flags_dict.update({config_remainder: True}) + dev_config_remainder.pop(0) + if bool(flags_dict): + tcp_dict.update({"flags": flags_dict}) + if bool(tcp_dict): + protocol_option_dict.update({"tcp": tcp_dict}) + if ( + protocol == "icmp" + or protocol == "icmpv6" + or protocol == "1" + or protocol == "58" + ): + if protocol == "1": + protocol = "icmp" + elif protocol == "58": + protocol = "icmpv6" + if ( + dev_config_remainder + and dev_config_remainder[0] not in others + ): + icmp_dict.update({dev_config_remainder[0]: True}) + dev_config_remainder.pop(0) + if bool(icmp_dict): + protocol_option_dict.update({protocol: icmp_dict}) + if protocol in ["ip", "ipv6"]: + if ( + dev_config_remainder + and dev_config_remainder[0] == "nexthop_group" + ): + dev_config_remainder.pop(0) + ip_dict.update( + {"nexthop_group": dev_config_remainder.pop(0)} + ) + if bool(ip_dict): + protocol_option_dict.update({protocol: ip_dict}) + if bool(protocol_option_dict): + ace_dict.update( + {"protocol_options": protocol_option_dict} + ) + if ( + dev_config_remainder + and dev_config_remainder[0] == "ttl" + ): + dev_config_remainder.pop(0) + op = dev_config_remainder.pop(0) + ttl_dict = {op: dev_config_remainder.pop(0)} + ace_dict.update({"ttl": ttl_dict}) + for config_remainder in dev_config_remainder: + if config_remainder in others: + if config_remainder == "hop_limit": + hop_index = dev_config_remainder.index( + config_remainder + ) + hoplimit_dict = { + dev_config_remainder[ + hop_index + 1 + ]: dev_config_remainder[hop_index + 2] + } + ace_dict.update({"hop_limit": hoplimit_dict}) + dev_config_remainder.pop(0) + continue + ace_dict.update({config_remainder: True}) + dev_config_remainder.pop(0) + if dev_config_remainder: + config.update({"line": dev_config}) + return utils.remove_empties(config) + if bool(ace_dict): + ace_list.append(ace_dict.copy()) + if len(ace_list): + name_dict = name_dict.copy() + name_dict.update({"aces": ace_list[:]}) + acls_list.append(name_dict.copy()) + config.update({"acls": acls_list}) + return utils.remove_empties(config) diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/bgp_address_family/__init__.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/bgp_address_family/__init__.py new file mode 100644 index 00000000..e69de29b --- /dev/null +++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/bgp_address_family/__init__.py diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/bgp_address_family/bgp_address_family.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/bgp_address_family/bgp_address_family.py new file mode 100644 index 00000000..497ccfd7 --- /dev/null +++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/bgp_address_family/bgp_address_family.py @@ -0,0 +1,116 @@ +# -*- coding: utf-8 -*- +# Copyright 2021 Red Hat +# 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 + +""" +The eos bgp_address_family fact class +It is in this file the configuration is collected from the device +for a given resource, parsed, and the facts tree is populated +based on the configuration. +""" + +from copy import deepcopy + +from ansible_collections.ansible.netcommon.plugins.module_utils.network.common import ( + utils, +) +from ansible_collections.arista.eos.plugins.module_utils.network.eos.rm_templates.bgp_address_family import ( + Bgp_afTemplate, +) +from ansible_collections.arista.eos.plugins.module_utils.network.eos.argspec.bgp_address_family.bgp_address_family import ( + Bgp_afArgs, +) +import re + + +class Bgp_afFacts(object): + """ The eos bgp_address_family facts class + """ + + def __init__(self, module, subspec="config", options="options"): + self._module = module + self.argument_spec = Bgp_afArgs.argument_spec + spec = deepcopy(self.argument_spec) + if subspec: + if options: + facts_argument_spec = spec[subspec][options] + else: + facts_argument_spec = spec[subspec] + else: + facts_argument_spec = spec + + self.generated_spec = utils.generate_dict(facts_argument_spec) + + def get_config(self, connection): + """Wrapper method for `connection.get()` + This method exists solely to allow the unit test framework to mock device connection calls. + """ + return connection.get("show running-config | section bgp ") + + def populate_facts(self, connection, ansible_facts, data=None): + """ Populate the facts for Bgp_af network resource + + :param connection: the device connection + :param ansible_facts: Facts dictionary + :param data: previously collected conf + + :rtype: dictionary + :returns: facts + """ + facts = {} + objs = [] + + if not data: + data = self.get_config(connection) + + # remove global configs from bgp_address_family + bgp_af_config = [] + vrf_set = "" + start = False + for bgp_line in data.splitlines(): + if "router bgp" in bgp_line: + bgp_af_config.append(bgp_line) + vrf_present = re.search(r"vrf\s\S+", bgp_line) + if vrf_present: + vrf_set = vrf_present.group(0) + if start: + bgp_af_config.append(bgp_line) + if "address-family" in bgp_line: + af_line = vrf_set + bgp_line + bgp_af_config.append(af_line) + start = True + if start and "!" in bgp_line: + start = False + + # parse native config using the Bgp_af template + bgp_af_parser = Bgp_afTemplate(lines=bgp_af_config) + objs = bgp_af_parser.parse() + if objs: + if "address_family" in objs: + objs["address_family"] = list(objs["address_family"].values()) + for af in objs["address_family"]: + if "neighbor" in af: + af["neighbor"] = list(af["neighbor"].values()) + if "network" in af: + af["network"] = list(af["network"].values()) + af["network"] = sorted( + af["network"], key=lambda k: k["address"] + ) + + ansible_facts["ansible_network_resources"].pop( + "bgp_address_family", None + ) + + params = utils.remove_empties( + utils.validate_config(self.argument_spec, {"config": objs}) + ) + + facts["bgp_address_family"] = params.get("config", []) + ansible_facts["ansible_network_resources"].update(facts) + + return ansible_facts diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/bgp_global/__init__.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/bgp_global/__init__.py new file mode 100644 index 00000000..e69de29b --- /dev/null +++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/bgp_global/__init__.py diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/bgp_global/bgp_global.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/bgp_global/bgp_global.py new file mode 100644 index 00000000..3603ea6d --- /dev/null +++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/bgp_global/bgp_global.py @@ -0,0 +1,119 @@ +# -*- coding: utf-8 -*- +# Copyright 2020 Red Hat +# 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 + +""" +The eos bgp_global fact class +It is in this file the configuration is collected from the device +for a given resource, parsed, and the facts tree is populated +based on the configuration. +""" + +from ansible.module_utils.six import iteritems +from ansible_collections.ansible.netcommon.plugins.module_utils.network.common import ( + utils, +) +from ansible_collections.arista.eos.plugins.module_utils.network.eos.rm_templates.bgp_global import ( + Bgp_globalTemplate, +) +from ansible_collections.arista.eos.plugins.module_utils.network.eos.argspec.bgp_global.bgp_global import ( + Bgp_globalArgs, +) + + +class Bgp_globalFacts(object): + """ The eos bgp_global facts class + """ + + def __init__(self, module, subspec="config", options="options"): + self._module = module + self.argument_spec = Bgp_globalArgs.argument_spec + + def get_config(self, connection): + """Wrapper method for `connection.get()` + This method exists solely to allow the unit test framework to mock device connection calls. + """ + return connection.get("show running-config | section bgp ") + + def populate_facts(self, connection, ansible_facts, data=None): + """ Populate the facts for Bgp_global network resource + + :param connection: the device connection + :param ansible_facts: Facts dictionary + :param data: previously collected conf + + :rtype: dictionary + :returns: facts + """ + facts = {} + objs = [] + + if not data: + data = self.get_config(connection) + + # remove address_family configs from bgp_global + bgp_global_config = [] + start = False + self._af = False + for bgp_line in data.splitlines(): + if not start: + bgp_global_config.append(bgp_line) + if "address-family" in bgp_line: + start = True + self._af = True + if start and "!" in bgp_line: + start = False + + # parse native config using the Bgp_global template + bgp_global_parser = Bgp_globalTemplate(lines=bgp_global_config) + objs = bgp_global_parser.parse() + + if objs: + global_vals = objs.get("vrfs", {}).pop("vrf_", {}) + for key, value in iteritems(global_vals): + objs[key] = value + + if "vrfs" in objs: + objs["vrfs"] = list(objs["vrfs"].values()) + for vrf in objs["vrfs"]: + if "neighbor" in vrf: + vrf["neighbor"] = list(vrf["neighbor"].values()) + if "network" in vrf: + vrf["network"] = list(vrf["network"].values()) + vrf["network"] = sorted( + vrf["network"], key=lambda k: k["address"] + ) + if "aggregate_address" in vrf: + vrf["aggregate_address"] = sorted( + vrf["aggregate_address"], + key=lambda k: k["address"], + ) + + if "neighbor" in objs: + objs["neighbor"] = list(objs["neighbor"].values()) + + if "network" in objs: + objs["network"] = list(objs["network"].values()) + objs["network"] = sorted( + objs["network"], key=lambda k: k["address"] + ) + if "aggregate_address" in objs: + objs["aggregate_address"] = sorted( + objs["aggregate_address"], key=lambda k: k["address"] + ) + + ansible_facts["ansible_network_resources"].pop("bgp_global", None) + + params = utils.remove_empties( + utils.validate_config(self.argument_spec, {"config": objs}) + ) + + facts["bgp_global"] = params.get("config", []) + ansible_facts["ansible_network_resources"].update(facts) + + return ansible_facts diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/facts.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/facts.py new file mode 100644 index 00000000..c1f4fbc0 --- /dev/null +++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/facts.py @@ -0,0 +1,127 @@ +# -*- coding: utf-8 -*- +# Copyright 2019 Red Hat +# GNU General Public License v3.0+ (see COPYING or https://www.gnu.org/licenses/gpl-3.0.txt) +""" +The facts class for eos +this file validates each subset of facts and selectively +calls the appropriate facts gathering function +""" + +from __future__ import absolute_import, division, print_function + +__metaclass__ = type + +from ansible_collections.ansible.netcommon.plugins.module_utils.network.common.facts.facts import ( + FactsBase, +) +from ansible_collections.arista.eos.plugins.module_utils.network.eos.facts.interfaces.interfaces import ( + InterfacesFacts, +) +from ansible_collections.arista.eos.plugins.module_utils.network.eos.facts.l2_interfaces.l2_interfaces import ( + L2_interfacesFacts, +) +from ansible_collections.arista.eos.plugins.module_utils.network.eos.facts.l3_interfaces.l3_interfaces import ( + L3_interfacesFacts, +) +from ansible_collections.arista.eos.plugins.module_utils.network.eos.facts.lacp.lacp import ( + LacpFacts, +) +from ansible_collections.arista.eos.plugins.module_utils.network.eos.facts.lacp_interfaces.lacp_interfaces import ( + Lacp_interfacesFacts, +) +from ansible_collections.arista.eos.plugins.module_utils.network.eos.facts.lag_interfaces.lag_interfaces import ( + Lag_interfacesFacts, +) +from ansible_collections.arista.eos.plugins.module_utils.network.eos.facts.lldp_global.lldp_global import ( + Lldp_globalFacts, +) +from ansible_collections.arista.eos.plugins.module_utils.network.eos.facts.lldp_interfaces.lldp_interfaces import ( + Lldp_interfacesFacts, +) +from ansible_collections.arista.eos.plugins.module_utils.network.eos.facts.vlans.vlans import ( + VlansFacts, +) +from ansible_collections.arista.eos.plugins.module_utils.network.eos.facts.legacy.base import ( + Default, + Hardware, + Config, + Interfaces, +) +from ansible_collections.arista.eos.plugins.module_utils.network.eos.facts.acl_interfaces.acl_interfaces import ( + Acl_interfacesFacts, +) +from ansible_collections.arista.eos.plugins.module_utils.network.eos.facts.acls.acls import ( + AclsFacts, +) +from ansible_collections.arista.eos.plugins.module_utils.network.eos.facts.static_routes.static_routes import ( + Static_routesFacts, +) +from ansible_collections.arista.eos.plugins.module_utils.network.eos.facts.ospfv2.ospfv2 import ( + Ospfv2Facts, +) +from ansible_collections.arista.eos.plugins.module_utils.network.eos.facts.ospfv3.ospfv3 import ( + Ospfv3Facts, +) +from ansible_collections.arista.eos.plugins.module_utils.network.eos.facts.ospf_interfaces.ospf_interfaces import ( + Ospf_interfacesFacts, +) +from ansible_collections.arista.eos.plugins.module_utils.network.eos.facts.bgp_address_family.bgp_address_family import ( + Bgp_afFacts, +) +from ansible_collections.arista.eos.plugins.module_utils.network.eos.facts.bgp_global.bgp_global import ( + Bgp_globalFacts, +) + + +FACT_LEGACY_SUBSETS = dict( + default=Default, hardware=Hardware, interfaces=Interfaces, config=Config +) +FACT_RESOURCE_SUBSETS = dict( + interfaces=InterfacesFacts, + l2_interfaces=L2_interfacesFacts, + l3_interfaces=L3_interfacesFacts, + lacp=LacpFacts, + lacp_interfaces=Lacp_interfacesFacts, + lag_interfaces=Lag_interfacesFacts, + lldp_global=Lldp_globalFacts, + lldp_interfaces=Lldp_interfacesFacts, + vlans=VlansFacts, + acl_interfaces=Acl_interfacesFacts, + acls=AclsFacts, + static_routes=Static_routesFacts, + ospfv2=Ospfv2Facts, + ospfv3=Ospfv3Facts, + ospf_interfaces=Ospf_interfacesFacts, + bgp_address_family=Bgp_afFacts, + bgp_global=Bgp_globalFacts, +) + + +class Facts(FactsBase): + """ The fact class for eos + """ + + VALID_LEGACY_GATHER_SUBSETS = frozenset(FACT_LEGACY_SUBSETS.keys()) + VALID_RESOURCE_SUBSETS = frozenset(FACT_RESOURCE_SUBSETS.keys()) + + def get_facts( + self, legacy_facts_type=None, resource_facts_type=None, data=None + ): + """ Collect the facts for eos + :param legacy_facts_type: List of legacy facts types + :param resource_facts_type: List of resource fact types + :param data: previously collected conf + :rtype: dict + :return: the facts gathered + """ + if self.VALID_RESOURCE_SUBSETS: + self.get_network_resources_facts( + FACT_RESOURCE_SUBSETS, resource_facts_type, data + ) + + if self.VALID_LEGACY_GATHER_SUBSETS: + self.get_network_legacy_facts( + FACT_LEGACY_SUBSETS, legacy_facts_type + ) + + return self.ansible_facts, self._warnings diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/interfaces/__init__.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/interfaces/__init__.py new file mode 100644 index 00000000..e69de29b --- /dev/null +++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/interfaces/__init__.py diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/interfaces/interfaces.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/interfaces/interfaces.py new file mode 100644 index 00000000..c52535a7 --- /dev/null +++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/interfaces/interfaces.py @@ -0,0 +1,110 @@ +# -*- coding: utf-8 -*- +# Copyright 2019 Red Hat +# GNU General Public License v3.0+ +# (see COPYING or https://www.gnu.org/licenses/gpl-3.0.txt) +""" +The eos interfaces fact class +It is in this file the configuration is collected from the device +for a given resource, parsed, and the facts tree is populated +based on the configuration. +""" + +from __future__ import absolute_import, division, print_function + +__metaclass__ = type + +from copy import deepcopy +import re + +from ansible_collections.ansible.netcommon.plugins.module_utils.network.common import ( + utils, +) +from ansible_collections.arista.eos.plugins.module_utils.network.eos.argspec.interfaces.interfaces import ( + InterfacesArgs, +) + + +class InterfacesFacts(object): + """ The eos interfaces fact class + """ + + def __init__(self, module, subspec="config", options="options"): + self._module = module + self.argument_spec = InterfacesArgs.argument_spec + spec = deepcopy(self.argument_spec) + if subspec: + if options: + facts_argument_spec = spec[subspec][options] + else: + facts_argument_spec = spec[subspec] + else: + facts_argument_spec = spec + + self.generated_spec = utils.generate_dict(facts_argument_spec) + + def get_device_data(self, connection): + return connection.get("show running-config | section ^interface") + + def populate_facts(self, connection, ansible_facts, data=None): + """ Populate the facts for interfaces + + :param connection: the device connection + :param ansible_facts: Facts dictionary + :param data: previously collected configuration + :rtype: dictionary + :returns: facts + """ + if not data: + data = self.get_device_data(connection) + + # operate on a collection of resource x + config = ("\n" + data).split("\ninterface ") + objs = [] + for conf in config: + if conf: + obj = self.render_config(self.generated_spec, conf) + if obj: + objs.append(obj) + facts = {"interfaces": []} + if objs: + params = utils.validate_config( + self.argument_spec, {"config": objs} + ) + for cfg in params["config"]: + facts["interfaces"].append(utils.remove_empties(cfg)) + ansible_facts["ansible_network_resources"].update(facts) + + return ansible_facts + + def render_config(self, spec, conf): + """ + Render config as dictionary structure and delete keys from spec for null values + + :param spec: The facts tree, generated from the argspec + :param conf: The configuration + :rtype: dictionary + :returns: The generated config + """ + config = deepcopy(spec) + + # populate the facts from the configuration + config["name"] = re.match(r"(\S+)", conf).group(1) + description = utils.parse_conf_arg(conf, "description") + if description is not None: + config["description"] = description.replace('"', "") + shutdown = utils.parse_conf_cmd_arg(conf, "shutdown", False) + config["enabled"] = shutdown if shutdown is False else True + config["mtu"] = utils.parse_conf_arg(conf, "mtu") + config["mode"] = utils.parse_conf_cmd_arg( + conf, "switchport", "layer2", "layer3" + ) + + state = utils.parse_conf_arg(conf, "speed") + if state: + if state == "auto": + config["duplex"] = state + else: + # remaining options are all e.g., 10half or 40gfull + config["speed"] = state[:-4] + config["duplex"] = state[-4:] + return utils.remove_empties(config) diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/l2_interfaces/__init__.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/l2_interfaces/__init__.py new file mode 100644 index 00000000..e69de29b --- /dev/null +++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/l2_interfaces/__init__.py diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/l2_interfaces/l2_interfaces.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/l2_interfaces/l2_interfaces.py new file mode 100644 index 00000000..4ef5f445 --- /dev/null +++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/l2_interfaces/l2_interfaces.py @@ -0,0 +1,117 @@ +# -*- coding: utf-8 -*- +# Copyright 2019 Red Hat +# GNU General Public License v3.0+ +# (see COPYING or https://www.gnu.org/licenses/gpl-3.0.txt) +""" +The eos l2_interfaces fact class +It is in this file the configuration is collected from the device +for a given resource, parsed, and the facts tree is populated +based on the configuration. +""" + +from __future__ import absolute_import, division, print_function + +__metaclass__ = type + +from copy import deepcopy +import re + +from ansible_collections.ansible.netcommon.plugins.module_utils.network.common import ( + utils, +) +from ansible_collections.arista.eos.plugins.module_utils.network.eos.argspec.l2_interfaces.l2_interfaces import ( + L2_interfacesArgs, +) + + +class L2_interfacesFacts(object): + """ The eos l2_interfaces fact class + """ + + def __init__(self, module, subspec="config", options="options"): + self._module = module + self.argument_spec = L2_interfacesArgs.argument_spec + spec = deepcopy(self.argument_spec) + if subspec: + if options: + facts_argument_spec = spec[subspec][options] + else: + facts_argument_spec = spec[subspec] + else: + facts_argument_spec = spec + + self.generated_spec = utils.generate_dict(facts_argument_spec) + + def get_device_data(self, connection): + return connection.get("show running-config | section ^interface") + + def populate_facts(self, connection, ansible_facts, data=None): + """ Populate the facts for l2_interfaces + + :param connection: the device connection + :param ansible_facts: Facts dictionary + :param data: previously collected configuration + :rtype: dictionary + :returns: facts + """ + if not data: + data = self.get_device_data(connection) + + # operate on a collection of resource x + config = ("\n" + data).split("\ninterface ") + objs = [] + for conf in config: + if conf: + obj = self.render_config(self.generated_spec, conf) + if obj: + objs.append(obj) + facts = {} + if objs: + params = utils.validate_config( + self.argument_spec, {"config": objs} + ) + facts["l2_interfaces"] = [ + utils.remove_empties(cfg) for cfg in params["config"] + ] + ansible_facts["ansible_network_resources"].update(facts) + + return ansible_facts + + def render_config(self, spec, conf): + """ + Render config as dictionary structure and delete keys from spec for null values + + :param spec: The facts tree, generated from the argspec + :param conf: The configuration + :rtype: dictionary + :returns: The generated config + """ + config = deepcopy(spec) + + # populate the facts from the configuration + config["name"] = re.match(r"(\S+)", conf).group(1).replace('"', "") + has_mode = re.search(r"switchport mode (\S+)", conf) + if has_mode: + config["mode"] = has_mode.group(1) + + has_access = re.search(r"switchport access vlan (\d+)", conf) + if has_access: + config["access"] = {"vlan": int(has_access.group(1))} + + has_trunk = re.findall(r"switchport trunk (.+)", conf) + if has_trunk: + trunk = {} + for match in has_trunk: + has_native = re.match(r"native vlan (\d+)", match) + if has_native: + trunk["native_vlan"] = int(has_native.group(1)) + continue + + has_allowed = re.match(r"allowed vlan (\S+)", match) + if has_allowed: + # TODO: listify? + trunk["trunk_allowed_vlans"] = has_allowed.group(1) + continue + config["trunk"] = trunk + + return utils.remove_empties(config) diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/l3_interfaces/__init__.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/l3_interfaces/__init__.py new file mode 100644 index 00000000..e69de29b --- /dev/null +++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/l3_interfaces/__init__.py diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/l3_interfaces/l3_interfaces.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/l3_interfaces/l3_interfaces.py new file mode 100644 index 00000000..afce1445 --- /dev/null +++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/l3_interfaces/l3_interfaces.py @@ -0,0 +1,118 @@ +# -*- coding: utf-8 -*- +# Copyright 2019 Red Hat +# GNU General Public License v3.0+ +# (see COPYING or https://www.gnu.org/licenses/gpl-3.0.txt) +""" +The eos l3_interfaces fact class +It is in this file the configuration is collected from the device +for a given resource, parsed, and the facts tree is populated +based on the configuration. +""" + +from __future__ import absolute_import, division, print_function + +__metaclass__ = type + +from copy import deepcopy +import re + +from ansible_collections.ansible.netcommon.plugins.module_utils.network.common import ( + utils, +) +from ansible_collections.arista.eos.plugins.module_utils.network.eos.argspec.l3_interfaces.l3_interfaces import ( + L3_interfacesArgs, +) + + +class L3_interfacesFacts(object): + """ The eos l3_interfaces fact class + """ + + def __init__(self, module, subspec="config", options="options"): + self._module = module + self.argument_spec = L3_interfacesArgs.argument_spec + spec = deepcopy(self.argument_spec) + if subspec: + if options: + facts_argument_spec = spec[subspec][options] + else: + facts_argument_spec = spec[subspec] + else: + facts_argument_spec = spec + + self.generated_spec = utils.generate_dict(facts_argument_spec) + + def get_device_data(self, connection): + return connection.get("show running-config | section ^interface") + + def populate_facts(self, connection, ansible_facts, data=None): + """ Populate the facts for l3_interfaces + :param connection: the device connection + :param ansible_facts: Facts dictionary + :param data: previously collected configuration + :rtype: dictionary + :returns: facts + """ + if not data: + data = self.get_device_data(connection) + + # split the config into instances of the resource + resource_delim = "interface" + find_pattern = r"(?:^|\n)%s.*?(?=(?:^|\n)%s|$)" % ( + resource_delim, + resource_delim, + ) + resources = [ + p.strip() for p in re.findall(find_pattern, data, re.DOTALL) + ] + + objs = [] + for resource in resources: + if resource: + obj = self.render_config(self.generated_spec, resource) + if obj: + objs.append(obj) + facts = {} + if objs: + params = utils.validate_config( + self.argument_spec, {"config": objs} + ) + facts["l3_interfaces"] = [ + utils.remove_empties(cfg) for cfg in params["config"] + ] + ansible_facts["ansible_network_resources"].update(facts) + return ansible_facts + + def render_config(self, spec, conf): + """ + Render config as dictionary structure and delete keys + from spec for null values + + :param spec: The facts tree, generated from the argspec + :param conf: The configuration + :rtype: dictionary + :returns: The generated config + """ + config = deepcopy(spec) + + config["name"] = utils.parse_conf_arg(conf, "interface") + + matches = re.findall(r".*ip address (.+)$", conf, re.MULTILINE) + if matches: + config["ipv4"] = [] + for match in matches: + address, dummy, remainder = match.partition(" ") + ipv4 = {"address": address} + if remainder == "secondary": + ipv4["secondary"] = True + config["ipv4"].append(ipv4) + + matches = re.findall(r".*ipv6 address (.+)$", conf, re.MULTILINE) + if matches: + config["ipv6"] = [] + for match in matches: + address, dummy, remainder = match.partition(" ") + ipv6 = {"address": address} + config["ipv6"].append(ipv6) + + return utils.remove_empties(config) diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/lacp/__init__.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/lacp/__init__.py new file mode 100644 index 00000000..e69de29b --- /dev/null +++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/lacp/__init__.py diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/lacp/lacp.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/lacp/lacp.py new file mode 100644 index 00000000..2c70188e --- /dev/null +++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/lacp/lacp.py @@ -0,0 +1,102 @@ +# -*- coding: utf-8 -*- +# Copyright 2019 Red Hat +# GNU General Public License v3.0+ +# (see COPYING or https://www.gnu.org/licenses/gpl-3.0.txt) +""" +The eos lacp fact class +It is in this file the configuration is collected from the device +for a given resource, parsed, and the facts tree is populated +based on the configuration. +""" + +from __future__ import absolute_import, division, print_function + +__metaclass__ = type + +import re +from copy import deepcopy + +from ansible_collections.ansible.netcommon.plugins.module_utils.network.common import ( + utils, +) +from ansible_collections.arista.eos.plugins.module_utils.network.eos.argspec.lacp.lacp import ( + LacpArgs, +) + + +class LacpFacts(object): + """ The eos lacp fact class + """ + + def __init__(self, module, subspec="config", options="options"): + self._module = module + self.argument_spec = LacpArgs.argument_spec + spec = deepcopy(self.argument_spec) + if subspec: + if options: + facts_argument_spec = spec[subspec][options] + else: + facts_argument_spec = spec[subspec] + else: + facts_argument_spec = spec + + self.generated_spec = utils.generate_dict(facts_argument_spec) + + def get_device_data(self, connection): + return connection.get("show running-config | section ^lacp") + + def populate_facts(self, connection, ansible_facts, data=None): + """ Populate the facts for lacp + :param connection: the device connection + :param ansible_facts: Facts dictionary + :param data: previously collected configuration + :rtype: dictionary + :returns: facts + """ + if not data: + data = self.get_device_data(connection) + + # split the config into instances of the resource + resource_delim = "lacp" + find_pattern = r"(?:^|\n)%s.*?(?=(?:^|\n)%s|$)" % ( + resource_delim, + resource_delim, + ) + resources = [ + p.strip() for p in re.findall(find_pattern, data, re.DOTALL) + ] + + objs = {} + for resource in resources: + if resource: + obj = self.render_config(self.generated_spec, resource) + if obj: + objs.update(obj) + + ansible_facts["ansible_network_resources"].pop("lacp", None) + facts = {"lacp": {}} + if objs: + params = utils.validate_config( + self.argument_spec, {"config": objs} + ) + facts["lacp"] = utils.remove_empties(params["config"]) + + ansible_facts["ansible_network_resources"].update(facts) + return ansible_facts + + def render_config(self, spec, conf): + """ + Render config as dictionary structure and delete keys + from spec for null values + + :param spec: The facts tree, generated from the argspec + :param conf: The configuration + :rtype: dictionary + :returns: The generated config + """ + config = deepcopy(spec) + config["system"]["priority"] = utils.parse_conf_arg( + conf, "system-priority" + ) + + return utils.remove_empties(config) diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/lacp_interfaces/__init__.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/lacp_interfaces/__init__.py new file mode 100644 index 00000000..e69de29b --- /dev/null +++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/lacp_interfaces/__init__.py diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/lacp_interfaces/lacp_interfaces.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/lacp_interfaces/lacp_interfaces.py new file mode 100644 index 00000000..82c42114 --- /dev/null +++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/lacp_interfaces/lacp_interfaces.py @@ -0,0 +1,104 @@ +# -*- coding: utf-8 -*- +# Copyright 2019 Red Hat +# GNU General Public License v3.0+ +# (see COPYING or https://www.gnu.org/licenses/gpl-3.0.txt) +""" +The eos lacp_interfaces fact class +It is in this file the configuration is collected from the device +for a given resource, parsed, and the facts tree is populated +based on the configuration. +""" + +from __future__ import absolute_import, division, print_function + +__metaclass__ = type + +import re +from copy import deepcopy + +from ansible_collections.ansible.netcommon.plugins.module_utils.network.common import ( + utils, +) +from ansible_collections.arista.eos.plugins.module_utils.network.eos.argspec.lacp_interfaces.lacp_interfaces import ( + Lacp_interfacesArgs, +) + + +class Lacp_interfacesFacts(object): + """ The eos lacp_interfaces fact class + """ + + def __init__(self, module, subspec="config", options="options"): + self._module = module + self.argument_spec = Lacp_interfacesArgs.argument_spec + spec = deepcopy(self.argument_spec) + if subspec: + if options: + facts_argument_spec = spec[subspec][options] + else: + facts_argument_spec = spec[subspec] + else: + facts_argument_spec = spec + + self.generated_spec = utils.generate_dict(facts_argument_spec) + + def get_device_data(self, connection): + return connection.get("show running-config | section lacp") + + def populate_facts(self, connection, ansible_facts, data=None): + """ Populate the facts for lacp_interfaces + :param connection: the device connection + :param ansible_facts: Facts dictionary + :param data: previously collected configuration + :rtype: dictionary + :returns: facts + """ + if not data: + data = self.get_device_data(connection) + + # split the config into instances of the resource + resource_delim = "interface" + find_pattern = r"(?:^|\n)%s.*?(?=(?:^|\n)%s|$)" % ( + resource_delim, + resource_delim, + ) + resources = [ + p.strip() for p in re.findall(find_pattern, data, re.DOTALL) + ] + + objs = [] + for resource in resources: + if resource: + obj = self.render_config(self.generated_spec, resource) + if obj: + objs.append(obj) + + ansible_facts["ansible_network_resources"].pop("lacp_interfaces", None) + facts = {} + if objs: + params = utils.validate_config( + self.argument_spec, {"config": objs} + ) + facts["lacp_interfaces"] = [ + utils.remove_empties(cfg) for cfg in params["config"] + ] + + ansible_facts["ansible_network_resources"].update(facts) + return ansible_facts + + def render_config(self, spec, conf): + """ + Render config as dictionary structure and delete keys + from spec for null values + + :param spec: The facts tree, generated from the argspec + :param conf: The configuration + :rtype: dictionary + :returns: The generated config + """ + config = deepcopy(spec) + config["name"] = utils.parse_conf_arg(conf, "interface") + config["port_priority"] = utils.parse_conf_arg(conf, "port-priority") + config["rate"] = utils.parse_conf_arg(conf, "rate") + + return utils.remove_empties(config) diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/lag_interfaces/__init__.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/lag_interfaces/__init__.py new file mode 100644 index 00000000..e69de29b --- /dev/null +++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/lag_interfaces/__init__.py diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/lag_interfaces/lag_interfaces.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/lag_interfaces/lag_interfaces.py new file mode 100644 index 00000000..fe96b65d --- /dev/null +++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/lag_interfaces/lag_interfaces.py @@ -0,0 +1,119 @@ +# -*- coding: utf-8 -*- +# Copyright 2019 Red Hat +# GNU General Public License v3.0+ +# (see COPYING or https://www.gnu.org/licenses/gpl-3.0.txt) +""" +The eos lag_interfaces fact class +It is in this file the configuration is collected from the device +for a given resource, parsed, and the facts tree is populated +based on the configuration. +""" + +from __future__ import absolute_import, division, print_function + +__metaclass__ = type + +from copy import deepcopy +import re + +from ansible_collections.ansible.netcommon.plugins.module_utils.network.common import ( + utils, +) +from ansible_collections.arista.eos.plugins.module_utils.network.eos.argspec.lag_interfaces.lag_interfaces import ( + Lag_interfacesArgs, +) + + +class Lag_interfacesFacts(object): + """ The eos lag_interfaces fact class + """ + + def __init__(self, module, subspec="config", options="options"): + self._module = module + self.argument_spec = Lag_interfacesArgs.argument_spec + spec = deepcopy(self.argument_spec) + if subspec: + if options: + facts_argument_spec = spec[subspec][options] + else: + facts_argument_spec = spec[subspec] + else: + facts_argument_spec = spec + + self.generated_spec = utils.generate_dict(facts_argument_spec) + + def get_device_data(self, connection): + return connection.get("show running-config | section ^interface") + + def populate_facts(self, connection, ansible_facts, data=None): + """ Populate the facts for lag_interfaces + :param connection: the device connection + :param ansible_facts: Facts dictionary + :param data: previously collected configuration + :rtype: dictionary + :returns: facts + """ + if not data: + data = self.get_device_data(connection) + + # split the config into instances of the resource + resource_delim = "interface" + find_pattern = r"(?:^|\n)%s.*?(?=(?:^|\n)%s|$)" % ( + resource_delim, + resource_delim, + ) + resources = [ + p.strip() for p in re.findall(find_pattern, data, re.DOTALL) + ] + + objs = {} + for resource in resources: + if resource: + obj = self.render_config(self.generated_spec, resource) + if obj: + group_name = obj["name"] + if group_name in objs and "members" in obj: + config = objs[group_name] + if "members" not in config: + config["members"] = [] + objs[group_name]["members"].extend(obj["members"]) + else: + objs[group_name] = obj + objs = list(objs.values()) + facts = {"lag_interfaces": []} + if objs: + params = utils.validate_config( + self.argument_spec, {"config": objs} + ) + facts["lag_interfaces"] = [ + utils.remove_empties(cfg) for cfg in params["config"] + ] + ansible_facts["ansible_network_resources"].update(facts) + return ansible_facts + + def render_config(self, spec, conf): + """ + Render config as dictionary structure and delete keys + from spec for null values + + :param spec: The facts tree, generated from the argspec + :param conf: The configuration + :rtype: dictionary + :returns: The generated config + """ + config = deepcopy(spec) + interface_name = utils.parse_conf_arg(conf, "interface") + if interface_name.startswith("Port-Channel"): + config["name"] = interface_name + return utils.remove_empties(config) + + interface = {"member": interface_name} + match = re.match( + r".*channel-group (\d+) mode (\S+)", conf, re.MULTILINE | re.DOTALL + ) + if match: + config["name"], interface["mode"] = match.groups() + config["name"] = "Port-Channel" + config["name"] + config["members"] = [interface] + + return utils.remove_empties(config) diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/legacy/__init__.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/legacy/__init__.py new file mode 100644 index 00000000..e69de29b --- /dev/null +++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/legacy/__init__.py diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/legacy/base.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/legacy/base.py new file mode 100644 index 00000000..8dc886f3 --- /dev/null +++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/legacy/base.py @@ -0,0 +1,181 @@ +# -*- coding: utf-8 -*- +# Copyright 2019 Red Hat +# 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 + +import platform +import re + +from ansible.module_utils.six import iteritems +from ansible_collections.arista.eos.plugins.module_utils.network.eos.eos import ( + run_commands, + get_capabilities, +) + + +class FactsBase(object): + + COMMANDS = frozenset() + + def __init__(self, module): + self.module = module + self.warnings = list() + self.facts = dict() + self.responses = None + + def populate(self): + self.responses = run_commands( + self.module, list(self.COMMANDS), check_rc=False + ) + + +class Default(FactsBase): + + SYSTEM_MAP = {"serialNumber": "serialnum"} + + COMMANDS = ["show version | json", "show hostname | json"] + + def populate(self): + super(Default, self).populate() + data = self.responses[0] + for key, value in iteritems(self.SYSTEM_MAP): + if key in data: + self.facts[value] = data[key] + + self.facts.update(self.responses[1]) + self.facts.update(self.platform_facts()) + + def platform_facts(self): + platform_facts = {} + + resp = get_capabilities(self.module) + device_info = resp["device_info"] + + platform_facts["system"] = device_info["network_os"] + + for item in ("model", "image", "version", "platform", "hostname"): + val = device_info.get("network_os_%s" % item) + if val: + platform_facts[item] = val + + platform_facts["api"] = resp["network_api"] + platform_facts["python_version"] = platform.python_version() + + return platform_facts + + +class Hardware(FactsBase): + + COMMANDS = ["dir all-filesystems", "show version | json"] + + def populate(self): + super(Hardware, self).populate() + self.facts.update(self.populate_filesystems()) + self.facts.update(self.populate_memory()) + + def populate_filesystems(self): + data = self.responses[0] + + if isinstance(data, dict): + data = data["messages"][0] + + fs = re.findall(r"^Directory of (.+)/", data, re.M) + return dict(filesystems=fs) + + def populate_memory(self): + values = self.responses[1] + return dict( + memfree_mb=int(values["memFree"]) / 1024, + memtotal_mb=int(values["memTotal"]) / 1024, + ) + + +class Config(FactsBase): + + COMMANDS = ["show running-config"] + + def populate(self): + super(Config, self).populate() + self.facts["config"] = self.responses[0] + + +class Interfaces(FactsBase): + + INTERFACE_MAP = { + "description": "description", + "physicalAddress": "macaddress", + "mtu": "mtu", + "bandwidth": "bandwidth", + "duplex": "duplex", + "lineProtocolStatus": "lineprotocol", + "interfaceStatus": "operstatus", + "forwardingModel": "type", + } + + COMMANDS = ["show interfaces | json", "show lldp neighbors | json"] + + def populate(self): + super(Interfaces, self).populate() + + self.facts["all_ipv4_addresses"] = list() + self.facts["all_ipv6_addresses"] = list() + + data = self.responses[0] + if data and "LLDP is not enabled" not in data: + self.facts["interfaces"] = self.populate_interfaces(data) + + if len(self.responses) > 1: + data = self.responses[1] + if data: + self.facts["neighbors"] = self.populate_neighbors( + data["lldpNeighbors"] + ) + + def populate_interfaces(self, data): + facts = dict() + for key, value in iteritems(data["interfaces"]): + intf = dict() + + for remote, local in iteritems(self.INTERFACE_MAP): + if remote in value: + intf[local] = value[remote] + + if "interfaceAddress" in value: + intf["ipv4"] = dict() + for entry in value["interfaceAddress"]: + intf["ipv4"]["address"] = entry["primaryIp"]["address"] + intf["ipv4"]["masklen"] = entry["primaryIp"]["maskLen"] + self.add_ip_address(entry["primaryIp"]["address"], "ipv4") + + if "interfaceAddressIp6" in value: + intf["ipv6"] = dict() + for entry in value["interfaceAddressIp6"]["globalUnicastIp6s"]: + intf["ipv6"]["address"] = entry["address"] + intf["ipv6"]["subnet"] = entry["subnet"] + self.add_ip_address(entry["address"], "ipv6") + + facts[key] = intf + + return facts + + def add_ip_address(self, address, family): + if family == "ipv4": + self.facts["all_ipv4_addresses"].append(address) + else: + self.facts["all_ipv6_addresses"].append(address) + + def populate_neighbors(self, neighbors): + facts = dict() + for value in neighbors: + port = value["port"] + if port not in facts: + facts[port] = list() + lldp = dict() + lldp["host"] = value["neighborDevice"] + lldp["port"] = value["neighborPort"] + facts[port].append(lldp) + return facts diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/lldp_global/__init__.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/lldp_global/__init__.py new file mode 100644 index 00000000..e69de29b --- /dev/null +++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/lldp_global/__init__.py diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/lldp_global/lldp_global.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/lldp_global/lldp_global.py new file mode 100644 index 00000000..0a9774f5 --- /dev/null +++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/lldp_global/lldp_global.py @@ -0,0 +1,96 @@ +# +# -*- coding: utf-8 -*- +# Copyright 2019 Red Hat +# GNU General Public License v3.0+ +# (see COPYING or https://www.gnu.org/licenses/gpl-3.0.txt) +""" +The eos lldp_global fact class +It is in this file the configuration is collected from the device +for a given resource, parsed, and the facts tree is populated +based on the configuration. +""" + +from __future__ import absolute_import, division, print_function + +__metaclass__ = type + +import re +from copy import deepcopy + +from ansible_collections.ansible.netcommon.plugins.module_utils.network.common import ( + utils, +) +from ansible_collections.arista.eos.plugins.module_utils.network.eos.argspec.lldp_global.lldp_global import ( + Lldp_globalArgs, +) + + +class Lldp_globalFacts(object): + """ The eos lldp_global fact class + """ + + def __init__(self, module, subspec="config", options="options"): + self._module = module + self.argument_spec = Lldp_globalArgs.argument_spec + spec = deepcopy(self.argument_spec) + if subspec: + if options: + facts_argument_spec = spec[subspec][options] + else: + facts_argument_spec = spec[subspec] + else: + facts_argument_spec = spec + + self.generated_spec = utils.generate_dict(facts_argument_spec) + + def get_device_data(self, connection): + return connection.get("show running-config | section lldp") + + def populate_facts(self, connection, ansible_facts, data=None): + """ Populate the facts for lldp_global + :param connection: the device connection + :param ansible_facts: Facts dictionary + :param data: previously collected conf + :rtype: dictionary + :returns: facts + """ + if not data: + data = self.get_device_data(connection) + + obj = {} + if data: + obj.update(self.render_config(self.generated_spec, data)) + + ansible_facts["ansible_network_resources"].pop("lldp_global", None) + facts = {} + if obj: + params = utils.validate_config(self.argument_spec, {"config": obj}) + facts["lldp_global"] = utils.remove_empties(params["config"]) + else: + facts["lldp_global"] = {} + + ansible_facts["ansible_network_resources"].update(facts) + return ansible_facts + + def render_config(self, spec, conf): + """ + Render config as dictionary structure and delete keys + from spec for null values + + :param spec: The facts tree, generated from the argspec + :param conf: The configuration + :rtype: dictionary + :returns: The generated config + """ + config = deepcopy(spec) + config["holdtime"] = utils.parse_conf_arg(conf, "holdtime") + config["reinit"] = utils.parse_conf_arg(conf, "reinit") + config["timer"] = utils.parse_conf_arg(conf, "timer") + + for match in re.findall( + r"^(no)? lldp tlv-select (\S+)", conf, re.MULTILINE + ): + tlv_option = match[1].replace("-", "_") + config["tlv_select"][tlv_option] = bool(match[0] != "no") + + return utils.remove_empties(config) diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/lldp_interfaces/__init__.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/lldp_interfaces/__init__.py new file mode 100644 index 00000000..e69de29b --- /dev/null +++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/lldp_interfaces/__init__.py diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/lldp_interfaces/lldp_interfaces.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/lldp_interfaces/lldp_interfaces.py new file mode 100644 index 00000000..fb1b938e --- /dev/null +++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/lldp_interfaces/lldp_interfaces.py @@ -0,0 +1,103 @@ +# -*- coding: utf-8 -*- +# Copyright 2019 Red Hat +# GNU General Public License v3.0+ +# (see COPYING or https://www.gnu.org/licenses/gpl-3.0.txt) +""" +The eos lldp_interfaces fact class +It is in this file the configuration is collected from the device +for a given resource, parsed, and the facts tree is populated +based on the configuration. +""" + +from __future__ import absolute_import, division, print_function + +__metaclass__ = type + +import re +from copy import deepcopy + +from ansible_collections.ansible.netcommon.plugins.module_utils.network.common import ( + utils, +) +from ansible_collections.arista.eos.plugins.module_utils.network.eos.argspec.lldp_interfaces.lldp_interfaces import ( + Lldp_interfacesArgs, +) + + +class Lldp_interfacesFacts(object): + """ The eos lldp_interfaces fact class + """ + + def __init__(self, module, subspec="config", options="options"): + self._module = module + self.argument_spec = Lldp_interfacesArgs.argument_spec + spec = deepcopy(self.argument_spec) + if subspec: + if options: + facts_argument_spec = spec[subspec][options] + else: + facts_argument_spec = spec[subspec] + else: + facts_argument_spec = spec + + self.generated_spec = utils.generate_dict(facts_argument_spec) + + def populate_facts(self, connection, ansible_facts, data=None): + """ Populate the facts for lldp_interfaces + :param connection: the device connection + :param ansible_facts: Facts dictionary + :param data: previously collected configuration + :rtype: dictionary + :returns: facts + """ + if not data: + data = connection.get("show running-config | section lldp") + + # split the config into instances of the resource + resource_delim = "interface" + find_pattern = r"(?:^|\n)%s.*?(?=(?:^|\n)%s|$)" % ( + resource_delim, + resource_delim, + ) + resources = [ + p.strip() for p in re.findall(find_pattern, data, re.DOTALL) + ] + + objs = [] + for resource in resources: + if resource: + obj = self.render_config(self.generated_spec, resource) + if obj: + objs.append(obj) + + ansible_facts["ansible_network_resources"].pop("lldp_interfaces", None) + facts = {} + if objs: + params = utils.validate_config( + self.argument_spec, {"config": objs} + ) + facts["lldp_interfaces"] = [ + utils.remove_empties(cfg) for cfg in params["config"] + ] + + ansible_facts["ansible_network_resources"].update(facts) + return ansible_facts + + def render_config(self, spec, conf): + """ + Render config as dictionary structure and delete keys + from spec for null values + + :param spec: The facts tree, generated from the argspec + :param conf: The configuration + :rtype: dictionary + :returns: The generated config + """ + config = deepcopy(spec) + config["name"] = utils.parse_conf_arg(conf, "interface") + + matches = re.findall(r"(no )?lldp (\S+)", conf) + for match in matches: + config[match[1]] = not bool(match[0]) + + return utils.remove_empties(config) diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/ospf_interfaces/__init__.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/ospf_interfaces/__init__.py new file mode 100644 index 00000000..e69de29b --- /dev/null +++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/ospf_interfaces/__init__.py diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/ospf_interfaces/ospf_interfaces.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/ospf_interfaces/ospf_interfaces.py new file mode 100644 index 00000000..16d99fd0 --- /dev/null +++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/ospf_interfaces/ospf_interfaces.py @@ -0,0 +1,103 @@ +# -*- coding: utf-8 -*- +# Copyright 2020 Red Hat +# 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 + +""" +The eos ospf_interfaces fact class +It is in this file the configuration is collected from the device +for a given resource, parsed, and the facts tree is populated +based on the configuration. +""" + +import re + +from ansible_collections.ansible.netcommon.plugins.module_utils.network.common import ( + utils, +) +from ansible_collections.arista.eos.plugins.module_utils.network.eos.rm_templates.ospf_interfaces import ( + Ospf_interfacesTemplate, +) +from ansible_collections.arista.eos.plugins.module_utils.network.eos.argspec.ospf_interfaces.ospf_interfaces import ( + Ospf_interfacesArgs, +) + + +class Ospf_interfacesFacts(object): + """ The eos ospf_interfaces facts class + """ + + def __init__(self, module, subspec="config", options="options"): + self._module = module + self.argument_spec = Ospf_interfacesArgs.argument_spec + + def get_config(self, connection): + """Wrapper method for `connection.get()` + This method exists solely to allow the unit test framework to mock device connection calls. + """ + return connection.get("show running-config | section interface ") + + def populate_facts(self, connection, ansible_facts, data=None): + """ Populate the facts for Ospf_interfaces network resource + + :param connection: the device connection + :param ansible_facts: Facts dictionary + :param data: previously collected conf + + :rtype: dictionary + :returns: facts + """ + facts = {} + + if not data: + data = self.get_config(connection) + + resource_delim = "interface" + find_pattern = r"(?:^|\n)%s.*?(?=(?:^|\n)%s|$)" % ( + resource_delim, + resource_delim, + ) + resources = [ + p.strip() for p in re.findall(find_pattern, data, re.DOTALL) + ] + # parse native config using the Ospf_interfaces template + ospf_interfaces_facts = [] + for resource in resources: + ospf_interfaces_parser = Ospf_interfacesTemplate( + lines=resource.splitlines() + ) + entry = ospf_interfaces_parser.parse() + if entry: + if "address_family" in entry and entry["address_family"]: + entry["address_family"] = sorted( + list(entry["address_family"].values()), + key=lambda k, sk="afi": k[sk], + ) + if entry: + if entry.get("address_family"): + for addr in entry["address_family"]: + if "ip_params" in addr: + addr["ip_params"] = sorted( + list(addr["ip_params"].values()), + key=lambda k, sk="afi": k[sk], + ) + ospf_interfaces_facts.append(entry) + + ansible_facts["ansible_network_resources"].pop("ospf_interfaces", None) + facts = {"ospf_interfaces": []} + params = utils.remove_empties( + utils.validate_config( + self.argument_spec, {"config": ospf_interfaces_facts} + ) + ) + + if params.get("config"): + for cfg in params["config"]: + facts["ospf_interfaces"].append(utils.remove_empties(cfg)) + ansible_facts["ansible_network_resources"].update(facts) + + return ansible_facts diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/ospfv2/__init__.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/ospfv2/__init__.py new file mode 100644 index 00000000..e69de29b --- /dev/null +++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/ospfv2/__init__.py diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/ospfv2/ospfv2.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/ospfv2/ospfv2.py new file mode 100644 index 00000000..a7419eaf --- /dev/null +++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/ospfv2/ospfv2.py @@ -0,0 +1,496 @@ +# +# -*- coding: utf-8 -*- +# Copyright 2019 Red Hat +# 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 + +""" +The eos ospfv2 fact class +It is in this file the configuration is collected from the device +for a given resource, parsed, and the facts tree is populated +based on the configuration. +""" +import re +from copy import deepcopy + +from ansible_collections.ansible.netcommon.plugins.module_utils.network.common import ( + utils, +) +from ansible_collections.arista.eos.plugins.module_utils.network.eos.argspec.ospfv2.ospfv2 import ( + Ospfv2Args, +) + + +class Ospfv2Facts(object): + """ The eos ospfv2 fact class + """ + + def __init__(self, module, subspec="config", options="options"): + self._module = module + self.argument_spec = Ospfv2Args.argument_spec + spec = deepcopy(self.argument_spec) + if subspec: + if options: + facts_argument_spec = spec[subspec][options] + else: + facts_argument_spec = spec[subspec] + else: + facts_argument_spec = spec + + self.generated_spec = utils.generate_dict(facts_argument_spec) + + def get_device_data(self, connection): + return connection.get("show running-config | section ospf") + + def populate_facts(self, connection, ansible_facts, data=None): + """ Populate the facts for ospfv2 + :param connection: the device connection + :param ansible_facts: Facts dictionary + :param data: previously collected conf + :rtype: dictionary + :returns: facts + """ + + if not data: + data = self.get_device_data(connection) + + # split the config into instances of the resource + resource_delim = "router ospf" + find_pattern = r"(?:^|\n)%s.*?(?=(?:^|\n)%s|$)" % ( + resource_delim, + resource_delim, + ) + resources = [ + p.strip() for p in re.findall(find_pattern, data, re.DOTALL) + ] + objs_list = [] + objs = {} + for resource in resources: + if resource and "router ospfv3" not in resource: + obj = self.render_config(self.generated_spec, resource) + if obj: + objs_list.append(obj) + objs = {"processes": objs_list} + ansible_facts["ansible_network_resources"].pop("ospfv2", None) + + facts = {} + if objs: + facts["ospfv2"] = {} + params = utils.validate_config( + self.argument_spec, {"config": objs} + ) + facts["ospfv2"].update(utils.remove_empties(params["config"])) + + ansible_facts["ansible_network_resources"].update(facts) + return ansible_facts + + def render_config(self, spec, conf): + """ + Render config as dictionary structure and delete keys + from spec for null values + + :param spec: The facts tree, generated from the argspec + :param conf: The configuration + :rtype: dictionary + :returns: The generated config + """ + instance_list = [] + ospf_params_dict = {} + areas_list = [] + distance_dict = {} + network_list = [] + redistribute_list = [] + timers_list = [] + areas_list = [] + for dev_config in conf.split("\n"): + if not dev_config: + continue + network_dict = {} + redistribute_dict = {} + dev_config = dev_config.strip() + dev_config = re.sub(r"-", "_", dev_config).strip() + matches = re.findall(r"router (ospf) (.*)", dev_config) + if matches: + if ospf_params_dict: + instance_list.append(ospf_params_dict) + ospf_params_dict = {} + instance = matches[0][1].split() + ospf_params_dict.update({"process_id": str(instance[0])}) + if "vrf" in dev_config: + vrf_name = instance[-1] + else: + vrf_name = None + ospf_params_dict.update({"vrf": vrf_name}) + if "traffic_engineering" in dev_config: + ospf_params_dict.update({"traffic_engineering": True}) + config_params = dev_config.split() + if config_params[0] == "adjacency": + threshold = config_params[-1] + adjacency_dict = {"exchange_start": {"threshold": threshold}} + ospf_params_dict.update({"adjacency": adjacency_dict}) + elif "auto_cost" in dev_config: + bw = config_params[-1] + ospf_params_dict.update( + {"auto_cost": {"reference_bandwidth": bw}} + ) + elif "bfd" in dev_config: + ospf_params_dict.update({"bfd": {"all_interfaces": True}}) + elif config_params[0] == "default_information": + def_dict = {"originate": True} + for i, val in enumerate(config_params[2::]): + if val == "always": + def_dict.update({"always": True}) + elif val in ["route_map", "metric", "metric_type"]: + def_dict.update({val: config_params[i + 3]}) + ospf_params_dict.update({"default_information": def_dict}) + elif "default_metric" in dev_config: + ospf_params_dict.update({"default_metric": config_params[-1]}) + elif "distance" in dev_config: + distance_dict.update({config_params[-2]: config_params[-1]}) + ospf_params_dict.update({"distance": distance_dict}) + elif "distribute_list" in dev_config: + ospf_params_dict.update( + {"distribute_list": {config_params[1]: config_params[2]}} + ) + elif "dn_bit_ignore" in dev_config: + ospf_params_dict.update({"dn_bit_ignore": True}) + elif "fips_restrictions" in dev_config: + ospf_params_dict.update({"fips_restrictions": True}) + elif "graceful_restart" in dev_config: + if "grace_period" in dev_config: + ospf_params_dict.update( + { + "graceful_restart": { + "grace_period": config_params[-1] + } + } + ) + else: + ospf_params_dict.update( + {"graceful_restart": {"set": True}} + ) + elif "graceful_restart_helper" in dev_config: + ospf_params_dict.update({"graceful_restart_helper": True}) + elif "log_adjacency_changes" in dev_config: + detail = True if "detail" in dev_config else False + ospf_params_dict.update( + {"log_adjacency_changes": {"detail": detail}} + ) + elif "max_lsa" in dev_config: + max_lsa_dict = {} + config_params.pop(0) + max_lsa_dict.update({"count": config_params.pop(0)}) + if config_params: + if config_params[0].isdigit(): + max_lsa_dict.update( + {"threshold": config_params.pop(0)} + ) + for i, el in enumerate(config_params): + if el == "warning_only": + max_lsa_dict.update({"warning": True}) + if el in ["ignore_count", "ignore_time", "reset_time"]: + max_lsa_dict.update({el: config_params[i + 1]}) + ospf_params_dict.update({"max_lsa": max_lsa_dict}) + elif "maximum_paths" in dev_config: + ospf_params_dict.update({"maximum_paths": config_params[1]}) + elif "mpls ldp sync default" in dev_config: + ospf_params_dict.update({"mpls_ldp": True}) + elif config_params[0] == "network": + config_params.pop(0) + prefix = re.search(r"\/", config_params[0]) + if prefix: + network_dict.update({"prefix": config_params.pop(0)}) + else: + network_dict.update( + {"network_address": config_params.pop(0)} + ) + network_dict.update({"mask": config_params.pop(0)}) + network_dict.update({"area": config_params[-1]}) + network_list.append(network_dict) + ospf_params_dict.update({"networks": network_list}) + elif "passive_interface" in dev_config: + if config_params[1] == "default": + ospf_params_dict.update( + {"passive_interface": {"default": True}} + ) + else: + ospf_params_dict.update( + { + "passive_interface": { + "interface_list": config_params[1] + } + } + ) + elif "point_to_point" in dev_config: + ospf_params_dict.update({"point_to_point": True}) + elif "redistribute" in dev_config: + redistribute_dict.update({"routes": config_params[1]}) + if config_params[1] == "isis": + if "level" in config_params[2]: + k = re.sub(r"_", "-", config_params[2]) + redistribute_dict.update({"isis_level": k}) + if "route_map" in dev_config: + redistribute_dict.update({"route_map": config_params[-1]}) + redistribute_list.append(redistribute_dict) + ospf_params_dict.update({"redistribute": redistribute_list}) + elif "router_id" in dev_config: + ospf_params_dict.update({"router_id": config_params[-1]}) + elif "retransmission_threshold" in dev_config: + ospf_params_dict.update( + {"retransmission_threshold": config_params[-1]} + ) + elif config_params[0] == "compatible": + ospf_params_dict.update({"rfc1583compatibility": True}) + elif "shutdown" in dev_config: + ospf_params_dict.update({"shutdown": True}) + elif "summary_address" in dev_config: + summary_address_dict = {} + config_params.pop(0) + prefix = re.search(r"\/", config_params[0]) + if prefix: + summary_address_dict.update( + {"prefix": config_params.pop(0)} + ) + else: + summary_address_dict.update( + {"address": config_params.pop(0)} + ) + summary_address_dict.update({"mask": config_params.pop(0)}) + if "not_advertise" in dev_config: + summary_address_dict.update({"not_advertise": True}) + config_params.pop(0) + else: + if config_params: + summary_address_dict.update( + {config_params[0]: config_params[1]} + ) + ospf_params_dict.update( + {"summary_address": summary_address_dict} + ) + elif "timers" in dev_config: + timers_dict = {} + if config_params[1] == "lsa": + if config_params[2] == "rx": + timers_dict.update( + { + "lsa": { + "rx": {"min_interval": config_params[-1]} + } + } + ) + else: + timers_dict.update( + { + "lsa": { + "tx": { + "delay": { + "initial": config_params[-3], + "min": config_params[-2], + "max": config_params[-1], + } + } + } + } + ) + elif config_params[1] == "out_delay": + timers_dict.update({"out_delay": config_params[-1]}) + elif config_params[1] == "pacing": + timers_dict.update({"pacing": config_params[-1]}) + elif config_params[1] == "spf": + if config_params[2] == "delay": + timers_dict.update( + { + "spf": { + "tx": { + "delay": { + "initial": config_params[-3], + "min": config_params[-2], + "max": config_params[-1], + } + } + } + } + ) + else: + timers_dict.update( + {"spf": {"seconds": config_params[-1]}} + ) + timers_list.append(timers_dict) + ospf_params_dict.update({"timers": timers_list}) + elif config_params[0] == "area": + areas_dict = {} + areas_dict.update({"area_id": config_params[1]}) + if config_params[2] == "default_cost": + areas_dict.update({"default_cost": config_params[-1]}) + elif config_params[2] == "filter": + prefix = re.search(r"\/", config_params[3]) + if prefix: + areas_dict.update( + {"filter": {"address": config_params[3]}} + ) + elif config_params[3] == "prefix_list": + areas_dict.update( + {"filter": {"prefix_list": config_params[-1]}} + ) + else: + areas_dict.update( + {"filter": {"subnet_address": config_params[3]}} + ) + areas_dict.update( + {"filter": {"subnet_mask": config_params[4]}} + ) + elif config_params[2] == "not_so_stubby": + if len(config_params) == 3: + areas_dict.update({"not_so_stubby": {"set": True}}) + continue + if config_params[3] == "lsa": + areas_dict.update({"not_so_stubby": {"lsa": True}}) + elif config_params[3] == "default_information_originate": + default_dict = {} + for i, val in enumerate(config_params): + if val == "nssa_only": + default_dict.update({"nssa_only": True}) + if val == "metric_type": + default_dict.update( + {"metric_type": config_params[i + 1]} + ) + if val == "metric": + default_dict.update( + {"metric": config_params[i + 1]} + ) + areas_dict.update( + { + "not_so_stubby": { + "default_information_originate": default_dict + } + } + ) + elif config_params[3] == "no_summary": + areas_dict.update( + {"not_so_stubby": {"no_summary": True}} + ) + elif config_params[3] == "nssa_only": + areas_dict.update( + {"not_so_stubby": {"nssa_only": True}} + ) + elif config_params[2] == "nssa": + if len(config_params) == 3: + areas_dict.update({"nssa": {"set": True}}) + continue + if config_params[3] == "default_information_originate": + default_dict = {} + for i, val in enumerate(config_params): + if val == "nssa_only": + default_dict.update({"nssa_only": True}) + if val == "metric_type": + default_dict.update( + {"metric_type": config_params[i + 1]} + ) + if val == "metric": + default_dict.update( + {"metric": config_params[i + 1]} + ) + areas_dict.update( + { + "nssa": { + "default_information_originate": default_dict + } + } + ) + elif config_params[3] == "no_summary": + areas_dict.update({"nssa": {"no_summary": True}}) + elif config_params[3] == "nssa_only": + areas_dict.update({"nssa": {"nssa_only": True}}) + elif config_params[2] == "range": + prefix = re.search(r"\/", config_params[3]) + range_dict = {} + if prefix: + range_dict.update({"address": config_params[3]}) + else: + range_dict.update({"subnet_address": config_params[3]}) + range_dict.update({"subnet_mask": config_params[4]}) + if "advertise" in dev_config: + range_dict.update({"advertise": True}) + if "not_advertise" in dev_config: + range_dict.update({"advertise": False}) + if "cost" in dev_config: + range_dict.update({"cost": config_params[-1]}) + areas_dict.update({"range": range_dict}) + elif config_params[2] == "stub": + if "no_summary" in dev_config: + areas_dict.update({"stub": {"no_summary": True}}) + else: + areas_dict.update({"stub": {"set": True}}) + areas_list.append(areas_dict) + ospf_params_dict.update({"areas": areas_list}) + elif config_params[0] == "max_metric": + config_params.pop(0) + router_lsa_dict = {} + config_params.pop(0) + if not config_params: + ospf_params_dict.update( + {"max_metric": {"router_lsa": {"set": True}}} + ) + else: + for i, val in enumerate(config_params): + if val == "include_stub": + router_lsa_dict.update({"include_stub": True}) + elif val == "on_startup": + if config_params[i + 1] == "wait_for_bgp": + router_lsa_dict.update( + {"on_startup": {"wait_for_bgp": True}} + ) + else: + router_lsa_dict.update( + { + "on_startup": { + "time": config_params[i + 1] + } + } + ) + elif val == "external_lsa": + if ( + i < len(config_params) + and config_params[i + 1].isdigit() + ): + router_lsa_dict.update( + { + "external_lsa": { + "max_metric_value": config_params[ + i + 1 + ] + } + } + ) + else: + router_lsa_dict.update( + {"external_lsa": {"set": True}} + ) + elif val == "summary_lsa": + if ( + i < len(config_params) - 1 + and config_params[i + 1].isdigit() + ): + router_lsa_dict.update( + { + "summary_lsa": { + "max_metric_value": config_params[ + i + 1 + ] + } + } + ) + else: + router_lsa_dict.update( + {"summary_lsa": {"set": True}} + ) + ospf_params_dict.update( + {"max_metric": {"router_lsa": router_lsa_dict}} + ) + # instance_list.append(ospf_params_dict) + # config.update({"ospf_version": "v2", "ospf_processes": instance_list}) + return utils.remove_empties(ospf_params_dict) diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/ospfv3/__init__.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/ospfv3/__init__.py new file mode 100644 index 00000000..e69de29b --- /dev/null +++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/ospfv3/__init__.py diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/ospfv3/ospfv3.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/ospfv3/ospfv3.py new file mode 100644 index 00000000..adef556f --- /dev/null +++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/ospfv3/ospfv3.py @@ -0,0 +1,115 @@ +# -*- coding: utf-8 -*- +# Copyright 2020 Red Hat +# 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 + +""" +The eos ospfv3 fact class +It is in this file the configuration is collected from the device +for a given resource, parsed, and the facts tree is populated +based on the configuration. +""" + +import re +from copy import deepcopy + +from ansible_collections.ansible.netcommon.plugins.module_utils.network.common import ( + utils, +) +from ansible_collections.arista.eos.plugins.module_utils.network.eos.rm_templates.ospfv3 import ( + Ospfv3Template, +) +from ansible_collections.arista.eos.plugins.module_utils.network.eos.argspec.ospfv3.ospfv3 import ( + Ospfv3Args, +) + + +class Ospfv3Facts(object): + """ The eos ospfv3 facts class + """ + + def __init__(self, module, subspec="config", options="options"): + self._module = module + self.argument_spec = Ospfv3Args.argument_spec + spec = deepcopy(self.argument_spec) + if subspec: + if options: + facts_argument_spec = spec[subspec][options] + else: + facts_argument_spec = spec[subspec] + else: + facts_argument_spec = spec + + self.generated_spec = utils.generate_dict(facts_argument_spec) + + def get_config(self, connection): + """Wrapper method for `connection.get()` + This method exists solely to allow the unit test framework to mock device connection calls. + """ + return connection.get("show running-config | section ospfv3") + + def populate_facts(self, connection, ansible_facts, data=None): + """ Populate the facts for Ospfv3 network resource + + :param connection: the device connection + :param ansible_facts: Facts dictionary + :param data: previously collected conf + + :rtype: dictionary + :returns: facts + """ + facts = {} + objs = [] + + if not data: + data = self.get_config(connection) + + # split the config into instances of the resource + resource_delim = "router ospfv3" + find_pattern = r"(?:^|\n)%s.*?(?=(?:^|\n)%s|$)" % ( + resource_delim, + resource_delim, + ) + resources = [ + p.strip() for p in re.findall(find_pattern, data, re.DOTALL) + ] + + # parse native config using the Ospfv3 template + ospfv3_facts = {"processes": []} + + for resource in resources: + ospfv3_parser = Ospfv3Template(lines=resource.splitlines()) + objs = ospfv3_parser.parse() + + for key, sortv in [("address_family", "afi")]: + if key in objs["processes"] and objs["processes"][key]: + objs["processes"][key] = list( + objs["processes"][key].values() + ) + + for addr_family in objs["processes"]["address_family"]: + if "areas" in addr_family: + addr_family["areas"] = list(addr_family["areas"].values()) + + for addr_family in objs["processes"]["address_family"]: + if not addr_family.get("afi"): + # global vars + objs["processes"].update(addr_family) + objs["processes"]["address_family"].remove(addr_family) + + ospfv3_facts["processes"].append(objs["processes"]) + + ansible_facts["ansible_network_resources"].pop("ospfv3", None) + params = utils.validate_config( + self.argument_spec, {"config": ospfv3_facts} + ) + params = utils.remove_empties(params) + + facts["ospfv3"] = params.get("config", []) + ansible_facts["ansible_network_resources"].update(facts) + + return ansible_facts diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/static_routes/__init__.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/static_routes/__init__.py new file mode 100644 index 00000000..e69de29b --- /dev/null +++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/static_routes/__init__.py diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/static_routes/static_routes.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/static_routes/static_routes.py new file mode 100644 index 00000000..6ae9cc2e --- /dev/null +++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/static_routes/static_routes.py @@ -0,0 +1,237 @@ +# +# -*- coding: utf-8 -*- +# Copyright 2019 Red Hat +# GNU General Public License v3.0+ +# (see COPYING or https://www.gnu.org/licenses/gpl-3.0.txt) +""" +The eos static_routes fact class +It is in this file the configuration is collected from the device +for a given resource, parsed, and the facts tree is populated +based on the configuration. +""" + +from __future__ import absolute_import, division, print_function + +__metaclass__ = type + +import re +from copy import deepcopy + +from ansible_collections.ansible.netcommon.plugins.module_utils.network.common import ( + utils, +) +from ansible_collections.arista.eos.plugins.module_utils.network.eos.argspec.static_routes.static_routes import ( + Static_routesArgs, +) + + +class Static_routesFacts(object): + """ The eos static_routes fact class + """ + + def __init__(self, module, subspec="config", options="options"): + self._module = module + self.argument_spec = Static_routesArgs.argument_spec + spec = deepcopy(self.argument_spec) + if subspec: + if options: + facts_argument_spec = spec[subspec][options] + else: + facts_argument_spec = spec[subspec] + else: + facts_argument_spec = spec + + self.generated_spec = utils.generate_dict(facts_argument_spec) + + def get_device_data(self, connection): + return connection.get("show running-config | grep route") + + def populate_facts(self, connection, ansible_facts, data=None): + """ Populate the facts for static_routes + :param connection: the device connection + :param ansible_facts: Facts dictionary + :param data: previously collected conf + :rtype: dictionary + :returns: facts + """ + if not data: + data = self.get_device_data(connection) + + # split the config into instances of the resource + resource_delim = "ip.* route" + find_pattern = r"(?:^|\n)%s.*?(?=(?:^|\n)%s|$)" % ( + resource_delim, + resource_delim, + ) + resources = [p.strip() for p in re.findall(find_pattern, data)] + resources_without_vrf = [] + resource_vrf = {} + for resource in resources: + if resource and "vrf" not in resource: + resources_without_vrf.append(resource) + else: + vrf = re.search(r"ip(v6)* route vrf (.*?) .*", resource) + if vrf.group(2) in resource_vrf.keys(): + vrf_val = resource_vrf[vrf.group(2)] + vrf_val.append(resource) + resource_vrf.update({vrf.group(2): vrf_val}) + else: + resource_vrf.update({vrf.group(2): [resource]}) + resources_without_vrf = ["\n".join(resources_without_vrf)] + for vrf in resource_vrf.keys(): + vrflist = ["\n".join(resource_vrf[vrf])] + resource_vrf.update({vrf: vrflist}) + objs = [] + for resource in resources_without_vrf: + if resource: + obj = self.render_config(self.generated_spec, resource) + if obj: + objs.append(obj) + for resource in resource_vrf.keys(): + if resource: + obj = self.render_config( + self.generated_spec, resource_vrf[resource][0] + ) + if obj: + objs.append(obj) + ansible_facts["ansible_network_resources"].pop("static_routes", None) + facts = {} + if objs: + facts["static_routes"] = [] + params = utils.validate_config( + self.argument_spec, {"config": objs} + ) + for cfg in params["config"]: + facts["static_routes"].append(utils.remove_empties(cfg)) + ansible_facts["ansible_network_resources"].update(facts) + return ansible_facts + + def render_config(self, spec, conf): + """ + Render config as dictionary structure and delete keys + from spec for null values + + :param spec: The facts tree, generated from the argspec + :param conf: The configuration + :rtype: dictionary + :returns: The generated config + """ + config = deepcopy(spec) + address_family_dict = {} + route_dict = {} + dest_list = [] + afi_list = [] + vrf_list = [] + routes = [] + config["address_families"] = [] + next_hops = {} + interface_list = [ + "Ethernet", + "Loopback", + "Management", + "Port-Channel", + "Tunnel", + "Vlan", + "Vxlan", + "vtep", + ] + conf_list = conf.split("\n") + for conf_elem in conf_list: + matches = re.findall( + r"(ip|ipv6) route ([\d\.\/:]+|vrf) (.+)$", conf_elem + ) + if matches: + remainder = matches[0][2].split() + route_update = False + if matches[0][1] == "vrf": + vrf = remainder.pop(0) + # new vrf + if vrf not in vrf_list and vrf_list: + route_dict.update({"next_hops": next_hops}) + routes.append(route_dict) + address_family_dict.update({"routes": routes}) + config["address_families"].append(address_family_dict) + route_update = True + config.update({"vrf": vrf}) + vrf_list.append(vrf) + dest = remainder.pop(0) + else: + config["vrf"] = None + dest = matches[0][1] + afi = "ipv4" if matches[0][0] == "ip" else "ipv6" + if afi not in afi_list: + if afi_list and not route_update: + # new afi and not the first updating all prev configs + route_dict.update({"next_hops": next_hops}) + routes.append(route_dict) + address_family_dict.update({"routes": routes}) + config["address_families"].append(address_family_dict) + route_update = True + address_family_dict = {} + address_family_dict.update({"afi": afi}) + routes = [] + afi_list.append(afi) + # To check the format of the dest + prefix = re.search(r"/", dest) + if not prefix: + dest = dest + " " + remainder.pop(0) + if dest not in dest_list: + # For new dest and not the first dest + if dest_list and not route_update: + route_dict.update({"next_hops": next_hops}) + routes.append(route_dict) + dest_list.append(dest) + next_hops = [] + route_dict = {} + route_dict.update({"dest": dest}) + nexthops = {} + nxthop_addr = re.search(r"[\.\:]", remainder[0]) + if nxthop_addr: + nexthops.update({"interface": remainder.pop(0)}) + if remainder and remainder[0] == "label": + nexthops.update({"mpls_label": remainder.pop(1)}) + remainder.pop(0) + elif re.search(r"Nexthop-Group", remainder[0]): + nexthops.update({"nexthop_grp": remainder.pop(1)}) + remainder.pop(0) + else: + interface = remainder.pop(0) + if interface in interface_list: + interface = interface + " " + remainder.pop(0) + nexthops.update({"interface": interface}) + for attribute in remainder: + forward_addr = re.search( + r"([\dA-Fa-f]+[:\.]+)+[\dA-Fa-f]+", attribute + ) + if forward_addr: + nexthops.update( + { + "forward_router_address": remainder.pop( + remainder.index(attribute) + ) + } + ) + for attribute in remainder: + for params in ["tag", "name", "track"]: + if attribute == params: + keyname = params + if attribute == "name": + keyname = "description" + nexthops.update( + { + keyname: remainder.pop( + remainder.index(attribute) + 1 + ) + } + ) + remainder.pop(remainder.index(attribute)) + if remainder: + metric = re.search(r"\d+", remainder[0]) + if metric: + nexthops.update({"admin_distance": remainder.pop(0)}) + next_hops.append(nexthops) + route_dict.update({"next_hops": next_hops}) + routes.append(route_dict) + address_family_dict.update({"routes": routes}) + config["address_families"].append(address_family_dict) + return utils.remove_empties(config) diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/vlans/__init__.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/vlans/__init__.py new file mode 100644 index 00000000..e69de29b --- /dev/null +++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/vlans/__init__.py diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/vlans/vlans.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/vlans/vlans.py new file mode 100644 index 00000000..0605fdb6 --- /dev/null +++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/facts/vlans/vlans.py @@ -0,0 +1,120 @@ +# -*- coding: utf-8 -*- +# Copyright 2019 Red Hat +# GNU General Public License v3.0+ +# (see COPYING or https://www.gnu.org/licenses/gpl-3.0.txt) +""" +The eos vlans fact class +It is in this file the configuration is collected from the device +for a given resource, parsed, and the facts tree is populated +based on the configuration. +""" + +from __future__ import absolute_import, division, print_function + +__metaclass__ = type + +from copy import deepcopy +import re + +from ansible_collections.ansible.netcommon.plugins.module_utils.network.common import ( + utils, +) +from ansible_collections.arista.eos.plugins.module_utils.network.eos.argspec.vlans.vlans import ( + VlansArgs, +) + + +class VlansFacts(object): + """ The eos vlans fact class + """ + + def __init__(self, module, subspec="config", options="options"): + self._module = module + self.argument_spec = VlansArgs.argument_spec + spec = deepcopy(self.argument_spec) + if subspec: + if options: + facts_argument_spec = spec[subspec][options] + else: + facts_argument_spec = spec[subspec] + else: + facts_argument_spec = spec + + self.generated_spec = utils.generate_dict(facts_argument_spec) + + def populate_facts(self, connection, ansible_facts, data=None): + """ Populate the facts for vlans + :param connection: the device connection + :param ansible_facts: Facts dictionary + :param data: previously collected conf + :rtype: dictionary + :returns: facts + """ + if not data: + data = connection.get("show running-config | section ^vlan") + + # split the config into instances of the resource + resource_delim = "vlan" + find_pattern = r"(?:^|\n)%s.*?(?=(?:^|\n)%s|$)" % ( + resource_delim, + resource_delim, + ) + resources = [ + p.strip() for p in re.findall(find_pattern, data, re.DOTALL) + ] + + objs = [] + for resource in resources: + if resource: + obj = self.render_config(self.generated_spec, resource) + if obj: + objs.extend(obj) + + ansible_facts["ansible_network_resources"].pop("vlans", None) + facts = {} + if objs: + params = utils.validate_config( + self.argument_spec, {"config": objs} + ) + facts["vlans"] = [ + utils.remove_empties(cfg) for cfg in params["config"] + ] + + ansible_facts["ansible_network_resources"].update(facts) + return ansible_facts + + def render_config(self, spec, conf): + """ + Render config as dictionary structure and delete keys + from spec for null values + + :param spec: The facts tree, generated from the argspec + :param conf: The configuration + :rtype: dictionary + :returns: The generated config + """ + config = deepcopy(spec) + vlans = [] + + vlan_list = vlan_to_list(utils.parse_conf_arg(conf, "vlan")) + for vlan in vlan_list: + config["vlan_id"] = vlan + config["name"] = utils.parse_conf_arg(conf, "name") + config["state"] = utils.parse_conf_arg(conf, "state") + if config["state"] is None: + config["state"] = "active" + vlans.append(utils.remove_empties(config)) + + return vlans + + +def vlan_to_list(vlan_str): + vlans = [] + for vlan in vlan_str.split(","): + if "-" in vlan: + start, stop = vlan.split("-") + vlans.extend(range(int(start), int(stop) + 1)) + else: + vlans.append(int(vlan)) + + return vlans diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/providers/__init__.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/providers/__init__.py new file mode 100644 index 00000000..e69de29b --- /dev/null +++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/providers/__init__.py diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/providers/cli/__init__.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/providers/cli/__init__.py new file mode 100644 index 00000000..e69de29b --- /dev/null +++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/providers/cli/__init__.py diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/providers/cli/config/__init__.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/providers/cli/config/__init__.py new file mode 100644 index 00000000..e69de29b --- /dev/null +++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/providers/cli/config/__init__.py diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/providers/cli/config/bgp/__init__.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/providers/cli/config/bgp/__init__.py new file mode 100644 index 00000000..e69de29b --- /dev/null +++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/providers/cli/config/bgp/__init__.py diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/providers/cli/config/bgp/address_family.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/providers/cli/config/bgp/address_family.py new file mode 100644 index 00000000..76524b0a --- /dev/null +++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/providers/cli/config/bgp/address_family.py @@ -0,0 +1,143 @@ +# +# (c) 2019, Ansible by Red Hat, inc +# 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 +import re + +from ansible.module_utils.six import iteritems +from ansible_collections.ansible.netcommon.plugins.module_utils.network.common.utils import ( + to_list, +) +from ansible_collections.arista.eos.plugins.module_utils.network.eos.providers.providers import ( + CliProvider, +) +from ansible_collections.arista.eos.plugins.module_utils.network.eos.providers.cli.config.bgp.neighbors import ( + AFNeighbors, +) + + +class AddressFamily(CliProvider): + def render(self, config=None): + commands = list() + safe_list = list() + + router_context = "router bgp %s" % self.get_value("config.bgp_as") + context_config = None + + for item in self.get_value("config.address_family"): + context = "address-family %s" % item["afi"] + context_commands = list() + + if config: + context_path = [router_context, context] + context_config = self.get_config_context( + config, context_path, indent=2 + ) + + for key, value in iteritems(item): + if value is not None: + meth = getattr(self, "_render_%s" % key, None) + if meth: + resp = meth(item, context_config) + if resp: + context_commands.extend(to_list(resp)) + + if context_commands: + commands.append(context) + commands.extend(context_commands) + commands.append("exit") + + safe_list.append(context) + + if self.params["operation"] == "replace": + if config: + resp = self._negate_config(config, safe_list) + commands.extend(resp) + + return commands + + def _negate_config(self, config, safe_list=None): + commands = list() + matches = re.findall(r"(address-family .+)$", config, re.M) + for item in set(matches).difference(safe_list): + commands.append("no %s" % item) + return commands + + def _render_auto_summary(self, item, config=None): + cmd = "auto-summary" + if item["auto_summary"] is False: + cmd = "no %s" % cmd + if not config or cmd not in config: + return cmd + + def _render_synchronization(self, item, config=None): + cmd = "synchronization" + if item["synchronization"] is False: + cmd = "no %s" % cmd + if not config or cmd not in config: + return cmd + + def _render_networks(self, item, config=None): + commands = list() + safe_list = list() + + for entry in item["networks"]: + network = entry["prefix"] + if entry["masklen"]: + network = "%s/%s" % (entry["prefix"], entry["masklen"]) + safe_list.append(network) + + cmd = "network %s" % network + + if entry["route_map"]: + cmd += " route-map %s" % entry["route_map"] + + if not config or cmd not in config: + commands.append(cmd) + + if self.params["operation"] == "replace": + if config: + matches = re.findall(r"network (\S+)", config, re.M) + for entry in set(matches).difference(safe_list): + commands.append("no network %s" % entry) + + return commands + + def _render_redistribute(self, item, config=None): + commands = list() + safe_list = list() + + for entry in item["redistribute"]: + option = entry["protocol"] + + cmd = "redistribute %s" % entry["protocol"] + + if entry["route_map"]: + cmd += " route-map %s" % entry["route_map"] + + if not config or cmd not in config: + commands.append(cmd) + + safe_list.append(option) + + if self.params["operation"] == "replace": + if config: + matches = re.findall( + r"redistribute (\S+)(?:\s*)(\d*)", config, re.M + ) + for i in range(0, len(matches)): + matches[i] = " ".join(matches[i]).strip() + for entry in set(matches).difference(safe_list): + commands.append("no redistribute %s" % entry) + + return commands + + def _render_neighbors(self, item, config): + """ generate bgp neighbor configuration + """ + return AFNeighbors(self.params).render( + config, nbr_list=item["neighbors"] + ) diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/providers/cli/config/bgp/neighbors.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/providers/cli/config/bgp/neighbors.py new file mode 100644 index 00000000..2dbcb81a --- /dev/null +++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/providers/cli/config/bgp/neighbors.py @@ -0,0 +1,193 @@ +# +# (c) 2019, Ansible by Red Hat, inc +# 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 +import re + +from ansible.module_utils.six import iteritems +from ansible_collections.ansible.netcommon.plugins.module_utils.network.common.utils import ( + to_list, +) +from ansible_collections.arista.eos.plugins.module_utils.network.eos.providers.providers import ( + CliProvider, +) + + +class Neighbors(CliProvider): + def render(self, config=None, nbr_list=None): + commands = list() + safe_list = list() + if not nbr_list: + nbr_list = self.get_value("config.neighbors") + + for item in nbr_list: + neighbor_commands = list() + context = "neighbor %s" % item["neighbor"] + cmd = "%s remote-as %s" % (context, item["remote_as"]) + + if not config or cmd not in config: + neighbor_commands.append(cmd) + + for key, value in iteritems(item): + if value is not None: + meth = getattr(self, "_render_%s" % key, None) + if meth: + resp = meth(item, config) + if resp: + neighbor_commands.extend(to_list(resp)) + commands.extend(neighbor_commands) + safe_list.append(context) + + if self.params["operation"] == "replace": + if config and safe_list: + commands.extend(self._negate_config(config, safe_list)) + + return commands + + def _negate_config(self, config, safe_list=None): + commands = list() + matches = re.findall(r"(neighbor \S+)", config, re.M) + for item in set(matches).difference(safe_list): + commands.append("no %s" % item) + return commands + + def _render_description(self, item, config=None): + cmd = "neighbor %s description %s" % ( + item["neighbor"], + item["description"], + ) + if not config or cmd not in config: + return cmd + + def _render_enabled(self, item, config=None): + cmd = "neighbor %s shutdown" % item["neighbor"] + if item["enabled"] is True: + if not config or cmd in config: + cmd = "no %s" % cmd + return cmd + elif not config or cmd not in config: + return cmd + + def _render_update_source(self, item, config=None): + cmd = "neighbor %s update-source %s" % ( + item["neighbor"], + item["update_source"], + ) + if not config or cmd not in config: + return cmd + + def _render_password(self, item, config=None): + cmd = "neighbor %s password %s" % (item["neighbor"], item["password"]) + if not config or cmd not in config: + return cmd + + def _render_ebgp_multihop(self, item, config=None): + cmd = "neighbor %s ebgp-multihop %s" % ( + item["neighbor"], + item["ebgp_multihop"], + ) + if not config or cmd not in config: + return cmd + + def _render_peer_group(self, item, config=None): + cmd = "neighbor %s peer-group %s" % ( + item["neighbor"], + item["peer_group"], + ) + if not config or cmd not in config: + return cmd + + def _render_route_reflector_client(self, item, config=None): + cmd = "neighbor %s route-reflector-client" % item["neighbor"] + if item["route_reflector_client"] is False: + if not config or cmd in config: + cmd = "no %s" % cmd + return cmd + elif not config or cmd not in config: + return cmd + + def _render_maximum_prefix(self, item, config=None): + cmd = "neighbor %s maximum-routes %s" % ( + item["neighbor"], + item["maximum_prefix"], + ) + if not config or cmd not in config: + return cmd + + def _render_remove_private_as(self, item, config=None): + cmd = "neighbor %s remove-private-AS" % item["neighbor"] + if item["remove_private_as"] is False: + if not config or cmd in config: + cmd = "no %s" % cmd + return cmd + elif not config or cmd not in config: + return cmd + + def _render_timers(self, item, config): + """generate bgp timer related configuration + """ + keepalive = item["timers"]["keepalive"] + holdtime = item["timers"]["holdtime"] + neighbor = item["neighbor"] + + if keepalive and holdtime: + cmd = "neighbor %s timers %s %s" % (neighbor, keepalive, holdtime) + if not config or cmd not in config: + return cmd + + +class AFNeighbors(CliProvider): + def render(self, config=None, nbr_list=None): + commands = list() + if not nbr_list: + return + + for item in nbr_list: + neighbor_commands = list() + for key, value in iteritems(item): + if value is not None: + meth = getattr(self, "_render_%s" % key, None) + if meth: + resp = meth(item, config) + if resp: + neighbor_commands.extend(to_list(resp)) + + commands.extend(neighbor_commands) + + return commands + + def _render_activate(self, item, config=None): + cmd = "neighbor %s activate" % item["neighbor"] + if item["activate"] is False: + if not config or cmd in config: + cmd = "no %s" % cmd + return cmd + elif not config or cmd not in config: + return cmd + + def _render_default_originate(self, item, config=None): + cmd = "neighbor %s default-originate" % item["neighbor"] + if item["default_originate"] is False: + if not config or cmd in config: + cmd = "no %s" % cmd + return cmd + elif not config or cmd not in config: + return cmd + + def _render_graceful_restart(self, item, config=None): + cmd = "neighbor %s graceful-restart" % item["neighbor"] + if item["graceful_restart"] is False: + cmd = "no " + cmd + if config: + config_el = [x.strip() for x in config.split("\n")] + if cmd in config_el: + return + return cmd + + def _render_weight(self, item, config=None): + cmd = "neighbor %s weight %s" % (item["neighbor"], item["weight"]) + if not config or cmd not in config: + return cmd diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/providers/cli/config/bgp/process.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/providers/cli/config/bgp/process.py new file mode 100644 index 00000000..bc75c0a2 --- /dev/null +++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/providers/cli/config/bgp/process.py @@ -0,0 +1,185 @@ +# +# (c) 2019, Ansible by Red Hat, inc +# 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 +import re + +from ansible.module_utils.six import iteritems +from ansible_collections.ansible.netcommon.plugins.module_utils.network.common.utils import ( + to_list, +) +from ansible_collections.arista.eos.plugins.module_utils.network.eos.providers.providers import ( + register_provider, +) +from ansible_collections.arista.eos.plugins.module_utils.network.eos.providers.providers import ( + CliProvider, +) +from ansible_collections.arista.eos.plugins.module_utils.network.eos.providers.cli.config.bgp.neighbors import ( + Neighbors, +) +from ansible_collections.arista.eos.plugins.module_utils.network.eos.providers.cli.config.bgp.address_family import ( + AddressFamily, +) + +REDISTRIBUTE_PROTOCOLS = frozenset( + ["ospf", "ospf3", "rip", "isis", "static", "connected"] +) + + +@register_provider("eos", "eos_bgp") +class Provider(CliProvider): + def render(self, config=None): + commands = list() + + existing_as = None + if config: + match = re.search(r"router bgp (\d+)", config, re.M) + if match: + existing_as = match.group(1) + + operation = self.params["operation"] + + context = None + if self.params["config"]: + context = "router bgp %s" % self.get_value("config.bgp_as") + + if operation == "delete": + if existing_as: + commands.append("no router bgp %s" % existing_as) + elif context: + commands.append("no %s" % context) + + else: + self._validate_input(config) + if operation == "replace": + if existing_as and int(existing_as) != self.get_value( + "config.bgp_as" + ): + commands.append("no router bgp %s" % existing_as) + config = None + + elif operation == "override": + if existing_as: + commands.append("no router bgp %s" % existing_as) + config = None + + context_commands = list() + + for key, value in iteritems(self.get_value("config")): + if value is not None: + meth = getattr(self, "_render_%s" % key, None) + if meth: + resp = meth(config) + if resp: + context_commands.extend(to_list(resp)) + + if context and context_commands: + commands.append(context) + commands.extend(context_commands) + commands.append("exit") + return commands + + def _render_router_id(self, config=None): + cmd = "router-id %s" % self.get_value("config.router_id") + if not config or cmd not in config: + return cmd + + def _render_log_neighbor_changes(self, config=None): + cmd = "bgp log-neighbor-changes" + log_neighbor_changes = self.get_value("config.log_neighbor_changes") + if log_neighbor_changes is True: + if not config or cmd not in config: + return cmd + elif log_neighbor_changes is False: + if config and cmd in config: + return "no %s" % cmd + + def _render_networks(self, config=None): + commands = list() + safe_list = list() + + for entry in self.get_value("config.networks"): + network = entry["prefix"] + if entry["masklen"]: + network = "%s/%s" % (entry["prefix"], entry["masklen"]) + safe_list.append(network) + + cmd = "network %s" % network + + if entry["route_map"]: + cmd += " route-map %s" % entry["route_map"] + + if not config or cmd not in config: + commands.append(cmd) + + if self.params["operation"] == "replace": + if config: + matches = re.findall(r"network (\S+)", config, re.M) + for entry in set(matches).difference(safe_list): + commands.append("no network %s" % entry) + + return commands + + def _render_redistribute(self, config=None): + commands = list() + safe_list = list() + + for entry in self.get_value("config.redistribute"): + option = entry["protocol"] + + cmd = "redistribute %s" % entry["protocol"] + + if entry["route_map"]: + cmd += " route-map %s" % entry["route_map"] + + if not config or cmd not in config: + commands.append(cmd) + + safe_list.append(option) + + if self.params["operation"] == "replace": + if config: + matches = re.findall( + r"redistribute (\S+)(?:\s*)(\d*)", config, re.M + ) + for i in range(0, len(matches)): + matches[i] = " ".join(matches[i]).strip() + for entry in set(matches).difference(safe_list): + commands.append("no redistribute %s" % entry) + + return commands + + def _render_neighbors(self, config): + """ generate bgp neighbor configuration + """ + return Neighbors(self.params).render(config) + + def _render_address_family(self, config): + """ generate address-family configuration + """ + return AddressFamily(self.params).render(config) + + def _validate_input(self, config): + def device_has_AF(config): + return re.search(r"address-family (?:.*)", config) + + address_family = self.get_value("config.address_family") + root_networks = self.get_value("config.networks") + operation = self.params["operation"] + + if operation == "replace" and root_networks: + if address_family: + for item in address_family: + if item["networks"]: + raise ValueError( + "operation is replace but provided both root level networks and networks under %s address family" + % item["afi"] + ) + + if config and device_has_AF(config): + raise ValueError( + "operation is replace and device has one or more address family activated but root level network(s) provided" + ) diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/providers/module.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/providers/module.py new file mode 100644 index 00000000..2325fcfd --- /dev/null +++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/providers/module.py @@ -0,0 +1,72 @@ +# +# (c) 2019, Ansible by Red Hat, inc +# 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 +from ansible.module_utils.basic import AnsibleModule +from ansible.module_utils.connection import Connection +from ansible_collections.arista.eos.plugins.module_utils.network.eos.providers import ( + providers, +) +from ansible.module_utils._text import to_text + + +class NetworkModule(AnsibleModule): + + fail_on_missing_provider = True + + def __init__(self, connection=None, *args, **kwargs): + super(NetworkModule, self).__init__(*args, **kwargs) + + if connection is None: + connection = Connection(self._socket_path) + + self.connection = connection + + @property + def provider(self): + if not hasattr(self, "_provider"): + capabilities = self.from_json(self.connection.get_capabilities()) + + network_os = capabilities["device_info"]["network_os"] + network_api = capabilities["network_api"] + + if network_api == "cliconf": + connection_type = "network_cli" + + cls = providers.get( + network_os, self._name.split(".")[-1], connection_type + ) + + if not cls: + msg = ( + "unable to find suitable provider for network os %s" + % network_os + ) + if self.fail_on_missing_provider: + self.fail_json(msg=msg) + else: + self.warn(msg) + + obj = cls(self.params, self.connection, self.check_mode) + + setattr(self, "_provider", obj) + + return getattr(self, "_provider") + + def get_facts(self, subset=None): + try: + self.provider.get_facts(subset) + except Exception as exc: + self.fail_json(msg=to_text(exc)) + + def edit_config(self, config_filter=None): + current_config = self.connection.get_config(flags=config_filter) + try: + commands = self.provider.edit_config(current_config) + changed = bool(commands) + return {"commands": commands, "changed": changed} + except Exception as exc: + self.fail_json(msg=to_text(exc)) diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/providers/providers.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/providers/providers.py new file mode 100644 index 00000000..a7dbdaa5 --- /dev/null +++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/providers/providers.py @@ -0,0 +1,128 @@ +# +# (c) 2019, Ansible by Red Hat, inc +# 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 +import json + +from threading import RLock + +from ansible.module_utils.six import itervalues +from ansible_collections.ansible.netcommon.plugins.module_utils.network.common.utils import ( + to_list, +) +from ansible_collections.ansible.netcommon.plugins.module_utils.network.common.config import ( + NetworkConfig, +) + + +_registered_providers = {} +_provider_lock = RLock() + + +def register_provider(network_os, module_name): + def wrapper(cls): + _provider_lock.acquire() + try: + if network_os not in _registered_providers: + _registered_providers[network_os] = {} + for ct in cls.supported_connections: + if ct not in _registered_providers[network_os]: + _registered_providers[network_os][ct] = {} + for item in to_list(module_name): + for entry in itervalues(_registered_providers[network_os]): + entry[item] = cls + finally: + _provider_lock.release() + return cls + + return wrapper + + +def get(network_os, module_name, connection_type): + network_os_providers = _registered_providers.get(network_os) + if network_os_providers is None: + raise ValueError("unable to find a suitable provider for this module") + if connection_type not in network_os_providers: + raise ValueError("provider does not support this connection type") + elif module_name not in network_os_providers[connection_type]: + raise ValueError("could not find a suitable provider for this module") + return network_os_providers[connection_type][module_name] + + +class ProviderBase(object): + + supported_connections = () + + def __init__(self, params, connection=None, check_mode=False): + self.params = params + self.connection = connection + self.check_mode = check_mode + + @property + def capabilities(self): + if not hasattr(self, "_capabilities"): + resp = self.from_json(self.connection.get_capabilities()) + setattr(self, "_capabilities", resp) + return getattr(self, "_capabilities") + + def get_value(self, path): + params = self.params.copy() + for key in path.split("."): + params = params[key] + return params + + def get_facts(self, subset=None): + raise NotImplementedError(self.__class__.__name__) + + def edit_config(self): + raise NotImplementedError(self.__class__.__name__) + + +class CliProvider(ProviderBase): + + supported_connections = ("network_cli",) + + @property + def capabilities(self): + if not hasattr(self, "_capabilities"): + resp = self.from_json(self.connection.get_capabilities()) + setattr(self, "_capabilities", resp) + return getattr(self, "_capabilities") + + def get_config_context(self, config, path, indent=2): + if config is not None: + netcfg = NetworkConfig(indent=indent, contents=config) + try: + config = netcfg.get_block_config(to_list(path)) + except ValueError: + config = None + return config + + def render(self, config=None): + raise NotImplementedError(self.__class__.__name__) + + def cli(self, command): + try: + if not hasattr(self, "_command_output"): + setattr(self, "_command_output", {}) + return self._command_output[command] + except KeyError: + out = self.connection.get(command) + try: + out = json.loads(out) + except ValueError: + pass + self._command_output[command] = out + return out + + def get_facts(self, subset=None): + return self.populate() + + def edit_config(self, config=None): + commands = self.render(config) + if commands and self.check_mode is False: + self.connection.edit_config(commands) + return commands diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/rm_templates/__init__.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/rm_templates/__init__.py new file mode 100644 index 00000000..e69de29b --- /dev/null +++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/rm_templates/__init__.py diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/rm_templates/bgp_address_family.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/rm_templates/bgp_address_family.py new file mode 100644 index 00000000..d54d6bbc --- /dev/null +++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/rm_templates/bgp_address_family.py @@ -0,0 +1,661 @@ +# -*- coding: utf-8 -*- +# Copyright 2020 Red Hat +# 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 + +""" +The Bgp_global parser templates file. This contains +a list of parser definitions and associated functions that +facilitates both facts gathering and native command generation for +the given network resource. +""" + +import re + +from ansible_collections.ansible.netcommon.plugins.module_utils.network.common.network_template import ( + NetworkTemplate, +) + + +def _tmplt_router_bgp_cmd(config_data): + command = "router bgp {as_number}".format(**config_data) + return command + + +def _tmplt_bgp_address_family(config_data): + command = "" + if config_data.get("vrf"): + command = "vrf {vrf}\n".format(**config_data) + command += "address-family {afi}".format(**config_data) + if config_data.get("safi"): + command += " {safi}".format(**config_data) + return command + + +def _tmplt_bgp_params(config_data): + command = "bgp" + if config_data["bgp_params"].get("additional_paths"): + command += " additional-paths {additional_paths}".format( + **config_data["bgp_params"] + ) + if config_data["bgp_params"]["additional_paths"] == "send": + command += " any" + elif config_data["bgp_params"].get("next_hop_address_family"): + command += " next-hop address-family ipv6" + elif config_data["bgp_params"].get("next_hop_unchanged"): + command += " next-hop-unchanged" + elif config_data["bgp_params"].get("redistribute_internal"): + command += " redistribute-internal" + elif config_data["bgp_params"].get("route"): + command += " route install-map {route}".format( + **config_data["bgp_params"] + ) + return command + + +def _tmplt_bgp_graceful_restart(config_data): + command = "graceful-restart" + return command + + +def _tmplt_bgp_neighbor(config_data): + command = "neighbor {peer}".format(**config_data["neighbor"]) + if config_data["neighbor"].get("additional_paths"): + command += " additional-paths {additional_paths}".format( + **config_data["neighbor"] + ) + if config_data["neighbor"]["additional_paths"] == "send": + command += "any" + elif config_data["neighbor"].get("activate"): + command += " activate" + elif config_data["neighbor"].get("default_originate"): + command += " default-originate" + if config_data["neighbor"]["default_originate"].get("route_map"): + command += " route-map {route_map}".format( + **config_data["neighbor"]["default_originate"] + ) + if config_data["neighbor"]["default_originate"].get("always"): + command += " always" + elif config_data["neighbor"].get("graceful_restart"): + command += " graceful-restart" + elif config_data["neighbor"].get("next_hop_unchanged"): + command += " next-hop-unchanged" + elif config_data["neighbor"].get("next_hop_address_family"): + command += " next-hop addres-family ipv6" + elif config_data["neighbor"].get("prefix_list"): + command += " prefix-list {name} {direction}".format( + **config_data["neighbor"]["prefix_list"] + ) + elif config_data["neighbor"].get("route_map"): + command += " route-map {name} {direction}".format( + **config_data["neighbor"]["route_map"] + ) + elif config_data["neighbor"].get("weight"): + command += " weight {weight}".format(**config_data["neighbor"]) + elif config_data["neighbor"].get("encapsulation"): + command += " encapsulation {transport}".format( + **config_data["neighbor"] + ) + if config_data["neighbor"]["encapsulation"].get("source_interface"): + command += " next-hop-self source-interface {source_interface}".format( + **config_data["neighbor"] + ) + return command + + +def _tmplt_bgp_network(config_data): + command = "network {address}".format(**config_data) + if config_data.get("route_map"): + command += " route-map {route_map}".format(**config_data) + return command + + +def _tmplt_bgp_redistribute(config_data): + command = "redistribute {protocol}".format(**config_data) + if config_data.get("isis_level"): + command += " {isis_level}".format(**config_data) + if config_data.get("ospf_route"): + command += " match {ospf_route}".format(**config_data) + if config_data.get("route_map"): + command += " route-map {route_map}".format(**config_data) + return command + + +def _tmplt_bgp_route_target(config_data): + command = "route-target {mode} {target}".format( + **config_data["route_target"] + ) + return command + + +class Bgp_afTemplate(NetworkTemplate): + def __init__(self, lines=None): + super(Bgp_afTemplate, self).__init__(lines=lines, tmplt=self) + + # fmt: off + PARSERS = [ + { + "name": "router", + "getval": re.compile( + r""" + ^router\s + bgp + \s(?P<as_num>\S+) + $""", + re.VERBOSE, + ), + "setval": _tmplt_router_bgp_cmd, + "compval": "as_number", + "result": {"as_number": "{{ as_num }}"}, + "shared": True + }, + { + "name": "address_family", + "getval": re.compile( + r""" + \s*(?P<vrf>vrf\s\S+)* + \s*address-family + \s(?P<afi>ipv4|ipv6|evpn) + \s*(?P<type>\S+)* + $""", + re.VERBOSE, + ), + "setval": _tmplt_bgp_address_family, + "result": { + "address_family": { + '{{ afi + "_" + vrf|d() }}': { + "afi": "{{ afi }}", + "safi": "{{ type }}", + "vrf": "{{ vrf.split(" ")[1] }}" + } + } + }, + "shared": True, + }, + { + "name": "bgp_params_additional_paths", + "getval": re.compile( + r""" + \s*bgp + \s+additional-paths + \s+(?P<action>\S+) + *$""", + re.VERBOSE, + ), + "setval": _tmplt_bgp_params, + "compval": "bgp_params.additional_paths", + "result": { + "address_family": { + '{{ afi + "_" + vrf|d() }}': { + "bgp_params": { + "additional_paths": "{{ action }}" + } + } + } + }, + }, + { + "name": "bgp_params.nexthop_address_family", + "getval": re.compile( + r""" + \s*bgp + \s+next-hop + \s+address-family + \s+ipv6 + *$""", + re.VERBOSE, + ), + "setval": _tmplt_bgp_params, + "compval": "bgp_params.next_hop_address_family", + "result": { + "address_family": { + '{{ afi + "_" + vrf|d() }}': { + "bgp_params": { + "next_hop_unchanged": "{{ 'ipv6' }}" + } + } + } + }, + }, + { + "name": "bgp_params.nexthop_unchanged", + "getval": re.compile( + r""" + \s*bgp + \s+next-hop-unchanged + *$""", + re.VERBOSE, + ), + "setval": _tmplt_bgp_params, + "compval": "bgp_params.next_hop_unchanged", + "result": { + "address_family": { + '{{ afi + "_" + vrf|d() }}': { + "bgp_params": { + "next_hop_unchanged": "{{ True }}" + } + } + } + }, + }, + { + "name": "bgp_params.redistribute_internal", + "getval": re.compile( + r""" + \s*bgp + \s+redistribute-internal + *$""", + re.VERBOSE, + ), + "setval": _tmplt_bgp_params, + "compval": "bgp_params.redistribute_internal", + "result": { + "address_family": { + '{{ afi + "_" + vrf|d() }}': { + "bgp_params": { + "redistribute_internal": "{{ True }}" + } + } + } + }, + }, + { + "name": "bgp_params.route", + "getval": re.compile( + r""" + \s*bgp + \s+route + \s+install-map + \s+(?P<route>\S+) + *$""", + re.VERBOSE, + ), + "setval": _tmplt_bgp_params, + "compval": "bgp_params.route", + "result": { + "address_family": { + '{{ afi + "_" + vrf|d() }}': { + "bgp_params": { + "route": "{{ route }}" + } + } + } + }, + }, + { + "name": "graceful_restart", + "getval": re.compile( + r""" + \s*graceful-restart + *$""", + re.VERBOSE, + ), + "setval": _tmplt_bgp_graceful_restart, + "result": { + "address_family": { + '{{ afi + "_" + vrf|d() }}': { + "graceful_restart": "{{ True }}" + } + } + }, + }, + { + "name": "neighbor.activate", + "getval": re.compile( + r""" + \s*neighbor + \s+(?P<peer>\S+) + \s+activate + *$""", + re.VERBOSE, + ), + "setval": _tmplt_bgp_neighbor, + "compval": "neighbor.activate", + "result": { + "address_family": { + '{{ afi + "_" + vrf|d() }}': { + "neighbor": { + "{{ peer }}": { + "peer": "{{ peer }}", + "activate": "{{ True }}", + } + } + } + } + }, + }, + { + "name": "neighbor.additional_paths", + "getval": re.compile( + r""" + \s*neighbor + \s+(?P<peer>\S+) + \s+additional-paths + \s+(?P<action>\S+) + *$""", + re.VERBOSE, + ), + "setval": _tmplt_bgp_neighbor, + "compval": "neighbor.additional_paths", + "result": { + "address_family": { + '{{ afi + "_" + vrf|d() }}': { + "neighbor": { + "{{ peer }}": { + "peer": "{{ peer }}", + "additional_paths": "{{ action }}", + } + } + } + } + }, + }, + { + "name": "neighbor.default_originate", + "getval": re.compile( + r""" + \s*neighbor + \s+(?P<peer>\S+) + \s+default-originate + \s*(?P<route_map>route-map\s\S+)* + \s*(?P<always>always)* + $""", + re.VERBOSE, + ), + "setval": _tmplt_bgp_neighbor, + "compval": "neighbor.default_originate", + "result": { + "address_family": { + '{{ afi + "_" + vrf|d() }}': { + "neighbor": { + "{{ peer }}": { + "peer": "{{ peer }}", + "default_originate": { + "route_map": "{{ route_map.split(" ")[1] }}", + "always": "{{ True if always is defined }}" + } + } + } + } + } + }, + }, + { + "name": "neighbor.graceful_restart", + "getval": re.compile( + r""" + \s*neighbor + \s+(?P<peer>\S+) + \s+graceful-restart + *$""", + re.VERBOSE, + ), + "setval": _tmplt_bgp_neighbor, + "compval": "neighbor.graceful_restart", + "result": { + "address_family": { + '{{ afi + "_" + vrf|d() }}': { + "neighbor": { + "{{ peer }}": { + "peer": "{{ peer }}", + "graceful_restart": "{{ True }}", + } + } + } + } + }, + }, + { + "name": "neighbor.next_hop_unchanged", + "getval": re.compile( + r""" + \s*neighbor + \s+(?P<peer>\S+) + \s+next-hop-unchanged + *$""", + re.VERBOSE, + ), + "setval": _tmplt_bgp_neighbor, + "compval": "neighbor.next_hop_unchanged", + "result": { + "address_family": { + '{{ afi + "_" + vrf|d() }}': { + "neighbor": { + "{{ peer }}": { + "peer": "{{ peer }}", + "next_hop_unchanged": "{{ True }}", + } + } + } + } + }, + }, + { + "name": "neighbor.next_hop_address_family", + "getval": re.compile( + r""" + \s*neighbor + \s+(?P<peer>\S+) + \s+next-hop + \s+address-family + \s+ipv6 + *$""", + re.VERBOSE, + ), + "setval": _tmplt_bgp_neighbor, + "compval": "neighbor.next_hop_address_family", + "result": { + "address_family": { + '{{ afi + "_" + vrf|d() }}': { + "neighbor": { + "{{ peer }}": { + "peer": "{{ peer }}", + "next_hop_address_family": "{{ 'ipv6' }}", + } + } + } + } + }, + }, + { + "name": "neighbor.prefix_list", + "getval": re.compile( + r""" + \s*neighbor + \s+(?P<peer>\S+) + \s+prefix-list + \s+(?P<name>\S+) + \s+(?P<dir>in|out) + *$""", + re.VERBOSE, + ), + "setval": _tmplt_bgp_neighbor, + "compval": "neighbor.prefix_list", + "result": { + "address_family": { + '{{ afi + "_" + vrf|d() }}': { + "neighbor": { + "{{ peer }}": { + "peer": "{{ peer }}", + "prefix_list": { + "name": "{{ name }}", + "direction": "{{ dir }}" + } + } + } + } + } + }, + }, + { + "name": "neighbor.route_map", + "getval": re.compile( + r""" + \s*neighbor + \s+(?P<peer>\S+) + \s+route-map + \s+(?P<name>\S+) + \s+(?P<dir>in|out) + *$""", + re.VERBOSE, + ), + "setval": _tmplt_bgp_neighbor, + "compval": "neighbor.route_map", + "result": { + "address_family": { + '{{ afi + "_" + vrf|d() }}': { + "neighbor": { + "{{ peer }}": { + "peer": "{{ peer }}", + "route_map": { + "name": "{{ name }}", + "direction": "{{ dir }}" + } + } + } + } + } + }, + }, + { + "name": "neighbor.weight", + "getval": re.compile( + r""" + \s*neighbor + \s+(?P<peer>\S+) + \s+weight + \s+(?P<weight>\d+) + *$""", + re.VERBOSE, + ), + "setval": _tmplt_bgp_neighbor, + "compval": "neighbor.weight", + "result": { + "address_family": { + '{{ afi + "_" + vrf|d() }}': { + "neighbor": { + "{{ peer }}": { + "peer": "{{ peer }}", + "weight": "{{ weight }}", + } + } + } + } + }, + }, + { + "name": "neighbor.encapsulation", + "getval": re.compile( + r""" + \s*neighbor + \s+default + \s+encapsulation + \s+(?P<type>mpls|vxlan) + \s*(next-hop-self)* + \s*(source-interface)* + \s*(?P<interface>\S+\s\S+)* + $""", + re.VERBOSE, + ), + "setval": _tmplt_bgp_neighbor, + "compval": "neighbor.encapsulation", + "result": { + "address_family": { + '{{ afi + "_" + vrf|d() }}': { + "neighbor": { + "{{ peer }}": { + "peer": "{{ peer }}", + "encapsulation": { + "transport": "{{ type }}", + "source_interface": "{{ interface }}" + } + } + } + } + } + }, + }, + { + "name": "network", + "getval": re.compile( + r""" + \s*network + \s+(?P<address>\S+) + \s*(route-map)* + \s*(?P<route_map>\S+)* + $""", + re.VERBOSE, + ), + "setval": _tmplt_bgp_network, + "compval": "network", + "result": { + "address_family": { + '{{ afi + "_" + vrf|d() }}': { + "network": { + "{{ address }}": { + "address": "{{ address }}", + "route_map": "{{ route_map }}", + } + } + } + } + }, + }, + { + "name": "redistribute", + "getval": re.compile( + r""" + \s*redistribute + \s+(?P<route>\S+) + \s*(?P<level>level-1|level-2|level-1-2)* + \s*(?P<match>match\s\S+)* + \s*(?P<route_map>route-map\s\S+)* + $""", + re.VERBOSE, + ), + "setval": _tmplt_bgp_redistribute, + "compval": "redistribute", + "result": { + "address_family": { + '{{ afi + "_" + vrf|d() }}': { + "redistribute": [ + { + "protocol": "{{ route }}", + "route_map": "{{ route_map.split(" ")[1] }}", + "isis_level": "{{ level }}", + "ospf_route": "{{ match.split(" ")[1] }}" + } + ] + } + } + }, + }, + { + "name": "route_target", + "getval": re.compile( + r""" + \s*route-target + \s+(?P<mode>both|import|export) + \s+(?P<target>\S+) + *$""", + re.VERBOSE, + ), + "setval": _tmplt_bgp_route_target, + "compval": "route_target", + "result": { + "address_family": { + '{{ afi + "_" + vrf|d() }}': { + "route_target": { + "mode": "{{ mode }}", + "target": "{{ target }}", + } + } + } + }, + }, + ] + # fmt: on diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/rm_templates/bgp_global.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/rm_templates/bgp_global.py new file mode 100644 index 00000000..9225574c --- /dev/null +++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/rm_templates/bgp_global.py @@ -0,0 +1,2881 @@ +# -*- coding: utf-8 -*- +# Copyright 2020 Red Hat +# 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 + +""" +The Bgp_global parser templates file. This contains +a list of parser definitions and associated functions that +facilitates both facts gathering and native command generation for +the given network resource. +""" + +import re + +from ansible_collections.ansible.netcommon.plugins.module_utils.network.common.network_template import ( + NetworkTemplate, +) + + +def _tmplt_router_bgp_cmd(config_data): + command = "router bgp {as_number}".format(**config_data) + return command + + +def _tmplt_bgp_vrf(config_data): + command = "vrf {vrf}".format(**config_data) + return command + + +def _tmplt_bgp_aggregate_address(config_data): + command = "aggregate-address {address}".format(**config_data) + if config_data.get("as_set"): + command += " as-set" + if config_data.get("summary_only"): + command += " summary-only" + if config_data.get("attribute_map"): + command += " attribute-map {attribute_map}".format(**config_data) + if config_data.get("match_map"): + command += " match-map {match_map}".format(**config_data) + if config_data.get("advertise_only"): + command += " advertise-only" + return command + + +def _tmplt_bgp_params(config_data): + command = "bgp" + if config_data["bgp_params"].get("additional_paths"): + command += ( + " additional-paths " + + config_data["bgp_params"]["additional_paths"] + ) + if config_data["bgp_params"]["additional_paths"] == "send": + command += " any" + elif config_data["bgp_params"].get("advertise_inactive"): + command += " advertise-inactive" + elif config_data["bgp_params"].get("allowas_in"): + command += " allowas-in" + if config_data["bgp_params"]["allowas_in"].get("count"): + command += " {count}".format( + **config_data["bgp_params"]["allowas_in"] + ) + elif config_data["bgp_params"].get("always_compare_med"): + command += " always-comapre-med" + elif config_data["bgp_params"].get("asn"): + command += " asn notaion {asn}".format(**config_data["bgp_params"]) + elif config_data["bgp_params"].get("auto_local_addr"): + command += " auto-local-addr" + elif config_data["bgp_params"].get("bestpath"): + command += " bestpath" + if config_data["bgp_params"]["bestpath"].get("as_path"): + command += " as-path {as_path}".format( + **config_data["bgp_params"]["as_path"] + ) + elif config_data["bgp_params"]["bestpath"].get("ecmp_fast"): + command += " ecmp-fast" + elif config_data["bgp_params"].get("med"): + command += " med" + if config_data["bgp_params"]["med"].get("confed"): + command += " confed" + else: + command += " missing-as-worst" + elif config_data["bgp_params"].get("skip"): + command += " skip next-hop igp-cost" + elif config_data["bgp_params"].get("tie_break"): + tie = re.sub(r"_", r"-", config_data["bgp_params"]["tie_break"]) + command += " tie-break " + tie + elif config_data["bgp_params"].get("client_to_client"): + command += " client-to-client" + elif config_data["bgp_params"].get("cluster_id"): + command += " cluster-id {cluster_id}".format( + **config_data["bgp_params"] + ) + elif config_data["bgp_params"].get("confederation"): + command += " confederation" + if config_data["bgp_params"]["confederation"].get("identifier"): + command += " identifier {identifier}".format( + **config_data["bgp_params"]["confederation"] + ) + else: + command += " peers {peers}".format( + **config_data["bgp_params"]["confederation"] + ) + elif config_data["bgp_params"].get("control_plane_filter"): + command += " control-plane-filter default-allow" + elif config_data["bgp_params"].get("convergence"): + command += " convergence" + if config_data["bgp_params"]["convergence"].get("slow_peer"): + command += " slow-peer" + command += " time {time}".format( + **config_data["bgp_params"]["convergence"] + ) + elif config_data["bgp_params"].get("default"): + command += " default {default}".format(**config_data["bgp_params"]) + elif config_data["bgp_params"].get("enforce_first-as"): + command += " enforce-first-as" + elif config_data["bgp_params"].get("host_routes"): + command += " host-routes fib direct-install" + elif config_data["bgp_params"].get("labeled_unicast"): + command += " labeled-unicast rib {labeled_unicast}".format( + **config_data["bgp_params"] + ) + elif config_data["bgp_params"].get("listen"): + command += " listen" + if config_data["bgp_params"]["listen"].get("limit"): + command += " limit {limit}".format( + **config_data["bgp_params"]["listen"] + ) + else: + command += " range {address} peer-group".format( + **config_data["bgp_params"]["listen"]["range"] + ) + if config_data["bgp_params"]["listen"]["range"]["peer_group"].get( + "peer_filter" + ): + command += " {name} peer-filter {peer_filter}".format( + **config_data["bgp_params"]["listen"]["range"][ + "peer_group" + ] + ) + else: + command += " {name} remote-as {remote_as}".format( + **config_data["bgp_params"]["listen"]["range"][ + "peer_group" + ] + ) + elif config_data["bgp_params"].get("log_neighbor_changes"): + command += " log-neighbor-changes" + elif config_data["bgp_params"].get("missing_policy"): + command += " missing-policy direction {direction} action {action}".format( + **config_data["bgp_params"]["missing_policy"] + ) + elif config_data["bgp_params"].get("monitoring"): + command += " monitoring" + elif config_data["bgp_params"].get("next_hop_unchanged"): + command += " next-hop-unchanged" + elif config_data["bgp_params"].get("redistribute_internal"): + command += " redistribute-internal" + elif config_data["bgp_params"].get("route"): + command += " route install-map {route}".format( + **config_data["bgp_params"] + ) + elif config_data["bgp_params"].get("route_reflector"): + command += " route-reflector preserve-attributes" + if config_data["bgp_params"]["reoute_reflector"].get("always"): + command += " always" + elif config_data["bgp_params"].get("transport"): + command += " transport listen-port {transport}".format( + **config_data["bgp_params"] + ) + return command + + +def _tmplt_bgp_redistribute(config_data): + command = "redistribute {protocol}".format(**config_data) + if config_data.get("isis_level"): + command += " {isis_level}".format(**config_data) + if config_data.get("ospf_route"): + if config_data["ospf_route"] == "nssa_external_2": + route = "nssa-external 2" + elif config_data["ospf_route"] == "nssa_external_1": + route = "nssa-external 1" + else: + route = config_data["ospf_route"] + command += " match " + route + if config_data.get("route_map"): + command += " route-map {route_map}".format(**config_data) + return command + + +def _tmplt_bgp_default_metric(config_data): + command = "default-metric {default_metric}".format(**config_data) + return command + + +def _tmplt_bgp_distance(config_data): + command = "distance bgp" + if config_data["distance"].get("external"): + command += " {external}".format(**config_data["distance"]) + if config_data["distance"].get("internal"): + command += " {internal}".format(**config_data["distance"]) + if config_data["distance"].get("local"): + command += " {local}".format(**config_data["distance"]) + return command + + +def _tmplt_bgp_graceful_restart(config_data): + command = "graceful-restart" + if config_data.get("restart_time"): + command += " restart-time {restart_time}".format(**config_data) + if config_data.get("stalepath_time"): + command += " stalepath-time {stalepath_time}".format(**config_data) + return command + + +def _tmplt_bgp_graceful_restart_helper(config_data): + command = "graceful-restart-helper" + return command + + +def _tmplt_bgp_access_group(config_data): + if config_data["accesS_group"].get("afi") == "ipv4": + afi = "ip" + else: + afi = "ipv6" + command = afi + " access-group {acl_name}".format( + **config_data["access_group"] + ) + if config_data["access_group"].get("direction"): + command += " {direction}".format(**config_data["access_group"]) + return command + + +def _tmplt_bgp_maximum_paths(config_data): + command = "maximum-paths {max_equal_cost_paths}".format( + **config_data["maximum_paths"] + ) + if config_data.get("max_installed_ecmp_paths"): + command += " ecmp {max_installed_ecmp_paths}".format( + **config_data["maximum_paths"] + ) + return command + + +def _tmplt_bgp_monitoring(config_data): + cmd = "monitoring" + command = "" + if config_data.get("timestamp"): + command = cmd + " timestamp {timestamp}".format(**config_data) + if config_data.get("port"): + command = cmd + " port {port}".format(**config_data) + if config_data.get("received"): + command = cmd + " received routes {received}".format(**config_data) + if config_data.get("station"): + command = cmd + " station {station}".format(**config_data) + return command + + +def _tmplt_bgp_neighbor(config_data): + command = "neighbor {peer}".format(**config_data["neighbor"]) + if config_data["neighbor"].get("additional_paths"): + command += " additional-paths {additional_paths}".format( + **config_data["neighbor"] + ) + if config_data["neighbor"]["additional_paths"] == "send": + command += "any" + elif config_data["neighbor"].get("peer_group"): + command += " peer-group" + if ( + config_data["neighbor"]["peer_group"] + != config_data["neighbor"]["peer_group"] + ): + command += config_data["neighbor"]["peer_group"] + elif config_data["neighbor"].get("allowas_in"): + command += " allowas-in" + if config_data["neighbor"]["allowas_in"].get("count"): + command += " {count}".format( + **config_data["neighbor"]["allowas_in"] + ) + elif config_data["neighbor"].get("auto_local_addr"): + command += " auto-local-addr" + elif config_data["neighbor"].get("default_originate"): + command += " default-originate" + if config_data["neighbor"]["default_originate"].get("route_map"): + command += " route-map {route_map}".format( + **config_data["neighbor"]["default_originate"] + ) + if config_data["neighbor"]["default_originate"].get("always"): + command += " always" + elif config_data["neighbor"].get("description"): + command += " description {description}".format( + **config_data["neighbor"] + ) + elif config_data["neighbor"].get("dont_capability_negotiate"): + command += " dont-capability-negotiate" + elif config_data["neighbor"].get("ebgp_multihop"): + command += " ebgp-multiphop" + if config_data["neighbor"]["ebgp_multihop"].get("ttl"): + command += " {ttl}".format( + **config_data["neighbor"]["ebgp_multihop"] + ) + elif config_data["neighbor"].get("enforce_first_as"): + command += " enforce-first-as" + elif config_data["neighbor"].get("export_localpref"): + command += " export-localpref {export_localpref}".format( + **config_data["neighbor"] + ) + elif config_data["neighbor"].get("fall_over"): + command += " fall-over bfd" + elif config_data["neighbor"].get("graceful_restart"): + command += " graceful-restart" + elif config_data["neighbor"].get("graceful_restart_helper"): + command += " graceful-restart-helper" + elif config_data["neighbor"].get("idle_restart_timer"): + command += " idle-restart-timer {idle_restart_timer}".format( + **config_data["neighbor"] + ) + elif config_data["neighbor"].get("import_localpref"): + command += " import-localpref {import_localpref}".format( + **config_data["neighbor"] + ) + elif config_data["neighbor"].get("link_bandwidth"): + command += " link-bandwidth" + if config_data["neighbor"]["link_bandwidth"].get("auto"): + command += " auto" + if config_data["neighbor"]["link_bandwidth"].get("default"): + command += " default {default}".format( + **config_data["neighbor"]["link_bandwidth"] + ) + if config_data["neighbor"]["link_bandwidth"].get("update_delay"): + command += " update-delay {update_delay}".format( + **config_data["neighbor"]["link_bandwidth"] + ) + elif config_data["neighbor"].get("local_as"): + command += " local-as {as_number} no-prepend replace-as".format( + **config_data["neighbor"] + ) + if config_data["neighbor"]["local_as"].get("fallback"): + command += " fallback" + elif config_data["neighbor"].get("local_v6_addr"): + command += " local-v6-addr {local_v6_addr}".format( + **config_data["neighbor"] + ) + elif config_data["neighbor"].get("maximum_accepted_routes"): + command += " maximum-accepted-routes {count}".format( + **config_data["neighbor"]["maximum_accepted_routes"] + ) + if config_data["neighbor"]["maximum_accepted_routes"].get( + "warning_limit" + ): + command += " warning-limit {warning_limit}".format( + **config_data["neighbor"]["maximum_accepted_routes"] + ) + elif config_data["neighbor"].get("maximum_received_routes"): + command += " maximum-routes {count}".format( + **config_data["neighbor"]["maximum_received_routes"] + ) + if config_data["neighbor"]["maximum_received_routes"].get( + "warning_limit" + ): + if config_data["neighbor"]["maximum_received_routes"][ + "warning_limit" + ].get("limit_count"): + command += " warning-limit {limit_count}".format( + **config_data["neighbor"]["maximum_received_routes"][ + "warning_limit" + ] + ) + if config_data["neighbor"]["maximum_received_routes"][ + "warning_limit" + ].get("limit_percent"): + command += ( + " warning-limit " + + str( + config_data["neighbor"]["maximum_received_routes"][ + "warning_limit" + ]["limit_percent"] + ) + + " percent" + ) + if config_data["neighbor"]["maximum_received_routes"].get( + "warning_only" + ): + command += " warning-only" + elif config_data["neighbor"].get("metric_out"): + command += " metric-out {metric_out}".format(**config_data["neighbor"]) + elif config_data["neighbor"].get("monitoring"): + command += " monitoring" + elif config_data["neighbor"].get("next_hop_self"): + command += " next-hop-self" + elif config_data["neighbor"].get("next_hop_unchanged"): + command += " next-hop-unchanged" + elif config_data["neighbor"].get("next_hop_v6_addr"): + command += " next-hop-v6-addt {next_hop_v6_addr} in".format( + **config_data["neighbor"] + ) + elif config_data["neighbor"].get("out_delay"): + command += " out-delay {out_delay}".format(**config_data["neighbor"]) + elif config_data["neighbor"].get("remote_as"): + command += " remote-as {remote_as}".format(**config_data["neighbor"]) + elif config_data["neighbor"].get("remove_private_as"): + command += " remove-private-as" + if config_data["neighbor"]["remove_private_as"].get("all"): + command += " all" + if config_data["neighbor"]["remove_private_as"].get("replace_as"): + command += " replace-as" + elif config_data["neighbor"].get("peer_as"): + command += " peer-as {peer_as}".format(**config_data["neighbor"]) + elif config_data["neighbor"].get("prefix_list"): + command += " prefix-list {name} {direction}".format( + **config_data["neighbor"]["prefix_list"] + ) + elif config_data["neighbor"].get("route_map"): + command += " route-map {name} {direction}".format( + **config_data["neighbor"]["route_map"] + ) + elif config_data["neighbor"].get("route_reflector_client"): + command += " route-reflector-client" + elif config_data["neighbor"].get("route_to_peer"): + command += " route-to-peer" + elif config_data["neighbor"].get("send_community"): + command += " send-community" + if config_data["neighbor"]["send_community"].get( + "community_attribute" + ): + command += ( + " " + + config_data["neighbor"]["send_community"][ + "community_attribute" + ] + ) + if config_data["neighbor"]["send_community"].get("sub_attribute"): + command += ( + " " + + config_data["neighbor"]["send_community"]["sub_attribute"] + ) + if config_data["neighbor"]["send_community"].get( + "link_bandwidth_attribute" + ): + command += ( + " " + + config_data["neighbor"]["send_community"][ + "link_bandwidth_attribute" + ] + ) + if config_data["neighbor"]["send_community"].get("speed"): + command += " " + config_data["neighbor"]["send_community"]["speed"] + if config_data["neighbor"]["send_community"].get("divide"): + command += ( + " " + config_data["neighbor"]["send_community"]["divide"] + ) + elif config_data["neighbor"].get("shutdown"): + command += " shutdown" + elif config_data["neighbor"].get("soft_reconfiguration"): + command += " soft-reconfiguration inbound" + if config_data["neighbor"]["soft_reconfiguration"] == "all": + command += " all" + elif config_data["neighbor"].get("transport"): + command += " transport" + if config_data["neighbor"]["transport"].get("connection_mode"): + command += " connection-mode passive" + else: + command += " remote-port {remote_port}".format( + **config_data["neighbor"]["transport"] + ) + elif config_data["neighbor"].get("timers"): + command += " timers {keepalive} {holdtime}".format( + **config_data["neighbor"]["timers"] + ) + elif config_data["neighbor"].get("ttl"): + command += " ttl maximum-hops {ttl}".format(**config_data["neighbor"]) + elif config_data["neighbor"].get("update_source"): + command += " update-source {update_source}".format( + **config_data["neighbor"] + ) + elif config_data["neighbor"].get("weight"): + command += " weight {weight}".format(**config_data["neighbor"]) + return command + + +def _tmplt_bgp_network(config_data): + command = "network {address}".format(**config_data) + if config_data.get("route_map"): + command += " route-map {route_map}".format(**config_data) + return command + + +def _tmplt_bgp_route_target(config_data): + command = "route-target {action} {target}".format( + **config_data["route_target"] + ) + return command + + +def _tmplt_bgp_router_id(config_data): + command = "router-id {router_id}".format(**config_data) + return command + + +def _tmplt_bgp_shutdown(config_data): + return "shutdown" + + +def _tmplt_bgp_timers(config_data): + command = "timers bgp {keepalive} {holdtime}".format( + **config_data["timers"] + ) + return command + + +def _tmplt_bgp_ucmp(config_data): + command = "ucmp" + if "fec" in config_data["ucmp"]: + command += " fec threshold trigger" + command += " {trigger} clear {clear} warning-only".format( + **config_data["ucmp"]["fec"] + ) + if "link_bandwidth" in config_data["ucmp"]: + command += " link-bandwidth {mode}".format( + **config_data["ucmp"]["link_bandwidth"] + ) + if config_data["ucmp"]["link_bandwidth"].get("mode") == "update_delay": + command += " {update_delay}".format( + **config_data["ucmp"]["link_bandwidth"] + ) + if "mode" in config_data["ucmp"]: + command += " mode 1" + if config_data["ucmp"]["mode"].get("nexthops"): + command += " {nexthops}".format(**config_data["ucmp"]["mode"]) + return command + + +def _tmplt_bgp_update(config_data): + command = "update {wait_for}".format(**config_data["update"]) + if config_data["update"].get("batch_size"): + command += " {batch_size}".format(**config_data["update"]) + return command + + +def _tmplt_bgp_vlan(config_data): + command = "vlan {vlan}".format(**config_data) + return command + + +def _tmplt_bgp_vlan_aware_bundle(config_data): + command = "vlan-aware-bundle " + config_data["vlan_aware_bundle"] + return command + + +class Bgp_globalTemplate(NetworkTemplate): + def __init__(self, lines=None): + super(Bgp_globalTemplate, self).__init__(lines=lines, tmplt=self) + + # fmt: off + PARSERS = [ + { + "name": "router", + "getval": re.compile( + r""" + ^router\s + bgp + \s(?P<as_num>\S+) + $""", + re.VERBOSE, + ), + "setval": _tmplt_router_bgp_cmd, + "compval": "as_number", + "result": {"as_number": "{{ as_num }}"}, + }, + { + "name": "vrf", + "getval": re.compile( + r""" + \s*vrf + \s(?P<vrf>\S+) + *$""", + re.VERBOSE, + ), + "setval": _tmplt_bgp_vrf, + "compval": "vrfs.vrf", + "result": { + "vrfs": { + '{{ "vrf_" + vrf|d() }}': { + "vrf": "{{ vrf }}" + } + } + }, + "shared": True, + }, + { + "name": "aggregate_address", + "getval": re.compile( + r""" + \s*aggregate-address + \s+(?P<address>\S+) + \s*(?P<as_set>as-set)* + \s*(?P<summary_only>summary-only)* + \s*(?P<attribute_map>attribute-map\s\S+)* + \s*(?P<match_map>match-map\s\S+)* + \s*(?P<advertise_only>advertise-only)* + $""", + re.VERBOSE, + ), + "setval": _tmplt_bgp_aggregate_address, + "compval": "aggregate_address", + "result": { + "vrfs": { + '{{ "vrf_" + vrf|d() }}': { + "aggregate_address": [ + { + "address": "{{ address }}", + "advertise_only": "{{ True if advertise_only is defined }}", + "as_set": "{{ True if as_set is defined }}", + "summary_only": "{{ True if summary_only is defined }}", + "attribute_map": "{{ attribute_map.split(" ")[1] }}", + "match_map": "{{ match_map.split(" ")[1] }}" + } + ] + } + } + }, + }, + { + "name": "bgp_params_additional_paths", + "getval": re.compile( + r""" + \s*bgp + \s+additional-paths + \s+(?P<action>\S+) + \s*(any) + *$""", + re.VERBOSE, + ), + "setval": _tmplt_bgp_params, + "compval": "bgp_params.additional_paths", + "result": { + "vrfs": { + '{{ "vrf_" + vrf|d() }}': { + "bgp_params": { + "additional_paths": "{{ action }}" + } + } + } + }, + }, + { + "name": "bgp_params_advertise_inactive", + "getval": re.compile( + r""" + \s*bgp + \s+advertise-inactive + *$""", + re.VERBOSE, + ), + "setval": _tmplt_bgp_params, + "compval": "bgp_params.advertise_inactive", + "result": { + "vrfs": { + '{{ "vrf_" + vrf|d() }}': { + "bgp_params": { + "advertise_inactive": "{{ True }}" + } + } + } + }, + }, + { + "name": "bgp_params_allowas_in", + "getval": re.compile( + r""" + \s*bgp + \s+allowas-in + \s*(?P<count>\d+)* + $""", + re.VERBOSE, + ), + "setval": _tmplt_bgp_params, + "compval": "bgp_params.allowas_in", + "result": { + "vrfs": { + '{{ "vrf_" + vrf|d() }}': { + "bgp_params": { + "allowas_in": { + "set": "{{ True if count is undefined }}", + "count": "{{ count }}" + } + } + } + } + }, + }, + { + "name": "bgp_params_always_compare_med", + "getval": re.compile( + r""" + \s*bgp + \s+always-compare-med + *$""", + re.VERBOSE, + ), + "setval": _tmplt_bgp_params, + "compval": "bgp_params.always_compare_med", + "result": { + "vrfs": { + '{{ "vrf_" + vrf|d() }}': { + "bgp_params": { + "always_compare_med": "{{ True }}" + } + } + } + }, + }, + { + "name": "bgp_params_asn", + "getval": re.compile( + r""" + \s*bgp + \s+asn + \s+notation + \s+(?P<notation>asdot|asplain) + *$""", + re.VERBOSE, + ), + "setval": _tmplt_bgp_params, + "compval": "bgp_params.asn", + "result": { + "vrfs": { + '{{ "vrf_" + vrf|d() }}': { + "bgp_params": { + "asn": "{{ notation }}" + } + } + } + }, + }, + { + "name": "bgp_params_auto_local_addr", + "getval": re.compile( + r""" + \s*bgp + \s+auto-local-addr + *$""", + re.VERBOSE, + ), + "setval": _tmplt_bgp_params, + "compval": "bgp_params.auto_local_addr", + "result": { + "vrfs": { + '{{ "vrf_" + vrf|d() }}': { + "bgp_params": { + "auto_local_addr": "{{ True }}" + } + } + } + }, + }, + { + "name": "bgp_params_bestpath_as_path", + "getval": re.compile( + r""" + \s*bgp + \s+bestpath + \s*(?P<as_path>as-path\s\S+) + *$""", + re.VERBOSE, + ), + "setval": _tmplt_bgp_params, + "compval": "bgp_params.bestpath", + "result": { + "vrfs": { + '{{ "vrf_" + vrf|d() }}': { + "bgp_params": { + "bestpath": { + "as_path": "{{ as_path.split(" ")[1] }}" + } + } + } + } + }, + }, + { + "name": "bgp_params_bestpath_ecmp_fast", + "getval": re.compile( + r""" + \s*bgp + \s+bestpath + \s+ecmp-fast + *$""", + re.VERBOSE, + ), + "setval": _tmplt_bgp_params, + "compval": "bgp_params.bestpath", + "result": { + "vrfs": { + '{{ "vrf_" + vrf|d() }}': { + "bgp_params": { + "bestpath": { + "ecmp_fast": "{{ True }}" + } + } + } + } + }, + }, + { + "name": "bgp_params_bestpath_med", + "getval": re.compile( + r""" + \s*bgp + \s+bestpath + \s+med + \s*(?P<confed>confed)* + \s*(?P<missing>missing-as-worst)* + $""", + re.VERBOSE, + ), + "setval": _tmplt_bgp_params, + "compval": "bgp_params.bestpath", + "result": { + "vrfs": { + '{{ "vrf_" + vrf|d() }}': { + "bgp_params": { + "med": { + "confed": "{{ True if confed is defined }}", + "missing_as_worst": "{{ True if missing is defined }}" + } + } + } + } + }, + }, + { + "name": "bgp_params_bestpath_skip", + "getval": re.compile( + r""" + \s*bgp + \s+bestpath + \s+(?P<skip>skip) + \s+next-hop + \s+igp-cost + *$""", + re.VERBOSE, + ), + "setval": _tmplt_bgp_params, + "compval": "bgp_params.bestpath", + "result": { + "vrfs": { + '{{ "vrf_" + vrf|d() }}': { + "bgp_params": { + "bestpath": { + "skip": "{{ True }}" + } + } + } + } + }, + }, + { + "name": "bgp_params_tie_break", + "getval": re.compile( + r""" + \s*bgp + \s+tie-break + \s+(?P<tie>\S+) + *$""", + re.VERBOSE, + ), + "setval": _tmplt_bgp_params, + "compval": "bgp_params.bestpath", + "result": { + "vrfs": { + '{{ "vrf_" + vrf|d() }}': { + "bgp_params": { + "tie_break": "{{ tie }}" + } + } + } + }, + }, + { + "name": "bgp_params_client_to_client", + "getval": re.compile( + r""" + \s*bgp + \s+client-to-client + \s+reflection + *$""", + re.VERBOSE, + ), + "setval": _tmplt_bgp_params, + "compval": "bgp_params.client_to_client", + "result": { + "vrfs": { + '{{ "vrf_" + vrf|d() }}': { + "bgp_params": { + "client_to_client": "{{ True }}" + } + } + } + }, + }, + { + "name": "bgp_params_cluster_id", + "getval": re.compile( + r""" + \s*bgp + \s+cluster-id + \s+(?P<address>\S+) + *$""", + re.VERBOSE, + ), + "setval": _tmplt_bgp_params, + "compval": "bgp_params.cluster_id", + "result": { + "vrfs": { + '{{ "vrf_" + vrf|d() }}': { + "bgp_params": { + "cluster_id": "{{ address }}" + } + } + } + }, + }, + { + "name": "bgp_params_confederation", + "getval": re.compile( + r""" + \s*bgp + \s+confederation + \s*(?P<identifier>identifier\s.+)* + \s*(?P<peers>peers\s.+)* + $""", + re.VERBOSE, + ), + "setval": _tmplt_bgp_params, + "compval": "bgp_params.confederation", + "result": { + "vrfs": { + '{{ "vrf_" + vrf|d() }}': { + "bgp_params": { + "confederation": { + "identifier": "{{ identifier.split(" ")[1] }}", + "peers": "{{ peers.split(" ")[1] }}" + } + } + } + } + }, + }, + { + "name": "bgp_params_control_plane_filter", + "getval": re.compile( + r""" + \s*bgp + \s+control-plane-filter + \s+default-allow + *$""", + re.VERBOSE, + ), + "setval": _tmplt_bgp_params, + "compval": "bgp_params.control_plane_filter", + "result": { + "vrfs": { + '{{ "vrf_" + vrf|d() }}': { + "bgp_params": { + "control_plane_filter": "{{ True }}" + } + } + } + }, + }, + { + "name": "bgp_params_convergence", + "getval": re.compile( + r""" + \s*bgp + \s+convergence + \s*(?P<slow>slow-peer)* + \s+(?P<time>time\s\d+) + *$""", + re.VERBOSE, + ), + "setval": _tmplt_bgp_params, + "compval": "bgp_params.convergence", + "result": { + "vrfs": { + '{{ "vrf_" + vrf|d() }}': { + "bgp_params": { + "convergence": { + "slow_peer": "{{ True if slow is defined else False}}", + "time": "{{ time.split(" ")[1] }}" + } + } + } + } + }, + }, + { + "name": "bgp_params_default", + "getval": re.compile( + r""" + \s*bgp + \s+default + \s(?P<param>\S+) + *$""", + re.VERBOSE, + ), + "setval": _tmplt_bgp_params, + "compval": "bgp_params.default", + "result": { + "vrfs": { + '{{ "vrf_" + vrf|d() }}': { + "bgp_params": { + "default": "{{ param }}" + } + } + } + }, + }, + { + "name": "bgp_params.enforce_first_as", + "getval": re.compile( + r""" + \s*bgp + \s+enforce-first-as + *$""", + re.VERBOSE, + ), + "setval": _tmplt_bgp_params, + "compval": "bgp_params.enforce_first_as", + "result": { + "vrfs": { + '{{ "vrf_" + vrf|d() }}': { + "bgp_params": { + "enforce_first_as": "{{ True }}" + } + } + } + }, + }, + { + "name": "bgp_params.host_routes", + "getval": re.compile( + r""" + \s*bgp + \s+host-routes + \s+fib + \s+direct-install + *$""", + re.VERBOSE, + ), + "setval": _tmplt_bgp_params, + "compval": "bgp_params.host_routes", + "result": { + "vrfs": { + '{{ "vrf_" + vrf|d() }}': { + "bgp_params": { + "host_routes": "{{ True }}" + } + } + } + }, + }, + { + "name": "bgp_params.labelled_unicast", + "getval": re.compile( + r""" + \s*bgp + \s+labeled-unicast + \s+rib + \s+(?P<rib>\S+) + *$""", + re.VERBOSE, + ), + "setval": _tmplt_bgp_params, + "compval": "bgp_params.labelled_unicast", + "result": { + "vrfs": { + '{{ "vrf_" + vrf|d() }}': { + "bgp_params": { + "labeled_unicast": "{{ rib }}" + } + } + } + }, + }, + { + "name": "bgp_params.listen_limit", + "getval": re.compile( + r""" + \s*bgp + \s+listen + \s+limit + \s+(?P<limit>\d+) + *$""", + re.VERBOSE, + ), + "setval": _tmplt_bgp_params, + "compval": "bgp_params.listen.limit", + "result": { + "vrfs": { + '{{ "vrf_" + vrf|d() }}': { + "bgp_params": { + "listen": { + "limit": "{{ limit }}", + } + } + } + } + }, + }, + { + "name": "bgp_params.listen_range", + "getval": re.compile( + r""" + \s*bgp + \s+listen + \s+range + \s+(?P<address>\S+) + \s+peer-group + \s+(?P<group>\S+) + \s*(?P<filter>peer-filter \S+)* + \s*(?P<remote_as>remote-as \S+)* + $""", + re.VERBOSE, + ), + "setval": _tmplt_bgp_params, + "compval": "bgp_params.listen.range", + "result": { + "vrfs": { + '{{ "vrf_" + vrf|d() }}': { + "bgp_params": { + "listen": { + "range": { + "address": "{{ address }}", + "peer_group": { + "name": "{{ group }}", + "peer_filter": "{{ filter }}", + "remote_as": "{{ remote_as }}", + }, + } + } + } + } + } + }, + }, + { + "name": "bgp_params.log_neighbor_changes", + "getval": re.compile( + r""" + \s*bgp + \s+log-neighbor-changes + *$""", + re.VERBOSE, + ), + "setval": _tmplt_bgp_params, + "compval": "bgp_params.log_neighbor_changes", + "result": { + "vrfs": { + '{{ "vrf_" + vrf|d() }}': { + "bgp_params": { + "log_neighbor_changes": "{{ True }}" + } + } + } + }, + }, + { + "name": "bgp_params.missing_policy", + "getval": re.compile( + r""" + \s*bgp + \s+missing-policy + \s+direction + \s+(?P<dir>in|out) + \s+action + \s+(?P<action>\S+) + *$""", + re.VERBOSE, + ), + "setval": _tmplt_bgp_params, + "compval": "bgp_params.missing_policy", + "result": { + "vrfs": { + '{{ "vrf_" + vrf|d() }}': { + "bgp_params": { + "missing_policy": { + "direction": "{{ dir }}", + "action": "{{ action }}" + } + } + } + } + }, + }, + { + "name": "bgp_params.monitoring", + "getval": re.compile( + r""" + \s*bgp + \s+monitoring + *$""", + re.VERBOSE, + ), + "setval": _tmplt_bgp_params, + "compval": "bgp_params.monitoring", + "result": { + "vrfs": { + '{{ "vrf_" + vrf|d() }}': { + "bgp_params": { + "monitoring": "{{ True }}" + } + } + } + }, + }, + { + "name": "bgp_params.nexthop_unchanged", + "getval": re.compile( + r""" + \s*bgp + \s+next-hop-unchanged + *$""", + re.VERBOSE, + ), + "setval": _tmplt_bgp_params, + "compval": "bgp_params.next_hop_unchanged", + "result": { + "vrfs": { + '{{ "vrf_" + vrf|d() }}': { + "bgp_params": { + "next_hop_unchanged": "{{ True }}" + } + } + } + }, + }, + { + "name": "bgp_params.redistribute_internal", + "getval": re.compile( + r""" + \s*bgp + \s+redistribute-internal + *$""", + re.VERBOSE, + ), + "setval": _tmplt_bgp_params, + "compval": "bgp_params.redistribute_internal", + "result": { + "vrfs": { + '{{ "vrf_" + vrf|d() }}': { + "bgp_params": { + "redistribute_internal": "{{ True }}" + } + } + } + }, + }, + { + "name": "bgp_params.route", + "getval": re.compile( + r""" + \s*bgp + \s+route + \s+install-map + \s+(?P<route>\S+) + *$""", + re.VERBOSE, + ), + "setval": _tmplt_bgp_params, + "compval": "bgp_params.route", + "result": { + "vrfs": { + '{{ "vrf_" + vrf|d() }}': { + "bgp_params": { + "route": "{{ route }}" + } + } + } + }, + }, + { + "name": "bgp_params.route_reflector", + "getval": re.compile( + r""" + \s*bgp + \s+route-reflector + \s+preserve-attributes + \s*(?P<preserve>always)* + $""", + re.VERBOSE, + ), + "setval": _tmplt_bgp_params, + "compval": "bgp_params.route_reflector", + "result": { + "vrfs": { + '{{ "vrf_" + vrf|d() }}': { + "bgp_params": { + "route_reflector": { + "set": "{{ True if presever is undefined }}", + "preserve": "{{ True if preserve is defined }}" + } + } + } + } + }, + }, + { + "name": "bgp_params.transport", + "getval": re.compile( + r""" + \s*bgp + \s+transport + \s+listen-port + \s+(?P<port>\d+) + *$""", + re.VERBOSE, + ), + "setval": _tmplt_bgp_params, + "compval": "bgp_params.transport", + "result": { + "vrfs": { + '{{ "vrf_" + vrf|d() }}': { + "bgp_params": { + "transport": "{{ port }}" + } + } + } + }, + }, + { + "name": "default_metric", + "getval": re.compile( + r""" + \s*default-metric + \s(?P<metric>\d+) + $""", + re.VERBOSE, + ), + "setval": _tmplt_bgp_default_metric, + "compval": "default_metric", + "result": { + "vrfs": { + '{{ "vrf_" + vrf|d() }}': { + "default_metric": "{{ metric }}" + } + } + }, + }, + { + "name": "redistribute", + "getval": re.compile( + r""" + \s*redistribute + \s+(?P<route>\S+) + \s*(?P<level>level-1|level-2|level-1-2)* + \s*(?P<match>match\s.+)* + \s*(?P<route_map>route-map\s\S+)* + $""", + re.VERBOSE, + ), + "setval": _tmplt_bgp_redistribute, + "compval": "redistribute", + "result": { + "vrfs": { + '{{ "vrf_" + vrf|d() }}': { + "redistribute": [ + { + "protocol": "{{ route }}", + "route_map": "{{ route_map.split(" ")[1] }}", + "isis_level": "{{ level }}", + "ospf_route": "{{ 'nssa_external_' + match.split(" ")[2] if match.split(" ")[1] == 'nssa-external' else match.split(" ")[1]}}" + } + ] + } + } + }, + }, + { + "name": "distance", + "getval": re.compile( + r""" + \s*distance + \s+bgp + \s(?P<external>\d+) + \s*(?P<internal>\d+)* + \s*(?P<local>\d+)* + $""", + re.VERBOSE, + ), + "setval": _tmplt_bgp_distance, + "compval": "distance", + "result": { + "vrfs": { + '{{ "vrf_" + vrf|d() }}': { + "distance": { + "external": "{{ external }}", + "internal": "{{ internal }}", + "local": "{{ local }}" + } + } + } + }, + }, + { + "name": "graceful_restart", + "getval": re.compile( + r""" + \s*graceful-restart + \s*(?P<restart_time>restart-time\s\d+)* + \s*(?P<stalepath_time>stalepath-time\s\d+)* + $""", + re.VERBOSE, + ), + "setval": _tmplt_bgp_graceful_restart, + "remval": "graceful-restart", + "result": { + "vrfs": { + '{{ "vrf_" + vrf|d() }}': { + "graceful_restart": { + "set": "{{ True if restart_time and stalepath_time is not defined }}", + "restart_time": "{{ restart_time.split(" ")[1]|int }}", + "stalepath_time": "{{ stalepath_time.split(" ")[1]|int }}", + } + } + } + }, + }, + { + "name": "graceful_restart_helper", + "getval": re.compile( + r""" + \s*graceful-restart-helper + *$""", + re.VERBOSE, + ), + "setval": _tmplt_bgp_graceful_restart_helper, + "result": { + "vrfs": { + '{{ "vrf_" + vrf|d() }}': { + "graceful_restart": { + "set": "{{ True }}" + } + } + } + }, + }, + { + "name": "acccess_group", + "getval": re.compile( + r""" + \s*(?P<afi>ip|ipv6) + \s+access-group + \s+(?P<acl_name>\S+) + \s*(?P<direction>in)* + $""", + re.VERBOSE, + ), + "setval": _tmplt_bgp_access_group, + "compval": "access_group", + "result": { + "vrfs": { + '{{ "vrf_" + vrf|d() }}': { + "access_group": { + "afi": "{{ ipv4 if afi == 'ip' else afi }}", + "acl_name": "{{ acl_name }}", + "direction": "{{ direction }}", + } + } + } + }, + }, + { + "name": "maximum_paths", + "getval": re.compile( + r""" + \s*maximum-paths + \s+(?P<max_equal_cost_paths>\d+) + \s*(ecmp)* + \s*(?P<max_installed_ecmp_paths>\d+)* + $""", + re.VERBOSE, + ), + "setval": _tmplt_bgp_maximum_paths, + "compval": "maximum_paths", + "result": { + "vrfs": { + '{{ "vrf_" + vrf|d() }}': { + "maximum_paths": { + "max_equal_cost_paths": "{{ max_equal_cost_paths }}", + "max_installed_ecmp_paths": "{{ max_installed_ecmp_paths }}", + } + } + } + }, + }, + { + "name": "monitoring", + "getval": re.compile( + r""" + \s*monitoring + \s+(?P<port>\d+) + \s*(?P<received>received\sroutes\s\S+)* + \s*(?P<time>timestamp\s\S+)* + \s*(?P<station>station\s\S+)* + $""", + re.VERBOSE, + ), + "setval": _tmplt_bgp_monitoring, + "compval": "monitoring", + "result": { + "vrfs": { + '{{ "vrf_" + vrf|d() }}': { + "monitoring": { + "port": "{{ port }}", + "received": "{{ received.split(" ")[2] }}", + "timestamp": "{{ timestamp.split(" ")[1] }}", + "station": "{{ station.split(" ")[1] }}", + } + } + } + }, + }, + { + "name": "neighbor.additional_paths", + "getval": re.compile( + r""" + \s*neighbor + \s+(?P<peer>\S+) + \s+additional-paths + \s+(?P<action>\S+) + *$""", + re.VERBOSE, + ), + "setval": _tmplt_bgp_neighbor, + "compval": "neighbor.additional_paths", + "result": { + "vrfs": { + '{{ "vrf_" + vrf|d() }}': { + "neighbor": { + "{{ peer }}": { + "peer": "{{ peer }}", + "additional_paths": "{{ action }}", + } + } + } + } + }, + }, + { + "name": "neighbor.allowas_in", + "getval": re.compile( + r""" + \s*neighbor + \s+(?P<peer>\S+) + \s+allowas-in + \s*(?P<count>\d+) + *$""", + re.VERBOSE, + ), + "setval": _tmplt_bgp_neighbor, + "compval": "neighbor.allowas_in", + "result": { + "vrfs": { + '{{ "vrf_" + vrf|d() }}': { + "neighbor": { + "{{ peer }}": { + "peer": "{{ peer }}", + "allowas_in": { + "set": "{{ True if count is undefined }}", + "count": "{{ count }}" + } + } + } + } + } + }, + }, + { + "name": "neighbor.auto_local_addr", + "getval": re.compile( + r""" + \s*neighbor + \s+(?P<peer>\S+) + \s+auto-local-addr + *$""", + re.VERBOSE, + ), + "setval": _tmplt_bgp_neighbor, + "compval": "neighbor.auto_local_addr", + "result": { + "vrfs": { + '{{ "vrf_" + vrf|d() }}': { + "neighbor": { + "{{ peer }}": { + "peer": "{{ peer }}", + "auto_local_addr": "{{ True }}", + } + } + } + } + }, + }, + { + "name": "neighbor.default_originate", + "getval": re.compile( + r""" + \s*neighbor + \s+(?P<peer>\S+) + \s+default-originate + \s*(?P<route_map>route-map\s\S+)* + \s*(?P<always>always)* + $""", + re.VERBOSE, + ), + "setval": _tmplt_bgp_neighbor, + "compval": "neighbor.default_originate", + "result": { + "vrfs": { + '{{ "vrf_" + vrf|d() }}': { + "neighbor": { + "{{ peer }}": { + "peer": "{{ peer }}", + "default_originate": { + "route_map": "{{ route_map.split(" ")[1] }}", + "always": "{{ True if always is defined }}" + } + } + } + } + } + }, + }, + { + "name": "neighbor.description", + "getval": re.compile( + r""" + \s*neighbor + \s+(?P<peer>\S+) + \s+description + \s+(?P<desc>\S+) + *$""", + re.VERBOSE, + ), + "setval": _tmplt_bgp_neighbor, + "compval": "neighbor.description", + "result": { + "vrfs": { + '{{ "vrf_" + vrf|d() }}': { + "neighbor": { + "{{ peer }}": { + "peer": "{{ peer }}", + "description": "{{ desc }}", + } + } + } + } + }, + }, + { + "name": "neighbor.dont_capability_negotiate", + "getval": re.compile( + r""" + \s*neighbor + \s+(?P<peer>\S+) + \s+dont-capability-negotiate + *$""", + re.VERBOSE, + ), + "setval": _tmplt_bgp_neighbor, + "compval": "neighbor.dont_capability_negotiate", + "result": { + "vrfs": { + '{{ "vrf_" + vrf|d() }}': { + "neighbor": { + "{{ peer }}": { + "peer": "{{ peer }}", + "dont_capability_negotiate": "{{ True }}", + } + } + } + } + }, + }, + { + "name": "neighbor.ebgp_multihop", + "getval": re.compile( + r""" + \s*neighbor + \s+(?P<peer>\S+) + \s+ebgp-multihop + \s*(?P<ttl>\d+)* + $""", + re.VERBOSE, + ), + "setval": _tmplt_bgp_neighbor, + "compval": "neighbor.ebgp_multihop", + "result": { + "vrfs": { + '{{ "vrf_" + vrf|d() }}': { + "neighbor": { + "{{ peer }}": { + "peer": "{{ peer }}", + "ebgp_multihop": { + "set": "{{ True if ttl is not set }}", + "ttl": "{{ ttl }}" + } + } + } + } + } + }, + }, + { + "name": "neighbor.enforce_first_as", + "getval": re.compile( + r""" + \s*neighbor + \s+(?P<peer>\S+) + \s+enforce-first-as + *$""", + re.VERBOSE, + ), + "setval": _tmplt_bgp_neighbor, + "compval": "neighbor.enforce_first_as", + "result": { + "vrfs": { + '{{ "vrf_" + vrf|d() }}': { + "neighbor": { + "{{ peer }}": { + "peer": "{{ peer }}", + "enforce_first_as": "{{ True }}", + } + } + } + } + }, + }, + { + "name": "neighbor.export_localpref", + "getval": re.compile( + r""" + \s*neighbor + \s+(?P<peer>\S+) + \s+export-localpref + \s+(?P<pref>\d+) + *$""", + re.VERBOSE, + ), + "setval": _tmplt_bgp_neighbor, + "compval": "neighbor.export_localpref", + "result": { + "vrfs": { + '{{ "vrf_" + vrf|d() }}': { + "neighbor": { + "{{ peer }}": { + "peer": "{{ peer }}", + "export_localpref": "{{ pref }}", + } + } + } + } + }, + }, + { + "name": "neighbor.fall_over", + "getval": re.compile( + r""" + \s*neighbor + \s+(?P<peer>\S+) + \s+fall-over + \s+bfd + *$""", + re.VERBOSE, + ), + "setval": _tmplt_bgp_neighbor, + "compval": "neighbor.fall_over", + "result": { + "vrfs": { + '{{ "vrf_" + vrf|d() }}': { + "neighbor": { + "{{ peer }}": { + "peer": "{{ peer }}", + "fall_over": "{{ True }}", + } + } + } + } + }, + }, + { + "name": "neighbor.graceful_restart", + "getval": re.compile( + r""" + \s*neighbor + \s+(?P<peer>\S+) + \s+graceful-restart + *$""", + re.VERBOSE, + ), + "setval": _tmplt_bgp_neighbor, + "remval": "neighbor {{ peer }} graceful-restart", + "compval": "neighbor.graceful_restart", + "result": { + "vrfs": { + '{{ "vrf_" + vrf|d() }}': { + "neighbor": { + "{{ peer }}": { + "peer": "{{ peer }}", + "graceful_restart": "{{ True }}", + } + } + } + } + }, + }, + { + "name": "neighbor.graceful_restart_helper", + "getval": re.compile( + r""" + \s*neighbor + \s+(?P<peer>\S+) + \s+graceful-restart-helper + *$""", + re.VERBOSE, + ), + "setval": _tmplt_bgp_neighbor, + "compval": "neighbor.graceful_restart_helper", + "result": { + "vrfs": { + '{{ "vrf_" + vrf|d() }}': { + "neighbor": { + "{{ peer }}": { + "peer": "{{ peer }}", + "graceful_restart_helper": "{{ True }}", + } + } + } + } + }, + }, + { + "name": "neighbor.idle_restart_timer", + "getval": re.compile( + r""" + \s*neighbor + \s+(?P<peer>\S+) + \s+idle-restart-timer + \s+(?P<time>\d+) + *$""", + re.VERBOSE, + ), + "setval": _tmplt_bgp_neighbor, + "compval": "neighbor.idle_restart_timer", + "result": { + "vrfs": { + '{{ "vrf_" + vrf|d() }}': { + "neighbor": { + "{{ peer }}": { + "peer": "{{ peer }}", + "idle_restart_timer": "{{ time }}", + } + } + } + } + }, + }, + { + "name": "neighbor.import_localpref", + "getval": re.compile( + r""" + \s*neighbor + \s+(?P<peer>\S+) + \s+import-localpref + \s+(?P<pref>\d+) + *$""", + re.VERBOSE, + ), + "setval": _tmplt_bgp_neighbor, + "compval": "neighbor.import_localpref", + "result": { + "vrfs": { + '{{ "vrf_" + vrf|d() }}': { + "neighbor": { + "{{ peer }}": { + "peer": "{{ peer }}", + "import_localpref": "{{ pref }}", + } + } + } + } + }, + }, + { + "name": "neighbor.link_bandwidth", + "getval": re.compile( + r""" + \s*neighbor + \s+(?P<peer>\S+) + \s+link-bandwidth + \s*(?P<auto>auto)* + \s*(?P<default>default\s\S+)* + \s*(?P<update>update-delay\s\S+)* + $""", + re.VERBOSE, + ), + "setval": _tmplt_bgp_neighbor, + "compval": "neighbor.link_bandwidth", + "result": { + "vrfs": { + '{{ "vrf_" + vrf|d() }}': { + "neighbor": { + "{{ peer }}": { + "peer": "{{ peer }}", + "link_bandwidth": { + "set": "{{ True }}", + "auto": "{{ True if auto is defined }}", + "default": "{{ default.split(" ")[1] if default is defined }}", + "update_delay": "{{ update.split(" ")[1] if update is defined }}" + } + } + } + } + } + }, + }, + { + "name": "neighbor.local_as", + "getval": re.compile( + r""" + \s*neighbor + \s+(?P<peer>\S+) + \s+local-as + \s+(?P<num>\S+) + \s+no-prepend + \s+replace-as + \s*(?P<fallback>fallback) + *$""", + re.VERBOSE, + ), + "setval": _tmplt_bgp_neighbor, + "compval": "neighbor.local_as", + "result": { + "vrfs": { + '{{ "vrf_" + vrf|d() }}': { + "neighbor": { + "{{ peer }}": { + "peer": "{{ peer }}", + "local_as": { + "as_number": "{{ num }}", + "fallback": "{{ True if fallback is defined }}" + } + } + } + } + } + }, + }, + { + "name": "neighbor.local_v6_addr", + "getval": re.compile( + r""" + \s*neighbor + \s+(?P<peer>\S+) + \s+local-v6-addr + \s+(?P<addr>\S+) + *$""", + re.VERBOSE, + ), + "setval": _tmplt_bgp_neighbor, + "compval": "neighbor.local_v6_addr", + "result": { + "vrfs": { + '{{ "vrf_" + vrf|d() }}': { + "neighbor": { + "{{ peer }}": { + "peer": "{{ peer }}", + "local_v6_addr": "{{ addr }}", + } + } + } + } + }, + }, + { + "name": "neighbor.maximum_accepted_routes", + "getval": re.compile( + r""" + \s*neighbor + \s+(?P<peer>\S+) + \s+maximum-accepted-routes + \s+(?P<count>\d+) + \s*warning-limit* + \s*(?P<limit>\d+)* + $""", + re.VERBOSE, + ), + "setval": _tmplt_bgp_neighbor, + "compval": "neighbor.maximum_accepted_routes", + "result": { + "vrfs": { + '{{ "vrf_" + vrf|d() }}': { + "neighbor": { + "{{ peer }}": { + "peer": "{{ peer }}", + "maximum_accepted_routes": { + "count": "{{ count }}", + "warning_limit": "{{ limit }}" + } + } + } + } + } + }, + }, + { + "name": "neighbor.maximum_received_routes", + "getval": re.compile( + r""" + \s*neighbor + \s+(?P<peer>\S+) + \s+maximum-routes + \s+(?P<count>\d+)* + \s*(warning-limit)* + \s*(?P<limit>\d+)* + \s*(?P<percent>percent)* + \s*(?P<warning_only>warning-only)* + $""", + re.VERBOSE, + ), + "setval": _tmplt_bgp_neighbor, + "compval": "neighbor.maximum_received_routes", + "result": { + "vrfs": { + '{{ "vrf_" + vrf|d() }}': { + "neighbor": { + "{{ peer }}": { + "peer": "{{ peer }}", + "maximum_received_routes": { + "count": "{{ count }}", + "warning_limit": { + "limit_count": "{{ limit if percent is undefined }}", + "limit_percent": "{{ limit if percent is defined }}" + }, + "warning_only": "{{ True if warning_only is defined }}" + } + } + } + } + } + }, + }, + { + "name": "neighbor.metric_out", + "getval": re.compile( + r""" + \s*neighbor + \s+(?P<peer>\S+) + \s+metric-out + \s+(?P<metric>\d+) + *$""", + re.VERBOSE, + ), + "setval": _tmplt_bgp_neighbor, + "compval": "neighbor.metric_out", + "result": { + "vrfs": { + '{{ "vrf_" + vrf|d() }}': { + "neighbor": { + "{{ peer }}": { + "peer": "{{ peer }}", + "metric_out": "{{ metric }}", + } + } + } + } + }, + }, + { + "name": "neighbor.monitoring", + "getval": re.compile( + r""" + \s*neighbor + \s+(?P<peer>\S+) + \s+monitoring + *$""", + re.VERBOSE, + ), + "setval": _tmplt_bgp_neighbor, + "compval": "neighbor.monitoring", + "result": { + "vrfs": { + '{{ "vrf_" + vrf|d() }}': { + "neighbor": { + "{{ peer }}": { + "peer": "{{ peer }}", + "monitoring": "{{ True }}", + } + } + } + } + }, + }, + { + "name": "neighbor.next_hop_self", + "getval": re.compile( + r""" + \s*neighbor + \s+(?P<peer>\S+) + \s+next-hop-self + *$""", + re.VERBOSE, + ), + "setval": _tmplt_bgp_neighbor, + "compval": "neighbor.next_hop_self", + "result": { + "vrfs": { + '{{ "vrf_" + vrf|d() }}': { + "neighbor": { + "{{ peer }}": { + "peer": "{{ peer }}", + "next_hop_self": "{{ True }}", + } + } + } + } + }, + }, + { + "name": "neighbor.next_hop_unchanged", + "getval": re.compile( + r""" + \s*neighbor + \s+(?P<peer>\S+) + \s+next-hop-unchanged + *$""", + re.VERBOSE, + ), + "setval": _tmplt_bgp_neighbor, + "compval": "neighbor.next_hop_unchanged", + "result": { + "vrfs": { + '{{ "vrf_" + vrf|d() }}': { + "neighbor": { + "{{ peer }}": { + "peer": "{{ peer }}", + "next_hop_unchanged": "{{ True }}", + } + } + } + } + }, + }, + { + "name": "neighbor.next_hop_v6_addr", + "getval": re.compile( + r""" + \s*neighbor + \s+(?P<peer>\S+) + \s+next-hop-v6-addr + \s+(?P<addr>\S+) + \s+in + *$""", + re.VERBOSE, + ), + "setval": _tmplt_bgp_neighbor, + "compval": "neighbor.next_hop_v6_addr", + "result": { + "vrfs": { + '{{ "vrf_" + vrf|d() }}': { + "neighbor": { + "{{ peer }}": { + "peer": "{{ peer }}", + "next_hop_v6_addr": "{{ addr }}", + } + } + } + } + }, + }, + { + "name": "neighbor.out_delay", + "getval": re.compile( + r""" + \s*neighbor + \s+(?P<peer>\S+) + \s+out-delay + \s+(?P<delay>\d+) + *$""", + re.VERBOSE, + ), + "setval": _tmplt_bgp_neighbor, + "compval": "neighbor.out_delay", + "result": { + "vrfs": { + '{{ "vrf_" + vrf|d() }}': { + "neighbor": { + "{{ peer }}": { + "peer": "{{ peer }}", + "out_delay": "{{ delay }}", + } + } + } + } + }, + }, + { + "name": "neighbor.remote_as", + "getval": re.compile( + r""" + \s*neighbor + \s+(?P<peer>\S+) + \s+remote-as + \s+(?P<num>\S+) + *$""", + re.VERBOSE, + ), + "setval": _tmplt_bgp_neighbor, + "compval": "neighbor.remote_as", + "result": { + "vrfs": { + '{{ "vrf_" + vrf|d() }}': { + "neighbor": { + "{{ peer }}": { + "peer": "{{ peer }}", + "remote_as": "{{ num }}", + } + } + } + } + }, + }, + { + "name": "neighbor.remove_private_as", + "getval": re.compile( + r""" + \s*neighbor + \s+(?P<peer>\S+) + \s+remove-private-as + \s*(?P<all>all)* + \s*(?P<replace>replace-as)* + $""", + re.VERBOSE, + ), + "setval": _tmplt_bgp_neighbor, + "compval": "neighbor.remove_private_as", + "result": { + "vrfs": { + '{{ "vrf_" + vrf|d() }}': { + "neighbor": { + "{{ peer }}": { + "peer": "{{ peer }}", + "remove_private_as": { + "set": "{{ True if all is undefined and replace is undefined }}", + "all": "{{ True if all is defined }}", + "replace_as": "{{ True if replace is defined }}" + } + } + } + } + } + }, + }, + { + "name": "neighbor.peer_group", + "getval": re.compile( + r""" + \s*neighbor + \s+(?P<peer>\S+) + \s+peer-group + \s*(?P<name>\S+) + *$""", + re.VERBOSE, + ), + "setval": _tmplt_bgp_neighbor, + "compval": "neighbor.peer_group", + "result": { + "vrfs": { + '{{ "vrf_" + vrf|d() }}': { + "neighbor": { + "{{ peer }}": { + "peer": "{{ peer }}", + "peer_group": "{{ name if name is defined else peer}}", + } + } + } + } + }, + }, + { + "name": "neighbor.prefix_list", + "getval": re.compile( + r""" + \s*neighbor + \s+(?P<peer>\S+) + \s+prefix-list + \s+(?P<name>\S+) + \s+(?P<dir>in|out) + *$""", + re.VERBOSE, + ), + "setval": _tmplt_bgp_neighbor, + "compval": "neighbor.prefix_list", + "result": { + "vrfs": { + '{{ "vrf_" + vrf|d() }}': { + "neighbor": { + "{{ peer }}": { + "peer": "{{ peer }}", + "prefix_list": { + "name": "{{ name }}", + "direction": "{{ dir }}" + } + } + } + } + } + }, + }, + { + "name": "neighbor.route_map", + "getval": re.compile( + r""" + \s*neighbor + \s+(?P<peer>\S+) + \s+route-map + \s+(?P<name>\S+) + \s+(?P<dir>in|out) + *$""", + re.VERBOSE, + ), + "setval": _tmplt_bgp_neighbor, + "compval": "neighbor.route_map", + "result": { + "vrfs": { + '{{ "vrf_" + vrf|d() }}': { + "neighbor": { + "{{ peer }}": { + "peer": "{{ peer }}", + "route_map": { + "name": "{{ name }}", + "direction": "{{ dir }}" + } + } + } + } + } + }, + }, + { + "name": "neighbor.route_reflector_client", + "getval": re.compile( + r""" + \s*neighbor + \s+(?P<peer>\S+) + \s+route-reflector-client + *$""", + re.VERBOSE, + ), + "setval": _tmplt_bgp_neighbor, + "compval": "neighbor.route_reflector_client", + "result": { + "vrfs": { + '{{ "vrf_" + vrf|d() }}': { + "neighbor": { + "{{ peer }}": { + "peer": "{{ peer }}", + "route_reflector_client": "{{ True }}", + } + } + } + } + }, + }, + { + "name": "neighbor.route_to_peer", + "getval": re.compile( + r""" + \s*neighbor + \s+(?P<peer>\S+) + \s+route-to-peer + *$""", + re.VERBOSE, + ), + "setval": _tmplt_bgp_neighbor, + "compval": "neighbor.route_to_peer", + "result": { + "vrfs": { + '{{ "vrf_" + vrf|d() }}': { + "neighbor": { + "{{ peer }}": { + "peer": "{{ peer }}", + "route_to_peer": "{{ True }}", + } + } + } + } + }, + }, + { + "name": "neighbor.send_community", + "getval": re.compile( + r""" + \s*neighbor + \s+(?P<peer>\S+) + \s+send-community + \s+(?P<comm>add|extended|link-bandwidth|remove|standard)* + \s*(?P<attr>extended|link-bandwidth|standard)* + \s*(?P<link>aggregate|divide)* + \s*(?P<div>equal|ratio)* + \s*(?P<speed>\S+)* + $""", + re.VERBOSE, + ), + "setval": _tmplt_bgp_neighbor, + "compval": "neighbor.send_community", + "result": { + "vrfs": { + '{{ "vrf_" + vrf|d() }}': { + "neighbor": { + "{{ peer }}": { + "peer": "{{ peer }}", + "send_community": { + "set": "{{ True if comm is not defined }}", + "community_attribute": "{{ comm }}", + "sub_attribute": "{{ attr }}", + "link_bandwidth_attribute": "{{ link }}", + "speed": "{{ speed }}", + "divide": "{{ div }}" + } + } + } + } + } + }, + }, + { + "name": "neighbor.shutdown", + "getval": re.compile( + r""" + \s*neighbor + \s+(?P<peer>\S+) + \s+shutdown + *$""", + re.VERBOSE, + ), + "setval": _tmplt_bgp_neighbor, + "compval": "neighbor.shutdown", + "result": { + "vrfs": { + '{{ "vrf_" + vrf|d() }}': { + "neighbor": { + "{{ peer }}": { + "peer": "{{ peer }}", + "shutdown": "{{ True }}", + } + } + } + } + }, + }, + { + "name": "neighbor.soft_reconfiguration", + "getval": re.compile( + r""" + \s*neighbor + \s+(?P<peer>\S+) + \s+soft-reconfiguration + \s+inbound + \s*(?P<all>\S+)* + $""", + re.VERBOSE, + ), + "setval": _tmplt_bgp_neighbor, + "compval": "neighbor.soft_reconfiguration", + "result": { + "vrfs": { + '{{ "vrf_" + vrf|d() }}': { + "neighbor": { + "{{ peer }}": { + "peer": "{{ peer }}", + "soft_reconfiguration": "{{ 'all' if all is defined else 'None' }}", + } + } + } + } + }, + }, + { + "name": "neighbor.transport", + "getval": re.compile( + r""" + \s*neighbor + \s+(?P<peer>\S+) + \s+transport + \s+(?P<mode>\S+) + \s*(passive)* + \s*(?P<port>\d+)* + $""", + re.VERBOSE, + ), + "setval": _tmplt_bgp_neighbor, + "compval": "neighbor.transport`", + "result": { + "vrfs": { + '{{ "vrf_" + vrf|d() }}': { + "neighbor": { + "{{ peer }}": { + "peer": "{{ peer }}", + "transport": { + "connection_mode": "{{ mode }}", + "remote_port": "{{ port if port is defined }}" + } + } + } + } + } + }, + }, + { + "name": "neighbor.timers", + "getval": re.compile( + r""" + \s*neighbor + \s+(?P<peer>\S+) + \s+timers + \s+(?P<keepalive>\d+) + \s+(?P<hold>\d+) + *$""", + re.VERBOSE, + ), + "setval": _tmplt_bgp_neighbor, + "compval": "neighbor.timers", + "result": { + "vrfs": { + '{{ "vrf_" + vrf|d() }}': { + "neighbor": { + "{{ peer }}": { + "peer": "{{ peer }}", + "timers": { + "keepalive": "{{ keepalive }}", + "holdtime": "{{ holdtime }}" + } + } + } + } + } + }, + }, + { + "name": "neighbor.ttl", + "getval": re.compile( + r""" + \s*neighbor + \s+(?P<peer>\S+) + \s+ttl + \s+maximum-hops + \s+(?P<hop>\d+) + *$""", + re.VERBOSE, + ), + "setval": _tmplt_bgp_neighbor, + "compval": "neighbor.ttl", + "result": { + "vrfs": { + '{{ "vrf_" + vrf|d() }}': { + "neighbor": { + "{{ peer }}": { + "peer": "{{ peer }}", + "ttl": "{{ hop }}", + } + } + } + } + }, + }, + { + "name": "neighbor.update_source", + "getval": re.compile( + r""" + \s*neighbor + \s+(?P<peer>\S+) + \s+update-source + \s+(?P<src>.*) + *$""", + re.VERBOSE, + ), + "setval": _tmplt_bgp_neighbor, + "compval": "neighbor.update_source", + "result": { + "vrfs": { + '{{ "vrf_" + vrf|d() }}': { + "neighbor": { + "{{ peer }}": { + "peer": "{{ peer }}", + "update_source": "{{ src }}", + } + } + } + } + }, + }, + { + "name": "neighbor.weight", + "getval": re.compile( + r""" + \s*neighbor + \s+(?P<peer>\S+) + \s+weight + \s+(?P<val>\d+) + *$""", + re.VERBOSE, + ), + "setval": _tmplt_bgp_neighbor, + "compval": "neighbor.weight", + "result": { + "vrfs": { + '{{ "vrf_" + vrf|d() }}': { + "neighbor": { + "{{ peer }}": { + "peer": "{{ peer }}", + "weight": "{{ val }}", + } + } + } + } + }, + }, + { + "name": "network", + "getval": re.compile( + r""" + \s*network + \s+(?P<address>\S+) + \s*(route-map)* + \s*(?P<route_map>\S+)* + $""", + re.VERBOSE, + ), + "setval": _tmplt_bgp_network, + "compval": "network", + "result": { + "vrfs": { + '{{ "vrf_" + vrf|d() }}': { + "network": { + "{{ address }}": { + "address": "{{ address }}", + "route_map": "{{ route_map }}", + } + } + } + } + }, + }, + { + "name": "route_target", + "getval": re.compile( + r""" + \s*route-target + \s+(?P<action>\S+) + \s+(?P<target>\S+) + *$""", + re.VERBOSE, + ), + "setval": _tmplt_bgp_route_target, + "compval": "route_target", + "result": { + "vrfs": { + '{{ "vrf_" + vrf|d() }}': { + "route_target": { + "action": "{{ action }}", + "target": "{{ target }}", + } + } + } + }, + }, + { + "name": "router_id", + "getval": re.compile( + r""" + \s*router-id + \s+(?P<id>\S+) + *$""", + re.VERBOSE, + ), + "setval": _tmplt_bgp_router_id, + "result": { + "vrfs": { + '{{ "vrf_" + vrf|d() }}': { + "router_id": "{{ id }}" + } + } + }, + }, + { + "name": "shutdown", + "getval": re.compile( + r""" + \s*shutdown + *$""", + re.VERBOSE, + ), + "setval": _tmplt_bgp_shutdown, + "result": { + "vrfs": { + '{{ "vrf_" + vrf|d() }}': { + "shutdown": "{{ True }}" + } + } + }, + }, + { + "name": "timers", + "getval": re.compile( + r""" + \s*timers + \s+bgp + \s+(?P<keepalive>\d+) + \s+(?P<holdtime>\d+) + *$""", + re.VERBOSE, + ), + "setval": _tmplt_bgp_timers, + "compval": "timers", + "result": { + "vrfs": { + '{{ "vrf_" + vrf|d() }}': { + "timers": { + "keepalive": "{{ keepalive }}", + "holdtime": "{{ holdtime }}", + } + } + } + }, + }, + { + "name": "ucmp_fec", + "getval": re.compile( + r""" + \s*ucmp + \s+fec + \s+threshold + \s+(?P<trigger>trigger \d+) + \s+(?P<clear>clear \d+) + \s+warning-only + *$""", + re.VERBOSE, + ), + "setval": _tmplt_bgp_ucmp, + "compval": "ucmp.fec", + "result": { + "vrfs": { + '{{ "vrf_" + vrf|d() }}': { + "ucmp": { + "fec": { + "trigger": "{{ trigger }}", + "clear": "{{ clear }}" + } + } + } + } + }, + }, + { + "name": "ucmp_link_bandwidth", + "getval": re.compile( + r""" + \s*ucmp + \s+link-bandwidth + \s*(?P<ucmp_mode>recursive|encoding-weighted) + \s*(?P<update_delay>update-delay \d+) + *$""", + re.VERBOSE, + ), + "setval": _tmplt_bgp_ucmp, + "compval": "ucmp.link_bandwidth", + "result": { + "vrfs": { + '{{ "vrf_" + vrf|d() }}': { + "ucmp": { + "link_bandwidth": { + "mode": "{{ ucmp_mode }}", + "update_delay": "{{ update_delay.split(" ")[1] }}" + } + } + } + } + }, + }, + { + "name": "ucmp_mode", + "getval": re.compile( + r""" + \s*ucmp + \s+mode + \s+(?P<ucmp_set>\d+) + \s*(?P<nexthop>\d+) + *$""", + re.VERBOSE, + ), + "setval": _tmplt_bgp_ucmp, + "compval": "ucmp.mode", + "result": { + "vrfs": { + '{{ "vrf_" + vrf|d() }}': { + "ucmp": { + "mode": { + "set": "{{ True if ucmp_set == '1'}}", + "nexthops": "{{ nexthop }}" + } + } + } + } + }, + }, + { + "name": "update", + "getval": re.compile( + r""" + \s*update + \s+(?P<wait>\S+) + \s*(?P<size>batch-size\s\d+)* + $""", + re.VERBOSE, + ), + "setval": _tmplt_bgp_update, + "compval": "update", + "result": { + "vrfs": { + '{{ "vrf_" + vrf|d() }}': { + "update": { + "wait_for": "{{ wait }}", + "batch_size": "{{ size.split(" ")[1] }}" + } + } + } + }, + }, + { + "name": "vlan", + "getval": re.compile( + r""" + \s*vlan + \s+(?P<id>\d+) + *$""", + re.VERBOSE, + ), + "setval": _tmplt_bgp_vlan, + "result": { + "vrfs": { + '{{ "vrf_" + vrf|d() }}': { + "vlan": "{{ id }}", + } + } + }, + }, + { + "name": "vlan_aware_bundle", + "getval": re.compile( + r""" + \s*vlan-aware-bundle + \s+(?P<bundle>.+) + *$""", + re.VERBOSE, + ), + "setval": _tmplt_bgp_vlan_aware_bundle, + "result": { + "vrfs": { + '{{ "vrf_" + vrf|d() }}': { + "vlan_aware_bundle": "{{ bundle }}", + } + } + }, + }, + ] + # fmt: on diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/rm_templates/ospf_interfaces.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/rm_templates/ospf_interfaces.py new file mode 100644 index 00000000..bd3e5774 --- /dev/null +++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/rm_templates/ospf_interfaces.py @@ -0,0 +1,1082 @@ +# -*- coding: utf-8 -*- +# Copyright 2020 Red Hat +# 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 + +""" +The Ospf_interfaces parser templates file. This contains +a list of parser definitions and associated functions that +facilitates both facts gathering and native command generation for +the given network resource. +""" + +import re +from ansible_collections.ansible.netcommon.plugins.module_utils.network.common.network_template import ( + NetworkTemplate, +) + + +def _tmplt_ospf_int_authentication(config_data): + if "authentication_v2" in config_data: + command = "ip ospf authentication" + if "message_digest" in config_data["authentication_v2"]: + command += " message-digest" + return command + if "authentication_v3" in config_data: + command = "ospfv3 authentication ipsec spi " + command += "{spi} {algorithm}".format( + **config_data["authentication_v3"] + ) + if "passphrase" in config_data["authentication_v3"]: + command += " passphrase" + if "keytype" in config_data["authentication_v3"]: + command += " {keytype}".format(**config_data["authentication_v3"]) + if "passphrase" not in config_data["authentication_v3"]: + command += " {key}".format(**config_data["authentication_v3"]) + else: + command += " {passphrase}".format( + **config_data["authentication_v3"] + ) + return command + + +def _tmplt_ospf_int_encryption_v3(config_data): + if "encryption" in config_data: + command = "ospfv3 encryption ipsec spi ".format(**config_data) + command += "{spi} esp {encryption} {algorithm}".format( + **config_data["encryption"] + ) + if "passphrase" in config_data["encryption"]: + command += " passphrase" + if "keytype" in config_data["encryption"]: + command += " {keytype}".format(**config_data["encryption"]) + if "passphrase" not in config_data["encryption"]: + command += " {key}".format(**config_data["encryption"]) + else: + command += " {passphrase}".format(**config_data["encryption"]) + return command + + +def _tmplt_ospf_int_authentication_key(config_data): + if "authentication_key" in config_data: + command = "ip ospf authentication-key" + if "encryption" in config_data["authentication_key"]: + command += " {encryption} {key}".format( + **config_data["authentication_key"] + ) + else: + command += " {key}".format(**config_data["authentication_key"]) + return command + + +def _tmplt_ospf_int_cost(config_data): + if config_data["afi"] == "ipv4": + command = "ip ospf cost {cost}".format(**config_data) + else: + command = "ospfv3 cost {cost}".format(**config_data) + return command + + +def _tmplt_ospf_int_bfd(config_data): + if config_data["afi"] == "ipv4": + command = "ip ospf bfd" + else: + command = "ospfv3 bfd" + return command + + +def _tmplt_ospf_int_hello_interval(config_data): + if "ip_params" in config_data: + command = "ospfv3 {afi} hello-interval {hello_interval}".format( + **config_data["ip_params"] + ) + else: + if config_data["afi"] == "ipv4": + command = "ip ospf hello-interval {hello_interval}".format( + **config_data + ) + else: + command = "ospfv3 hello-interval {hello_interval}".format( + **config_data + ) + return command + + +def _tmplt_ospf_int_mtu_ignore(config_data): + if "ip_params" in config_data: + command = "ospfv3 {afi} mtu-ignore".format(**config_data["ip_params"]) + else: + if config_data["afi"] == "ipv4": + command = "ip ospf mtu-ignore" + else: + command = "ospfv3 mtu-ignore" + return command + + +def _tmplt_ospf_int_network(config_data): + if "ip_params" in config_data: + command = "ospfv3 {afi} network {network}".format( + **config_data["ip_params"] + ) + else: + if config_data["afi"] == "ipv4": + command = "ip ospf network {network}".format(**config_data) + else: + command = "ospfv3 network {network}".format(**config_data) + return command + + +def _tmplt_ospf_int_priority(config_data): + if "ip_params" in config_data: + command = "ospfv3 {afi} priority {priority}".format( + **config_data["ip_params"] + ) + else: + if config_data["afi"] == "ipv4": + command = "ip ospf priority {priority}".format(**config_data) + else: + command = "ospfv3 priority {priority}".format(**config_data) + return command + + +def _tmplt_ospf_int_retransmit_interval(config_data): + if "ip_params" in config_data: + command = "ospfv3 {afi} retransmit-interval {retransmit_interval}".format( + **config_data["ip_params"] + ) + else: + if config_data["afi"] == "ipv4": + command = "ip ospf retransmit-interval {retransmit_interval}".format( + **config_data + ) + else: + command = "ospfv3 retransmit-interval {retransmit_interval}".format( + **config_data + ) + return command + + +def _tmplt_ospf_int_transmit_delay(config_data): + if "ip_params" in config_data: + command = "ospfv3 {afi} transmit-delay {transmit_delay}".format( + **config_data["ip_params"] + ) + else: + if config_data["afi"] == "ipv4": + command = "ip ospf transmit-delay {transmit_delay}".format( + **config_data + ) + else: + command = "ospfv3 transmit-delay {transmit_delay}".format( + **config_data + ) + return command + + +def _tmplt_ospf_int_dead_interval(config_data): + if "ip_params" in config_data: + command = "ospfv3 {afi} dead-interval {dead_interval}".format( + **config_data["ip_params"] + ) + else: + if config_data["afi"] == "ipv4": + command = "ip ospf dead-interval {dead_interval}".format( + **config_data + ) + else: + command = "ospfv3 dead-interval {dead_interval}".format( + **config_data + ) + return command + + +class Ospf_interfacesTemplate(NetworkTemplate): + def __init__(self, lines=None): + super(Ospf_interfacesTemplate, self).__init__(lines=lines, tmplt=self) + + PARSERS = [ + { + "name": "interfaces", + "getval": re.compile( + r""" + ^interface + \s+(?P<name>\S+) + $""", + re.VERBOSE, + ), + "setval": "interface {{ name }}", + "result": {"name": "{{ name }}"}, + "shared": True, + }, + { + "name": "area", + "getval": re.compile( + r""" + \s*ip + \s+ospf + \s+area + \s+(?P<area_id>\S+)* + $""", + re.VERBOSE, + ), + "setval": "ip ospf area {{ area.area_id }}", + "compval": "area", + "result": { + "address_family": { + "{{ 'ipv4' }}": { + "afi": '{{ "ipv4" }}', + "area": {"area_id": "{{ area_id }}"}, + } + } + }, + }, + { + "name": "authentication_v2", + "getval": re.compile( + r""" + \s*ip + \s+ospf + \s+authentication + \s*(?P<message_digest>\S+)* + $""", + re.VERBOSE, + ), + "setval": _tmplt_ospf_int_authentication, + "compval": "authentication_v2", + "result": { + "address_family": { + "{{ 'ipv4' }}": { + "afi": '{{ "ipv4" }}', + "authentication_v2": { + "set": "{{ True if message_digest is undefined }}", + "message_digest": "{{ True if message_digest is defined }}", + }, + } + } + }, + }, + { + "name": "authentication_v3", + "getval": re.compile( + r""" + \s*ospfv3 + \s+authentication + \s+ipsec + \s+spi + \s+(?P<val>\d+) + \s+(?P<algorithm>md5|sha1) + \s*(?P<passphrase>passphrase)* + \s*(?P<type>0|7)* + \s*(?P<line>\S+) + *$""", + re.VERBOSE, + ), + "setval": _tmplt_ospf_int_authentication, + "compval": "authentication_v3", + "result": { + "address_family": { + "{{ 'ipv6' }}": { + "afi": '{{ "ipv6" }}', + "authentication_v3": { + "spi": "{{ val }}", + "algorithm": "{{ algorithm }}", + "keytype": "{{ type }}", + "passphrase": "{{ line if passphrase is defined }}", + "key": "{{ str(line) if passphrase is undefined }}", + }, + } + } + }, + }, + { + "name": "authentication_key", + "getval": re.compile( + r""" + \s*ip + \s+ospf + \s+authentication-key + \s*(?P<encryption>\d+)* + \s*(?P<line>\S+)* + $""", + re.VERBOSE, + ), + "setval": _tmplt_ospf_int_authentication_key, + "compval": "authentication_key", + "result": { + "address_family": { + "{{ 'ipv4' }}": { + "afi": '{{ "ipv4" }}', + "authentication_key": { + "encryption": "{{ encryption }}", + "key": "{{ line }}", + }, + } + } + }, + }, + { + "name": "bfd", + "getval": re.compile( + r""" + \s*ip + \s+ospf + \s+bfd + *$""", + re.VERBOSE, + ), + "setval": _tmplt_ospf_int_bfd, + "compval": "bfd", + "result": { + "address_family": { + "{{ 'ipv4' }}": { + "afi": '{{ "ipv4" }}', + "bfd": "{{ True }}", + } + } + }, + }, + { + "name": "deadinterval", + "getval": re.compile( + r""" + \s*ip + \s+ospf + \s+dead-interval + \s+(?P<interval>\d+) + *$""", + re.VERBOSE, + ), + "setval": _tmplt_ospf_int_dead_interval, + "compval": "dead_interval", + "result": { + "address_family": { + "{{ 'ipv4' }}": { + "afi": '{{ "ipv4" }}', + "dead_interval": "{{ interval }}", + } + } + }, + }, + { + "name": "encryption", + "getval": re.compile( + r""" + \s*ospfv3 + \s+encryption + \s+ipsec + \s+spi + \s+(?P<val>\d+) + \s+esp + \s+(?P<encryption>\S+) + \s*(?P<algorithm>md5|sha1) + \s*(?P<passphrase>passphrase) + \s*(?P<type>0|7) + \s*(?P<line>\S+) + *$""", + re.VERBOSE, + ), + "setval": _tmplt_ospf_int_encryption_v3, + "compval": "encryption_v3", + "result": { + "address_family": { + "{{ 'ipv6' }}": { + "afi": '{{ "ipv6" }}', + "encryption_v3": { + "spi": "{{ val }}", + "encryption": "{{ encryption }}", + "algorithm": "{{ algorithm }}", + "keytype": "{{ type }}", + "passphrase": "{{ line if passphrase is defined }}", + "key": "{{ str(line) if passphrase is undefined }}", + }, + } + } + }, + }, + { + "name": "hellointerval", + "getval": re.compile( + r""" + \s*ip + \s+ospf + \s+hello-interval + \s+(?P<interval>\d+) + *$""", + re.VERBOSE, + ), + "setval": _tmplt_ospf_int_hello_interval, + "compval": "hello_interval", + "result": { + "address_family": { + "{{ 'ipv4' }}": { + "afi": '{{ "ipv4" }}', + "hello_interval": "{{ interval }}", + } + } + }, + }, + { + "name": "bfd", + "getval": re.compile( + r""" + \s+ospfv3 + \s+bfd + *$""", + re.VERBOSE, + ), + "setval": _tmplt_ospf_int_bfd, + "compval": "bfd", + "result": { + "address_family": { + "{{ 'ipv6' }}": { + "afi": '{{ "ipv6" }}', + "bfd": "{{ True }}", + } + } + }, + }, + { + "name": "cost", + "getval": re.compile( + r""" + \s+ip + \s+ospf + \s+cost + \s+(?P<val>\d+) + *$""", + re.VERBOSE, + ), + "setval": _tmplt_ospf_int_cost, + "compval": "cost", + "result": { + "address_family": { + "{{ 'ipv4' }}": { + "afi": '{{ "ipv4" }}', + "cost": "{{ val }}", + } + } + }, + }, + { + "name": "cost", + "getval": re.compile( + r""" + \s+ospfv3 + \s+cost + \s+(?P<cost>\d+) + *$""", + re.VERBOSE, + ), + "setval": _tmplt_ospf_int_cost, + "compval": "cost", + "result": { + "address_family": { + "{{ 'ipv6' }}": { + "afi": '{{ "ipv6" }}', + "cost": "{{ cost }}", + } + } + }, + }, + { + "name": "deadinterval", + "getval": re.compile( + r""" + \s+ospfv3 + \s+dead-interval + \s+(?P<interval>\d+) + *$""", + re.VERBOSE, + ), + "setval": _tmplt_ospf_int_dead_interval, + "compval": "dead_interval", + "result": { + "address_family": { + "{{ 'ipv6' }}": { + "afi": '{{ "ipv6" }}', + "dead_interval": "{{ interval }}", + } + } + }, + }, + { + "name": "hellointerval", + "getval": re.compile( + r""" + \s+ospfv3 + \s+hello-interval + \s+(?P<interval>\d+) + *$""", + re.VERBOSE, + ), + "setval": _tmplt_ospf_int_hello_interval, + "compval": "hello_interval", + "result": { + "address_family": { + "{{ 'ipv6' }}": { + "afi": '{{ "ipv6" }}', + "hello_interval": "{{ interval }}", + } + } + }, + }, + { + "name": "ip_params_area", + "getval": re.compile( + r""" + \s*ospfv3 + \s+(?P<afi>ipv4|ipv6) + \s+area + \s+(?P<area_id>\S+) + *$""", + re.VERBOSE, + ), + "setval": "ospfv3 {{ ip_params.afi }} area {{ ip_params.area.area_id }}", + "compval": "ip_params.area", + "result": { + "address_family": { + "{{ 'ipv6' }}": { + "afi": '{{ "ipv6" }}', + "ip_params": { + "{{ afi }}": { + "afi": "{{ afi }}", + "area": {"area_id": "{{ area_id }}"}, + } + }, + } + } + }, + }, + { + "name": "ip_params_bfd", + "getval": re.compile( + r""" + \s*ospfv3 + \s+(?P<afi>ipv4|ipv6) + \s+bfd + *$""", + re.VERBOSE, + ), + "setval": "ospfv3 {{ afi }} bfd", + "compval": "ip_params.bfd", + "result": { + "address_family": { + "{{ 'ipv6' }}": { + "afi": '{{ "ipv6" }}', + "ip_params": { + "{{ afi }}": { + "afi": "{{ afi }}", + "bfd": "{{ True }}", + } + }, + } + } + }, + }, + { + "name": "ip_params_cost", + "getval": re.compile( + r""" + \s*ospfv3 + \s+(?P<afi>ipv4|ipv6) + \s+cost + \s+(?P<cost>\d+) + *$""", + re.VERBOSE, + ), + "setval": "ospfv3 {{ afi }} cost {{ cost }}", + "compval": "ip_params.cost", + "result": { + "address_family": { + "{{ 'ipv6' }}": { + "afi": '{{ "ipv6" }}', + "ip_params": { + "{{ afi }}": { + "afi": "{{ afi }}", + "cost": "{{ cost }}", + } + }, + } + } + }, + }, + { + "name": "ip_params_dead_interval", + "getval": re.compile( + r""" + \s*ospfv3 + \s+(?P<afi>ipv4|ipv6) + \s+dead-interval + \s+(?P<interval>\d+) + *$""", + re.VERBOSE, + ), + "setval": _tmplt_ospf_int_dead_interval, + "compval": "ip_params.dead_interval", + "result": { + "address_family": { + "{{ 'ipv6' }}": { + "afi": '{{ "ipv6" }}', + "ip_params": { + "{{ afi }}": { + "afi": "{{ afi }}", + "dead_interval": "{{ interval }}", + } + }, + } + } + }, + }, + { + "name": "ip_params_hello_interval", + "getval": re.compile( + r""" + \s*ospfv3 + \s+(?P<afi>ipv4|ipv6) + \s+hello-interval + \s+(?P<interval>\d+) + *$""", + re.VERBOSE, + ), + "setval": "ospfv3 {{ ip_params.afi }} hello-interval {{ ip_params.hello_interval }}", + "compval": "ip_params.hello_interval", + "result": { + "address_family": { + "{{ 'ipv6' }}": { + "afi": '{{ "ipv6" }}', + "ip_params": { + "{{ afi }}": { + "afi": "{{ afi }}", + "hello_interval": "{{ interval }}", + } + }, + } + } + }, + }, + { + "name": "ip_params_mtu_ignore", + "getval": re.compile( + r""" + \s*ospfv3 + \s+(?P<afi>ipv4|ipv6) + \s+(?P<mtu>mtu-ignore) + *$""", + re.VERBOSE, + ), + "setval": _tmplt_ospf_int_mtu_ignore, + "compval": "ip_params.mtu_ignore", + "result": { + "address_family": { + "{{ 'ipv6' }}": { + "afi": '{{ "ipv6" }}', + "ip_params": { + "{{ afi }}": { + "afi": "{{ afi }}", + "mtu_ignore": "{{ True if mtu is defined}}", + } + }, + } + } + }, + }, + { + "name": "ip_params_network", + "getval": re.compile( + r""" + \s*ospfv3 + \s+(?P<afi>ipv4|ipv6) + \s+network + \s+(?P<val>\S+) + *$""", + re.VERBOSE, + ), + "setval": _tmplt_ospf_int_network, + "compval": "ip_params.network", + "result": { + "address_family": { + "{{ 'ipv6' }}": { + "afi": '{{ "ipv6" }}', + "ip_params": { + "{{ afi }}": { + "afi": "{{ afi }}", + "network": "{{ val }}", + } + }, + } + } + }, + }, + { + "name": "ip_params_priority", + "getval": re.compile( + r""" + \s*ospfv3 + \s+(?P<afi>ipv4|ipv6) + \s+priority + \s+(?P<val>\d+) + *$""", + re.VERBOSE, + ), + "setval": _tmplt_ospf_int_priority, + "compval": "ip_params.priority", + "result": { + "address_family": { + "{{ 'ipv6' }}": { + "afi": '{{ "ipv6" }}', + "ip_params": { + "{{ afi }}": { + "afi": "{{ afi }}", + "priority": "{{ val }}", + } + }, + } + } + }, + }, + { + "name": "ip_params_passive_interface", + "getval": re.compile( + r""" + \s*ospfv3 + \s+(?P<afi>ipv4|ipv6) + \s+passive-interface + *$""", + re.VERBOSE, + ), + "setval": "ospfv3 {{ ip_params.afi }} passive-interface", + "compval": "ip_params.passive_interface", + "result": { + "address_family": { + "{{ 'ipv6' }}": { + "afi": '{{ "ipv6" }}', + "ip_params": { + "{{ afi }}": { + "afi": "{{ afi }}", + "passive_interface": "{{ True }}", + } + }, + } + } + }, + }, + { + "name": "ip_params_retransmit_interval", + "getval": re.compile( + r""" + \s*ospfv3 + \s+(?P<afi>ipv4|ipv6) + \s+retransmit-interval + \s+(?P<val>\d+) + *$""", + re.VERBOSE, + ), + "setval": _tmplt_ospf_int_retransmit_interval, + "compval": "ip_params.retransmit_interval", + "result": { + "address_family": { + "{{ 'ipv6' }}": { + "afi": '{{ "ipv6" }}', + "ip_params": { + "{{ afi }}": { + "afi": "{{ afi }}", + "retransmit_interval": "{{ val }}", + } + }, + } + } + }, + }, + { + "name": "ip_params_transmit_delay", + "getval": re.compile( + r""" + \s*ospfv3 + \s+(?P<afi>ipv4|ipv6) + \s+transmit-delay + \s+(?P<val>\d+) + *$""", + re.VERBOSE, + ), + "setval": _tmplt_ospf_int_transmit_delay, + "compval": "ip_params.transmit_delay", + "result": { + "address_family": { + "{{ 'ipv6' }}": { + "afi": '{{ "ipv6" }}', + "ip_params": { + "{{ afi }}": { + "afi": "{{ afi }}", + "transmit_delay": "{{ val }}", + } + }, + } + } + }, + }, + { + "name": "mtu_ignore", + "getval": re.compile( + r""" + \s*ospfv3 + \s+mtu-ignore + *$""", + re.VERBOSE, + ), + "setval": _tmplt_ospf_int_mtu_ignore, + "compval": "mtu_ignore", + "result": { + "address_family": { + "{{ 'ipv6' }}": { + "afi": '{{ "ipv6" }}', + "mtu_ignore": "{{ True }}", + } + } + }, + }, + { + "name": "network", + "getval": re.compile( + r""" + \s*ospfv3 + \s+network + \s+(?P<interface>\S+) + *$""", + re.VERBOSE, + ), + "setval": _tmplt_ospf_int_network, + "compval": "network", + "result": { + "address_family": { + "{{ 'ipv6' }}": { + "afi": '{{ "ipv6" }}', + "network": "{{ interface }}", + } + } + }, + }, + { + "name": "priority", + "getval": re.compile( + r""" + \s*ospfv3 + \s+priority + \s+(?P<val>\d+) + *$""", + re.VERBOSE, + ), + "setval": _tmplt_ospf_int_priority, + "compval": "priority", + "result": { + "address_family": { + "{{ 'ipv6' }}": { + "afi": '{{ "ipv6" }}', + "priority": "{{ val }}", + } + } + }, + }, + { + "name": "passive_interface", + "getval": re.compile( + r""" + \s*ospfv3 + \s+passive-interface + *$""", + re.VERBOSE, + ), + "setval": "ospfv3 passive-interface", + "compval": "passive_interface", + "result": { + "address_family": { + "{{ 'ipv6' }}": { + "afi": '{{ "ipv6" }}', + "passive_interface": "{{ True }}", + } + } + }, + }, + { + "name": "retransmit_interval", + "getval": re.compile( + r""" + \s*ospfv3 + \s+retransmit-interval + \s+(?P<val>\d+) + *$""", + re.VERBOSE, + ), + "setval": _tmplt_ospf_int_retransmit_interval, + "compval": "retransmit_interval", + "result": { + "address_family": { + "{{ 'ipv6' }}": { + "afi": '{{ "ipv6" }}', + "retransmit_interval": "{{ val }}", + } + } + }, + }, + { + "name": "transmit_delay", + "getval": re.compile( + r""" + \s*ospfv3 + \s+transmit-delay + \s+(?P<val>\d+) + *$""", + re.VERBOSE, + ), + "setval": _tmplt_ospf_int_transmit_delay, + "compval": "transmit_delay", + "result": { + "address_family": { + "{{ 'ipv6' }}": { + "afi": '{{ "ipv6" }}', + "transmit_delay": "{{ val }}", + } + } + }, + }, + { + "name": "mtu_ignore", + "getval": re.compile( + r""" + \s*ip + \s+ospf + \s+mtu-ignore + *$""", + re.VERBOSE, + ), + "setval": _tmplt_ospf_int_mtu_ignore, + "compval": "mtu_ignore", + "result": { + "address_family": { + "{{ 'ipv4' }}": { + "afi": '{{ "ipv4" }}', + "mtu_ignore": "{{ True }}", + } + } + }, + }, + { + "name": "network", + "getval": re.compile( + r""" + \s*ip + \s+ospf + \s+network + \s+(?P<interface>\S+) + *$""", + re.VERBOSE, + ), + "setval": _tmplt_ospf_int_network, + "compval": "network", + "result": { + "address_family": { + "{{ 'ipv4' }}": { + "afi": '{{ "ipv4" }}', + "network": "{{ interface }}", + } + } + }, + }, + { + "name": "priority", + "getval": re.compile( + r""" + \s*ip ospf + \s+priority + \s+(?P<val>\d+) + *$""", + re.VERBOSE, + ), + "setval": _tmplt_ospf_int_priority, + "compval": "priority", + "result": { + "address_family": { + "{{ 'ipv4' }}": { + "afi": '{{ "ipv4" }}', + "priority": "{{ val }}", + } + } + }, + }, + { + "name": "retransmit_interval", + "getval": re.compile( + r""" + \s*ip + \s+ospf + \s+retransmit-interval + \s+(?P<val>\d+) + *$""", + re.VERBOSE, + ), + "setval": _tmplt_ospf_int_retransmit_interval, + "compval": "retransmit_interval", + "result": { + "address_family": { + "{{ 'ipv4' }}": { + "afi": '{{ "ipv4" }}', + "retransmit_interval": "{{ val }}", + } + } + }, + }, + { + "name": "transmit_delay", + "getval": re.compile( + r""" + \s*ip + \s+ospf + \s+transmit-delay + \s+(?P<val>\d+) + *$""", + re.VERBOSE, + ), + "setval": _tmplt_ospf_int_transmit_delay, + "compval": "transmit_delay", + "result": { + "address_family": { + "{{ 'ipv4' }}": { + "afi": '{{ "ipv4" }}', + "transmit_delay": "{{ val }}", + } + } + }, + }, + { + "name": "message_digest_key", + "getval": re.compile( + r""" + \s*ip + \s+ospf + \s+message-digest-key + \s+(?P<id>\d+) + \s+md5 + \s*(?P<type>0|7)* + \s+(?P<line>\S+) + *$""", + re.VERBOSE, + ), + "setval": "ip ospf message-digest-key {{ message_digest_key.key_id }} md5 {{ message_digest_key.encryption }} {{ message_digest_key.key }}", + "compval": "message_digest_key", + "result": { + "address_family": { + "{{ 'ipv4' }}": { + "afi": '{{ "ipv4" }}', + "message_digest_key": { + "key_id": "{{ id }}", + "encryption": "{{ type }}", + "key": "{{ line }}", + }, + } + } + }, + }, + ] diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/rm_templates/ospfv3.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/rm_templates/ospfv3.py new file mode 100644 index 00000000..857521b8 --- /dev/null +++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/rm_templates/ospfv3.py @@ -0,0 +1,1055 @@ +# -*- coding: utf-8 -*- +# Copyright 2020 Red Hat +# 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 + +""" +The Ospfv3 parser templates file. This contains +a list of parser definitions and associated functions that +facilitates both facts gathering and native command generation for +the given network resource. +""" + +import re +from ansible_collections.ansible.netcommon.plugins.module_utils.network.common.network_template import ( + NetworkTemplate, +) + + +def _tmplt_ospf_vrf_cmd(process): + command = "router ospfv3" + vrf = "{vrf}".format(**process) + if "vrf" in process and vrf != "default": + command += " vrf " + vrf + return command + + +def _tmplt_ospf_address_family_cmd(config_data): + afi = "{afi}".format(**config_data) + if afi == "router": + command = "" + else: + command = "address-family " + afi + return command + + +def _tmplt_ospf_adjacency_cmd(config_data): + command = "adjacency exchange-start threshold" + if "adjacency" in config_data: + command += " {threshold}".format( + **config_data["adjacency"]["exchange_start"] + ) + return command + + +def _tmplt_ospf_auto_cost(config_data): + if "auto_cost" in config_data: + command = "auto-cost" + if "reference_bandwidth" in config_data["auto_cost"]: + command += " reference-bandwidth {reference_bandwidth}".format( + **config_data["auto_cost"] + ) + return command + + +def _tmplt_ospf_area_authentication(config_data): + if "area_id" in config_data: + command = "area {area_id} authentication ipsec spi ".format( + **config_data + ) + command += "{spi} {algorithm}".format(**config_data["authentication"]) + if "passphrase" in config_data["authentication"]: + command += " passphrase" + if ( + "encrypt_key" in config_data["authentication"] + and config_data["authentication"]["encrypt_key"] is False + ): + command += " 0" + if ( + "hidden_key" in config_data["authentication"] + and config_data["authentication"]["hidden_key"] is True + ): + command += " 7" + if "passphrase" not in config_data["authentication"]: + command += " {key}".format(**config_data["authentication"]) + else: + command += " {passphrase}".format(**config_data["authentication"]) + return command + + +def _tmplt_ospf_area_encryption(config_data): + if "area_id" in config_data: + command = "area {area_id} encryption ipsec spi ".format(**config_data) + command += "{spi} esp {encryption} {algorithm}".format( + **config_data["encryption"] + ) + if "passphrase" in config_data["encryption"]: + command += " passphrase" + if ( + "encrypt_key" in config_data["encryption"] + and config_data["encryption"]["encrypt_key"] is False + ): + command += " 0" + if ( + "hidden_key" in config_data["encryption"] + and config_data["encryption"]["hidden_key"] is True + ): + command += " 7" + if "passphrase" not in config_data["encryption"]: + command += " {key}".format(**config_data["encryption"]) + else: + command += " {passphrase}".format(**config_data["encryption"]) + return command + + +def _tmplt_ospf_area_nssa(config_data): + if "nssa" in config_data: + command = "area {area_id} nssa".format(**config_data) + if "default_information_originate" in config_data["nssa"]: + command += " default-information-originate" + if ( + "metric" + in config_data["nssa"]["default_information_originate"] + ): + command += " metric {metric}".format( + **config_data["nssa"]["default_information_originate"] + ) + if ( + "metric_type" + in config_data["nssa"]["default_information_originate"] + ): + command += " metric-type {metric_type}".format( + **config_data["nssa"]["default_information_originate"] + ) + if ( + "nssa_only" + in config_data["nssa"]["default_information_originate"] + ): + command += " nssa-only" + if config_data["nssa"].get("nssa_only"): + command += " nssa-only" + if config_data["nssa"].get("translate"): + command += " translate type7 always" + if config_data["nssa"].get("no_summary"): + command += " no-summary" + return command + + +def _tmplt_ospf_area_range(config_data): + if "area_id" in config_data: + command = "area {area_id} range".format(**config_data) + if "address" in config_data: + command += " {address}".format(**config_data) + if "subnet_address" in config_data: + command += " {subnet_address}".format(**config_data) + if "subnet_mask" in config_data: + command += " {subnet_mask}".format(**config_data) + if "advertise" in config_data: + if config_data.get("advertise"): + command += " advertise" + else: + command += " not-advertise" + if "cost" in config_data: + command += " cost {cost}".format(**config_data) + return command + + +def _tmplt_ospf_area_stub(config_data): + if "stub" in config_data: + command = "area {area_id} stub".format(**config_data) + if "summary_lsa" in config_data["stub"]: + if not config_data["stub"]["summary_lsa"]: + command += " no-summary" + return command + + +def _tmplt_ospf_default_information(config_data): + if "default_information" in config_data: + command = "default-information" + if "originate" in config_data["default_information"]: + command += " originate" + if "always" in config_data["default_information"]: + command += " always" + if "metric" in config_data["default_information"]: + command += " metric {metric}".format( + **config_data["default_information"] + ) + if "metric_type" in config_data["default_information"]: + command += " metric-type {metric_type}".format( + **config_data["default_information"] + ) + if "route_map" in config_data["default_information"]: + command += " route-map {route_map}".format( + **config_data["default_information"] + ) + return command + + +def _tmplt_ospf_log_adjacency_changes(config_data): + if "log_adjacency_changes" in config_data: + command = "log-adjacency-changes" + if "detail" in config_data["log_adjacency_changes"]: + if config_data["log_adjacency_changes"].get("detail"): + command += " detail" + return command + + +def _tmplt_ospf_max_metric(config_data): + if "max_metric" in config_data: + command = "max-metric" + if "router_lsa" in config_data["max_metric"]: + command += " router-lsa" + if "external_lsa" in config_data["max_metric"]["router_lsa"]: + command += " external-lsa" + if ( + "max_metric_value" + in config_data["max_metric"]["router_lsa"]["external_lsa"] + ): + command += " {max_metric_value}".format( + **config_data["max_metric"]["router_lsa"]["external_lsa"] + ) + if "include_stub" in config_data["max_metric"]["router_lsa"]: + if config_data["max_metric"]["router_lsa"].get("include_stub"): + command += " include-stub" + if "on_startup" in config_data["max_metric"]["router_lsa"]: + command += " on-startup {wait_period}".format( + **config_data["max_metric"]["router_lsa"]["on_startup"] + ) + if "summary_lsa" in config_data["max_metric"]["router_lsa"]: + command += " summary-lsa" + if ( + "max_metric_value" + in config_data["max_metric"]["router_lsa"]["summary_lsa"] + ): + command += " {max_metric_value}".format( + **config_data["max_metric"]["router_lsa"]["summary_lsa"] + ) + return command + + +def _tmplt_ospf_redistribute(config_data): + command = "redistribute {routes}".format(**config_data) + if "route_map" in config_data: + command += " route-map {route_map}".format(**config_data) + return command + + +def _tmplt_ospf_timers_throttle(config_data): + if "throttle" in config_data["timers"]: + command = "timers throttle" + if "lsa" in config_data["timers"]["throtle"]: + if config_data["timers"]["throtle"].get("lsa"): + command += " lsa all" + if "spf" in config_data["timers"]["throtle"]: + if config_data["timers"]["throtle"].get("spf"): + command += " spf" + if "initial" in config_data["timers"]["throttle"]: + command += " {initial}".format(**config_data["timers"]["throttle"]) + if "min" in config_data["timers"]["throttle"]: + command += " {min}".format(**config_data["timers"]["throttle"]) + if "max" in config_data["timers"]["throttle"]: + command += " {max}".format(**config_data["timers"]["max"]) + + return command + + +class Ospfv3Template(NetworkTemplate): + def __init__(self, lines=None): + super(Ospfv3Template, self).__init__(lines=lines, tmplt=self) + + PARSERS = [ + { + "name": "vrf", + "getval": re.compile( + r""" + ^router\s + ospfv3 + \svrf + \s(?P<vrf>\S+) + $""", + re.VERBOSE, + ), + "setval": _tmplt_ospf_vrf_cmd, + "result": {"processes": {"vrf": "{{ vrf }}"}}, + "shared": True, + }, + { + "name": "vrf", + "getval": re.compile( + r""" + ^router\s + ospfv3 + $""", + re.VERBOSE, + ), + "setval": _tmplt_ospf_vrf_cmd, + "result": {"processes": {"vrf": '{{ "default" }}'}}, + "shared": True, + }, + { + "name": "address_family", + "getval": re.compile( + r""" + \s*address-family + \s(?P<afi>\S+)* + $""", + re.VERBOSE, + ), + "setval": _tmplt_ospf_address_family_cmd, + "compval": "address_family", + "result": { + "processes": { + "address_family": {"{{ afi }}": {"afi": "{{ afi }}"}} + } + }, + "shared": True, + }, + { + "name": "adjacency", + "getval": re.compile( + r""" + \s*adjacency + \s+exchange-start + \s+threshold + \s+(?P<threshold>\d+)* + $""", + re.VERBOSE, + ), + "setval": _tmplt_ospf_adjacency_cmd, + "result": { + "processes": { + "address_family": { + '{{ afi|default("router", true) }}': { + "adjacency": { + "exchange_start": { + "threshold": "{{ threshold|int }}" + } + } + } + } + } + }, + }, + { + "name": "auto_cost", + "getval": re.compile( + r"""\s+(?P<auto_cost>auto-cost)* + \s*(?P<ref_band>reference-bandwidth\s\d+) + *$""", + re.VERBOSE, + ), + "setval": _tmplt_ospf_auto_cost, + "result": { + "processes": { + "address_family": { + '{{ afi|default("router", true) }}': { + "auto_cost": { + "reference_bandwidth": '{{ ref_band.split(" ")[1] }}' + } + } + } + } + }, + }, + { + "name": "area.default_cost", + "getval": re.compile( + r"""\s+area + \s(?P<area_id>\S+)* + \sdefault-cost* + \s*(?P<default_cost>\S+) + *$""", + re.VERBOSE, + ), + "setval": "area {{ area_id }} default-cost {{ default_cost }}", + "compval": "default_cost", + "result": { + "processes": { + "address_family": { + '{{ afi|default("router", true) }}': { + "areas": { + "{{ area_id }}": { + "area_id": "{{ area_id }}", + "default_cost": "{{ default_cost|int }}", + } + } + } + } + } + }, + }, + { + "name": "area.authentication", + "getval": re.compile( + r""" + \s*area + \s+(?P<area_id>\S+) + \s+authentication + \s+ipsec + \s+spi + \s+(?P<val>\d+) + \s+(?P<algorithm>md5|sha1) + \s*(?P<passphrase>passphrase)* + \s*(?P<type>0|7)* + \s*(?P<line>\S+) + *$""", + re.VERBOSE, + ), + "setval": _tmplt_ospf_area_authentication, + "compval": "authentication", + "remval": "area {{ area_id }} authentication", + "result": { + "processes": { + "address_family": { + '{{ afi|default("router", true) }}': { + "areas": { + "{{ area_id }}": { + "area_id": "{{ area_id }}", + "authentication": { + "spi": "{{ val }}", + "algorithm": "{{ algorithm }}", + "encrypt_key": '{{ False if type is defined and type == "0" }}', + "hidden_key": '{{ True if type is defined and type == "7" }}', + "passphrase": "{{ line if passphrase is defined }}", + "key": "{{ str(line) if passphrase is undefined }}", + }, + } + } + } + } + } + }, + }, + { + "name": "area.encryption", + "getval": re.compile( + r""" + \s*area + \s+(?P<area_id>\S+)* + \s+encryption + \s+ipsec + \s+spi + \s+(?P<val>\d+) + \s+esp + \s*(?P<encryption>\S+) + \s*(?P<algorithm>md5|sha1)* + \s*(?P<passphrase>passphrase)* + \s*(?P<type>0|7)* + \s*(?P<line>\S+) + *$""", + re.VERBOSE, + ), + "setval": _tmplt_ospf_area_encryption, + "compval": "encryption", + "remval": "area {{ area_id }} encryption", + "result": { + "processes": { + "address_family": { + '{{ afi|default("router", true) }}': { + "areas": { + "{{ area_id }}": { + "area_id": "{{ area_id }}", + "encryption": { + "spi": "{{ val }}", + "encryption": "{{ encryption }}", + "algorithm": "{{ algorithm }}", + "encrypt_key": "{{ False if type is defined and type == '0'}}", + "passphrase": "{{line if passphrase is defined }}", + "hidden_key": "{{ True if type is defined and type == '7'}}", + "key": "{{ line if passphrase is not defined }}", + }, + } + } + } + } + } + }, + }, + { + "name": "area.nssa", + "getval": re.compile( + r""" + \s+area\s(?P<area_id>\S+) + \s(?P<nssa>nssa) + \s*(?P<def_origin>default-information-originate)* + \s*(metric)* + \s*(?P<metric>\d+)* + \s*(metric-type)* + \s*(?P<metric_type>\d+)* + \s*(?P<no_summary>no-summary)* + \s*(?P<translate>translate.*)* + $""", + re.VERBOSE, + ), + "setval": _tmplt_ospf_area_nssa, + "compval": "nssa", + "result": { + "processes": { + "address_family": { + '{{ afi|default("router", true) }}': { + "areas": { + "{{ area_id }}": { + "area_id": "{{ area_id }}", + "nssa": { + "set": "{{ True if nssa is defined and def_origin is undefined and " + "no_summary is undefined and translate is undefined }}", + "default_information_originate": { + "set": "{{ True if def_origin is defined and metric is undefined and " + "metric_type is undefined and nssa_only is undefined }}", + "metric": "{{ metric.split(" + ")[1]|int }}", + "metric_type": "{{ metric_type.split(" + ")[1]|int }}", + "nssa_only": "{{ True if nssa_only is defined }}", + }, + "translate": "{{ True if translate is defined }}", + "no_summary": "{{ True if no_summary is defined }}", + }, + } + } + } + } + } + }, + }, + { + "name": "area.ranges", + "getval": re.compile( + r""" + \s*area + \s+(?P<area_id>\S+)* + \s+range + \s+(?P<address>\S+)* + \s*(?P<not_advertise>not-advertise)* + \s*(?P<subnet_mask>\d{1,3}\.\d{1,3}\.\d{1,3}\.\d{1,3})* + \s*(?P<subnet_address>\S+)* + \s*(?P<cost>cost)* + \s*(?P<cost_val>\d+) + *$""", + re.VERBOSE, + ), + "setval": _tmplt_ospf_area_range, + "compval": "ranges", + "result": { + "processes": { + "address_family": { + '{{ afi|default("router", true) }}': { + "areas": { + "{{ area_id }}": { + "area_id": "{{ area_id }}", + "ranges": [ + { + "address": "{{ address }}", + "subnet_mask": "{{ subnet_mask }}", + "advertise": "{{ not not_advertise }}", + "cost": "{{ cost_val }}", + } + ], + } + } + } + } + } + }, + }, + { + "name": "area.stub", + "getval": re.compile( + r"""\s+area\s(?P<area_id>\S+) + \s(?P<stub>stub)* + \s*(?P<no_sum>no-summary) + *$""", + re.VERBOSE, + ), + "setval": _tmplt_ospf_area_stub, + "compval": "stub", + "result": { + "processes": { + "address_family": { + '{{ afi|default("router", true) }}': { + "areas": { + "{{ area_id }}": { + "area_id": "{{ area_id }}", + "stub": { + "set": "{{ True if stub is defined and no_sum is undefined }}", + "summary_lsa": "{{ True if no_sum is defined }}", + }, + } + } + } + } + } + }, + }, + { + "name": "bfd", + "getval": re.compile( + r"""\s+bfd* + \s*(?P<bfd>all-interfaces) + *$""", + re.VERBOSE, + ), + "setval": "bfd all-interfaces", + "result": { + "processes": { + "address_family": { + '{{ afi|default("router", true) }}': { + "bfd": { + "all_interfaces": "{{ True if bfd is defined }}" + } + } + } + } + }, + }, + { + "name": "default_information", + "getval": re.compile( + r"""\s+default-information* + \s*(?P<originate>originate)* + \s*(?P<always>always)* + \s*(?P<metric>metric\s\d+)* + \s*(?P<metric_type>metric-type\s\d+)* + \s*(?P<route_map>route-map\s\S+) + *$""", + re.VERBOSE, + ), + "setval": _tmplt_ospf_default_information, + "result": { + "processes": { + "address_family": { + '{{ afi|default("router", true) }}': { + "default_information": { + "originate": "{{ True if originate is defined }}", + "always": "{{ True if always is defined }}", + "metric": "{{ metric.split(" ")[1]|int }}", + "metric_type": "{{ metric_type.split(" + ")[1]|int }}", + "route_map": "{{ route_map.split(" ")[1] }}", + } + } + } + } + }, + }, + { + "name": "default_metric", + "getval": re.compile( + r"""\s+default-metric(?P<default_metric>\s\d+) + *$""", + re.VERBOSE, + ), + "setval": "default-metric {{ default_metric }}", + "result": { + "processes": { + "address_family": { + '{{ afi|default("router", true) }}': { + "default_metric": "{{ default_metric| int}}" + } + } + } + }, + }, + { + "name": "distance", + "getval": re.compile( + r"""\s+distance + \s+ospf + \s+intra-area + \s+(?P<distance>\d+) + *$""", + re.VERBOSE, + ), + "setval": "distance ospf intra-area {{ distance }}", + "result": { + "processes": { + "address_family": { + '{{ afi|default("router", true) }}': { + "distance": "{{ distance| int}}" + } + } + } + }, + }, + { + "name": "fips_restrictions", + "getval": re.compile( + r""" + \s+(?P<fips>fips\s*\S+) + *$""", + re.VERBOSE, + ), + "setval": "fips restrictions", + "result": { + "processes": { + "address_family": { + '{{ afi|default("router", true) }}': { + "fips_restrictions": "{{ True if fips is defined }}" + } + } + } + }, + }, + { + "name": "graceful_restart_period", + "getval": re.compile( + r""" + \s+graceful-restart + \s*grace-period* + \s*(?P<period>\d+)* + $""", + re.VERBOSE, + ), + "setval": "graceful-restart grace-period {{ graceful_restart.grace_period }}", + "remval": "graceful-restart", + "result": { + "processes": { + "address_family": { + '{{ afi|default("router", true) }}': { + "graceful_restart": { + "grace_period": "{{ period|int }}" + } + } + } + } + }, + }, + { + "name": "graceful_restart", + "getval": re.compile( + r""" + \s+graceful-restart + $""", + re.VERBOSE, + ), + "setval": "graceful-restart", + "remval": "graceful-restart", + "result": { + "processes": { + "address_family": { + '{{ afi|default("router", true) }}': { + "graceful_restart": {"set": "{{ True }}"} + } + } + } + }, + }, + { + "name": "graceful_restart_helper", + "getval": re.compile( + r"""\s+(?P<grace>graceful-restart-helper) + *$""", + re.VERBOSE, + ), + "setval": "{{ grace }}", + "result": { + "processes": { + "address_family": { + '{{ afi|default("router", true) }}': { + "graceful_restart_helper": { + "set": "{{ True if grace is defined }}" + } + } + } + } + }, + }, + { + "name": "log_adjacency_changes", + "getval": re.compile( + r"""\s+(?P<log>log-adjacency-changes)* + \s*(?P<detail>detail) + *$""", + re.VERBOSE, + ), + "setval": _tmplt_ospf_log_adjacency_changes, + "result": { + "processes": { + "address_family": { + '{{ afi|default("router", true) }}': { + "log_adjacency_changes": { + "set": "{{ True if log is defined and detail is undefined }}", + "detail": "{{ True if detail is defined }}", + } + } + } + } + }, + }, + { + "name": "max_metric", + "getval": re.compile( + r"""\s+max-metric + \s+(?P<router_lsa>router-lsa) + \s*(?P<external_lsa>external-lsa)* + \s*(?P<external_lsa_metric>\d+)* + \s*(?P<on_startup>on-startup)* + \s*(?P<wait_for_bgp>wait-for-bgp)* + \s*(?P<startup_time>\d+)* + \s*(?P<summary_lsa>summary-lsa)* + \s*(?P<summary_lsa_metric>\d+)* + \s*(?P<include_stub>include-stub) + *$""", + re.VERBOSE, + ), + "setval": _tmplt_ospf_max_metric, + "remval": "max-metric router-lsa", + "result": { + "processes": { + "address_family": { + '{{ afi|default("router", true) }}': { + "max_metric": { + "router_lsa": { + "set": "{{ True if router_lsa is defined and external_lsa is undefined and " + "include_stub is undefined and on_startup is undefined and " + "summary_lsa is undefined }}", + "external_lsa": { + "set": "{{ True if external_lsa is defined and external_lsa_metric is undefined }}", + "max_metric_value": "{{ external_lsa_metric }}", + }, + "include_stub": "{{ True if include_stub is defined }}", + "on_startup": { + "wait_period": "{{ startup_time }}", + "wait_for_bgp": "{{ True if wait_for_bgp is defined }}", + }, + "summary_lsa": { + "set": "{{ True if summary_lsa is defined and summary_lsa_metric is undefined }}", + "max_metric_value": "{{ summary_lsa_metric }}", + }, + } + } + } + } + } + }, + }, + { + "name": "maximum_paths", + "getval": re.compile( + r"""\s+maximum-paths* + \s+(?P<paths>\d+) + *$""", + re.VERBOSE, + ), + "setval": "maximum-paths {{ maximum_paths }}", + "result": { + "processes": { + "address_family": { + '{{ afi|default("router", true) }}': { + "maximum_paths": "{{ paths }}" + } + } + } + }, + }, + { + "name": "passive_interface", + "getval": re.compile( + r""" + \s*(?P<passive>passive-interface.*) + *$""", + re.VERBOSE, + ), + "setval": "passive-interface default", + "result": { + "processes": { + "address_family": { + '{{ afi|default("router", true) }}': { + "passive_interface": "{{ True if passive is defined }}" + } + } + } + }, + }, + { + "name": "redistribute", + "getval": re.compile( + r""" + \s+redistribute + \s(?P<route>\S+) + \s*(?P<rmpa>route-map)* + \s*(?P<map>\S+)* + $""", + re.VERBOSE, + ), + "setval": _tmplt_ospf_redistribute, + "result": { + "processes": { + "address_family": { + '{{ afi|default("router", true) }}': { + "redistribute": [ + { + "routes": "{{ route }}", + "route_map": "{{ map }}", + } + ] + } + } + } + }, + }, + { + "name": "router_id", + "getval": re.compile( + r""" + \s+router-id + \s(?P<id>\S+)$""", + re.VERBOSE, + ), + "setval": ("router-id" " {{ router_id }}"), + "remval": "router-id", + "result": { + "processes": { + "address_family": { + '{{ afi|default("router", true) }}': { + "router_id": "{{ id }}" + } + } + } + }, + }, + { + "name": "shutdown", + "getval": re.compile( + r"""\s+(?P<shutdown>shutdown) + *$""", + re.VERBOSE, + ), + "setval": "shutdown", + "result": { + "processes": { + "address_family": { + '{{ afi|default("router", true) }}': { + "shutdown": "{{ True if shutdown is defined }}" + } + } + } + }, + }, + { + "name": "timers.lsa", + "getval": re.compile( + r"""\s+timers + \slsa + \sarrival + \s(?P<lsa>\d+) + *$""", + re.VERBOSE, + ), + "setval": "timers lsa arrival {{ timers.lsa }}", + "result": { + "processes": { + "address_family": { + '{{ afi|default("router", true) }}': { + "timers": {"lsa": "{{ lsa }}"} + } + } + } + }, + }, + { + "name": "timers.out_delay", + "getval": re.compile( + r"""\s+timers + \sout-delay + \s(?P<out_delay>\d+) + *$""", + re.VERBOSE, + ), + "setval": "timers out-delay {{ timers.out_delay }}", + "result": { + "processes": { + "address_family": { + '{{ afi|default("router", true) }}': { + "timers": {"out_delay": "{{ out_delay }}"} + } + } + } + }, + }, + { + "name": "timers.pacing", + "getval": re.compile( + r"""\s+timers + \spacing + \sflood + \s(?P<pacing>\d+) + *$""", + re.VERBOSE, + ), + "setval": "timers pacing flood {{ timers.pacing }}", + "result": { + "processes": { + "address_family": { + '{{ afi|default("router", true) }}': { + "timers": {"pacing": "{{ pacing }}"} + } + } + } + }, + }, + { + "name": "timers.throttle.lsa", + "getval": re.compile( + r"""\s+timers + \sthrottle + \s*(?P<lsa>lsa all)* + \s*(?P<initial>\d+)* + \s*(?P<min>\d+)* + \s*(?P<max>\d+) + *$""", + re.VERBOSE, + ), + "setval": "timers throttle lsa all {{ timers.throttle.initial }} {{ timers.throttle.min }} {{ timers.throttle.max }}", + "result": { + "processes": { + "address_family": { + '{{ afi|default("router", true) }}': { + "timers": { + "throttle": { + "lsa": "{{ True if lsa is defined }}", + "initial": "{{ initial }}", + "min": "{{ min_delay }}", + "max": "{{ max_delay }}", + } + } + } + } + } + }, + }, + { + "name": "timers.throttle.spf", + "getval": re.compile( + r"""\s+timers + \sthrottle + \s*(?P<spf>spf) + \s*(?P<initial>\d+) + \s*(?P<min>\d+) + \s*(?P<max>\d+) + *$""", + re.VERBOSE, + ), + "setval": "timers throttle spf {{ timers.throttle.initial }} {{ timers.throttle.min }} {{ timers.throttle.max }}", + "result": { + "processes": { + "address_family": { + '{{ afi|default("router", true) }}': { + "timers": { + "throttle": { + "spf": "{{ True if spf is defined }}", + "initial": "{{ initial }}", + "min": "{{ min }}", + "max": "{{ max }}", + } + } + } + } + } + }, + }, + ] diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/utils/__init__.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/utils/__init__.py new file mode 100644 index 00000000..e69de29b --- /dev/null +++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/utils/__init__.py diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/utils/utils.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/utils/utils.py new file mode 100644 index 00000000..f28f17c3 --- /dev/null +++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/module_utils/network/eos/utils/utils.py @@ -0,0 +1,85 @@ +# -*- coding: utf-8 -*- +# Copyright 2019 Red Hat +# GNU General Public License v3.0+ +# (see COPYING or https://www.gnu.org/licenses/gpl-3.0.txt) + +# utils + +from __future__ import absolute_import, division, print_function + +__metaclass__ = type + + +def get_interface_number(name): + digits = "" + for char in name: + if char.isdigit() or char in "/.": + digits += char + return digits + + +def normalize_interface(name): + """Return the normalized interface name + """ + if not name: + return None + + if name.lower().startswith("et"): + if_type = "Ethernet" + elif name.lower().startswith("lo"): + if_type = "Loopback" + elif name.lower().startswith("ma"): + if_type = "Management" + elif name.lower().startswith("po"): + if_type = "Port-Channel" + elif name.lower().startswith("tu"): + if_type = "Tunnel" + elif name.lower().startswith("vl"): + if_type = "Vlan" + elif name.lower().startswith("vx"): + if_type = "Vxlan" + else: + if_type = None + + number_list = name.split(" ") + if len(number_list) == 2: + number = number_list[-1].strip() + else: + number = get_interface_number(name) + + if if_type: + proper_interface = if_type + number + else: + proper_interface = name + + return proper_interface + + +def vlan_range_to_list(vlans): + result = [] + if vlans: + if isinstance(vlans, str): + vlans = vlans.split(",") + for part in vlans: + if part == "none": + break + if "-" in part: + a, b = part.split("-") + a, b = int(a), int(b) + result.extend(range(a, b + 1)) + else: + a = int(part) + result.append(a) + return numerical_sort(result) + return result + + +def numerical_sort(string_int_list): + """Sorts list of integers that are digits in numerical order. + """ + as_int_list = [] + + for vlan in string_int_list: + as_int_list.append(int(vlan)) + as_int_list.sort() + return list(set(as_int_list)) diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/modules/__init__.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/modules/__init__.py new file mode 100644 index 00000000..e69de29b --- /dev/null +++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/modules/__init__.py diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/modules/eos_acl_interfaces.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/modules/eos_acl_interfaces.py new file mode 100644 index 00000000..fbec02f1 --- /dev/null +++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/modules/eos_acl_interfaces.py @@ -0,0 +1,421 @@ +#!/usr/bin/python +# -*- coding: utf-8 -*- +# Copyright 2019 Red Hat +# GNU General Public License v3.0+ +# (see COPYING or https://www.gnu.org/licenses/gpl-3.0.txt) + +############################################# +# WARNING # +############################################# +# +# This file is auto generated by the resource +# module builder playbook. +# +# Do not edit this file manually. +# +# Changes to this file will be over written +# by the resource module builder. +# +# Changes should be made in the model used to +# generate this file or in the resource module +# builder template. +# +############################################# + +""" +The module file for eos_acl_interfaces +""" + +from __future__ import absolute_import, division, print_function + +__metaclass__ = type + + +DOCUMENTATION = """ +module: eos_acl_interfaces +short_description: ACL interfaces resource module +description: +- This module manages adding and removing Access Control Lists (ACLs) from interfaces + on devices running EOS software. +version_added: 1.0.0 +author: GomathiSelvi S (@GomathiselviS) +options: + config: + description: A dictionary of ACL options for interfaces. + type: list + elements: dict + suboptions: + name: + description: + - Name/Identifier for the interface. + type: str + required: true + access_groups: + type: list + elements: dict + description: + - Specifies ACLs attached to the interfaces. + suboptions: + afi: + description: + - Specifies the AFI for the ACL(s) to be configured on this interface. + type: str + choices: + - ipv4 + - ipv6 + required: true + acls: + type: list + description: + - Specifies the ACLs for the provided AFI. + elements: dict + suboptions: + name: + description: + - Specifies the name of the IPv4/IPv4 ACL for the interface. + type: str + required: true + direction: + description: + - Specifies the direction of packets that the ACL will be applied + on. + type: str + choices: + - in + - out + required: true + running_config: + description: + - The module, by default, will connect to the remote device and retrieve the current + running-config to use as a base for comparing against the contents of source. + There are times when it is not desirable to have the task get the current running-config + for every task in a playbook. The I(running_config) argument allows the implementer + to pass in the configuration to use as the base config for comparison. This + value of this option should be the output received from device by executing + command + type: str + state: + description: + - The state the configuration should be left in. + type: str + choices: + - merged + - replaced + - overridden + - deleted + - gathered + - parsed + - rendered + default: merged + +""" +EXAMPLES = """ +# Using Merged + +# Before state: +# ------------- +# +# eos#sh running-config | include interface|access-group +# interface Ethernet1 +# interface Ethernet2 +# interface Ethernet3 + +- name: Merge module attributes of given access-groups + arista.eos.eos_acl_interfaces: + config: + - name: Ethernet2 + access_groups: + - afi: ipv4 + acls: + name: acl01 + direction: in + - afi: ipv6 + acls: + name: acl03 + direction: out + state: merged + +# Commands Fired: +# --------------- +# +# interface Ethernet2 +# ip access-group acl01 in +# ipv6 access-group acl03 out + +# After state: +# ------------- +# +# eos#sh running-config | include interface| access-group +# interface Loopback888 +# interface Ethernet1 +# interface Ethernet2 +# ip access-group acl01 in +# ipv6 access-group acl03 out +# interface Ethernet3 + + +# Using Replaced + +# Before state: +# ------------- +# +# eos#sh running-config | include interface|access-group +# interface Ethernet1 +# interface Ethernet2 +# ip access-group acl01 in +# ipv6 access-group acl03 out +# interface Ethernet3 +# ip access-group acl01 in + +- name: Replace module attributes of given access-groups + arista.eos.eos_acl_interfaces: + config: + - name: Ethernet2 + access_groups: + - afi: ipv4 + acls: + name: acl01 + direction: out + state: replaced + +# Commands Fired: +# --------------- +# +# interface Ethernet2 +# no ip access-group acl01 in +# no ipv6 access-group acl03 out +# ip access-group acl01 out + +# After state: +# ------------- +# +# eos#sh running-config | include interface| access-group +# interface Loopback888 +# interface Ethernet1 +# interface Ethernet2 +# ip access-group acl01 out +# interface Ethernet3 +# ip access-group acl01 in + + +# Using Overridden + +# Before state: +# ------------- +# +# eos#sh running-config | include interface|access-group +# interface Ethernet1 +# interface Ethernet2 +# ip access-group acl01 in +# ipv6 access-group acl03 out +# interface Ethernet3 +# ip access-group acl01 in + +- name: Override module attributes of given access-groups + arista.eos.eos_acl_interfaces: + config: + - name: Ethernet2 + access_groups: + - afi: ipv4 + acls: + name: acl01 + direction: out + state: overridden + +# Commands Fired: +# --------------- +# +# interface Ethernet2 +# no ip access-group acl01 in +# no ipv6 access-group acl03 out +# ip access-group acl01 out +# interface Ethernet3 +# no ip access-group acl01 in + +# After state: +# ------------- +# +# eos#sh running-config | include interface| access-group +# interface Loopback888 +# interface Ethernet1 +# interface Ethernet2 +# ip access-group acl01 out +# interface Ethernet3 + + +# Using Deleted + +# Before state: +# ------------- +# +# eos#sh running-config | include interface|access-group +# interface Ethernet1 +# interface Ethernet2 +# ip access-group acl01 in +# ipv6 access-group acl03 out +# interface Ethernet3 +# ip access-group acl01 out + +- name: Delete module attributes of given access-groups + arista.eos.eos_acl_interfaces: + config: + - name: Ethernet2 + access_groups: + - afi: ipv4 + acls: + name: acl01 + direction: in + - afi: ipv6 + acls: + name: acl03 + direction: out + state: deleted + +# Commands Fired: +# --------------- +# +# interface Ethernet2 +# no ip access-group acl01 in +# no ipv6 access-group acl03 out + +# After state: +# ------------- +# +# eos#sh running-config | include interface| access-group +# interface Loopback888 +# interface Ethernet1 +# interface Ethernet2 +# interface Ethernet3 +# ip access-group acl01 out + + +# Before state: +# ------------- +# +# eos#sh running-config | include interface| access-group +# interface Ethernet1 +# interface Ethernet2 +# ip access-group acl01 in +# ipv6 access-group acl03 out +# interface Ethernet3 +# ip access-group acl01 out + +- name: Delete module attributes of given access-groups from ALL Interfaces + arista.eos.eos_acl_interfaces: + config: + state: deleted + +# Commands Fired: +# --------------- +# +# interface Ethernet2 +# no ip access-group acl01 in +# no ipv6 access-group acl03 out +# interface Ethernet3 +# no ip access-group acl01 out + +# After state: +# ------------- +# +# eos#sh running-config | include interface| access-group +# interface Loopback888 +# interface Ethernet1 +# interface Ethernet2 +# interface Ethernet3 + +# Before state: +# ------------- +# +# eos#sh running-config | include interface| access-group +# interface Ethernet1 +# interface Ethernet2 +# ip access-group acl01 in +# ipv6 access-group acl03 out +# interface Ethernet3 +# ip access-group acl01 out + +- name: Delete acls under afi + arista.eos.eos_acl_interfaces: + config: + - name: Ethernet3 + access_groups: + - afi: ipv4 + - name: Ethernet2 + access_groups: + - afi: ipv6 + state: deleted + +# Commands Fired: +# --------------- +# +# interface Ethernet2 +# no ipv6 access-group acl03 out +# interface Ethernet3 +# no ip access-group acl01 out + +# After state: +# ------------- +# +# eos#sh running-config | include interface| access-group +# interface Loopback888 +# interface Ethernet1 +# interface Ethernet2 +# ip access-group acl01 in +# interface Ethernet3 + + +""" +RETURN = """ +before: + description: The configuration prior to the model invocation. + returned: always + type: list + sample: > + The configuration returned will always be in the same format + of the parameters above. +after: + description: The resulting configuration model invocation. + returned: when changed + type: list + sample: > + The configuration returned will always be in the same format + of the parameters above. +commands: + description: The set of commands pushed to the remote device. + returned: always + type: list + sample: + - interface Ethernet2 + - ip access-group acl01 in + - ipv6 access-group acl03 out + - interface Ethernet3 + - ip access-group acl01 out +""" + + +from ansible.module_utils.basic import AnsibleModule +from ansible_collections.arista.eos.plugins.module_utils.network.eos.argspec.acl_interfaces.acl_interfaces import ( + Acl_interfacesArgs, +) +from ansible_collections.arista.eos.plugins.module_utils.network.eos.config.acl_interfaces.acl_interfaces import ( + Acl_interfaces, +) + + +def main(): + """ + Main entry point for module execution + + :returns: the result form module invocation + """ + module = AnsibleModule( + argument_spec=Acl_interfacesArgs.argument_spec, + supports_check_mode=True, + ) + + result = Acl_interfaces(module).execute_module() + module.exit_json(**result) + + +if __name__ == "__main__": + main() diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/modules/eos_acls.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/modules/eos_acls.py new file mode 100644 index 00000000..c37c4244 --- /dev/null +++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/modules/eos_acls.py @@ -0,0 +1,931 @@ +#!/usr/bin/python +# -*- coding: utf-8 -*- +# Copyright 2019 Red Hat +# GNU General Public License v3.0+ +# (see COPYING or https://www.gnu.org/licenses/gpl-3.0.txt) + +############################################# +# WARNING # +############################################# +# +# This file is auto generated by the resource +# module builder playbook. +# +# Do not edit this file manually. +# +# Changes to this file will be over written +# by the resource module builder. +# +# Changes should be made in the model used to +# generate this file or in the resource module +# builder template. +# +############################################# + +""" +The module file for eos_acls +""" + +from __future__ import absolute_import, division, print_function + +__metaclass__ = type + + +DOCUMENTATION = """ +module: eos_acls +short_description: ACLs resource module +description: This module manages the IP access-list attributes of Arista EOS interfaces. +version_added: 1.0.0 +author: Gomathiselvi S (@GomathiselviS) +notes: +- Tested against Arista vEOS v4.20.10M +options: + config: + description: A dictionary of IP access-list options + type: list + elements: dict + suboptions: + afi: + description: + - The Address Family Indicator (AFI) for the Access Control Lists (ACL). + type: str + required: true + choices: + - ipv4 + - ipv6 + acls: + description: + - A list of Access Control Lists (ACL). + type: list + elements: dict + suboptions: + standard: + description: standard access-list or not + type: bool + name: + description: Name of the acl-list + type: str + required: true + aces: + description: Filtering data + type: list + elements: dict + suboptions: + sequence: + description: sequence number for the ordered list of rules + type: int + remark: + description: Specify a comment + type: str + fragment_rules: + description: Add fragment rules + type: bool + grant: + description: Action to be applied on the rule + type: str + choices: + - permit + - deny + line: + description: For fact gathering, any ACE that is not fully parsed, + while show up as a value of this attribute. + type: str + aliases: + - ace + protocol: + description: + - Specify the protocol to match. + - Refer to vendor documentation for valid values. + type: str + vlan: + description: Vlan options + type: str + protocol_options: + description: All the possible sub options for the protocol chosen. + type: dict + suboptions: + tcp: + description: Options for tcp protocol. + type: dict + suboptions: + flags: + description: Match TCP packet flags + type: dict + suboptions: + ack: + description: Match on the ACK bit + type: bool + established: + description: Match established connections + type: bool + fin: + description: Match on the FIN bit + type: bool + psh: + description: Match on the PSH bit + type: bool + rst: + description: Match on the RST bit + type: bool + syn: + description: Match on the SYN bit + type: bool + urg: + description: Match on the URG bit + type: bool + icmp: + description: + - Internet Control Message Protocol settings. + type: dict + suboptions: + administratively_prohibited: + description: Administratively prohibited + type: bool + alternate_address: + description: Alternate address + type: bool + conversion_error: + description: Datagram conversion + type: bool + dod_host_prohibited: + description: Host prohibited + type: bool + dod_net_prohibited: + description: Net prohibited + type: bool + echo: + description: Echo (ping) + type: bool + echo_reply: + description: Echo reply + type: bool + general_parameter_problem: + description: Parameter problem + type: bool + host_isolated: + description: Host isolated + type: bool + host_precedence_unreachable: + description: Host unreachable for precedence + type: bool + host_redirect: + description: Host redirect + type: bool + host_tos_redirect: + description: Host redirect for TOS + type: bool + host_tos_unreachable: + description: Host unreachable for TOS + type: bool + host_unknown: + description: Host unknown + type: bool + host_unreachable: + description: Host unreachable + type: bool + information_reply: + description: Information replies + type: bool + information_request: + description: Information requests + type: bool + mask_reply: + description: Mask replies + type: bool + mask_request: + description: Mask requests + type: bool + message_code: + description: ICMP message code + type: int + message_type: + description: ICMP message type + type: int + mobile_redirect: + description: Mobile host redirect + type: bool + net_redirect: + description: Network redirect + type: bool + net_tos_redirect: + description: Net redirect for TOS + type: bool + net_tos_unreachable: + description: Network unreachable for TOS + type: bool + net_unreachable: + description: Net unreachable + type: bool + network_unknown: + description: Network unknown + type: bool + no_room_for_option: + description: Parameter required but no room + type: bool + option_missing: + description: Parameter required but not present + type: bool + packet_too_big: + description: Fragmentation needed and DF set + type: bool + parameter_problem: + description: All parameter problems + type: bool + port_unreachable: + description: Port unreachable + type: bool + precedence_unreachable: + description: Precedence cutoff + type: bool + protocol_unreachable: + description: Protocol unreachable + type: bool + reassembly_timeout: + description: Reassembly timeout + type: bool + redirect: + description: All redirects + type: bool + router_advertisement: + description: Router discovery advertisements + type: bool + router_solicitation: + description: Router discovery solicitations + type: bool + source_quench: + description: Source quenches + type: bool + source_route_failed: + description: Source route failed + type: bool + time_exceeded: + description: All time exceededs + type: bool + timestamp_reply: + description: Timestamp replies + type: bool + timestamp_request: + description: Timestamp requests + type: bool + traceroute: + description: Traceroute + type: bool + ttl_exceeded: + description: TTL exceeded + type: bool + unreachable: + description: All unreachables + type: bool + message_num: + description: icmp msg type number. + type: int + icmpv6: + description: Options for icmpv6. + type: dict + suboptions: + address_unreachable: + description: address unreachable + type: bool + beyond_scope: + description: beyond_scope + type: bool + echo_reply: + description: echo_reply + type: bool + echo_request: + description: echo reques + type: bool + erroneous_header: + description: erroneous header + type: bool + fragment_reassembly_exceeded: + description: fragment_reassembly_exceeded + type: bool + hop_limit_exceeded: + description: hop limit exceeded + type: bool + neighbor_advertisement: + description: neighbor advertisement + type: bool + neighbor_solicitation: + description: neighbor_solicitation + type: bool + no_admin: + description: no admin + type: bool + no_route: + description: no route + type: bool + packet_too_big: + description: packet too big + type: bool + parameter_problem: + description: parameter problem + type: bool + port_unreachable: + description: port unreachable + type: bool + redirect_message: + description: redirect message + type: bool + reject_route: + description: reject route + type: bool + router_advertisement: + description: router_advertisement + type: bool + router_solicitation: + description: router_solicitation + type: bool + source_address_failed: + description: source_address_failed + type: bool + source_routing_error: + description: source_routing_error + type: bool + time_exceeded: + description: time_exceeded + type: bool + unreachable: + description: unreachable + type: bool + unrecognized_ipv6_option: + description: unrecognized_ipv6_option + type: bool + unrecognized_next_header: + description: unrecognized_next_header + type: bool + ip: + description: Internet Protocol. + type: dict + suboptions: + nexthop_group: + description: Nexthop-group name. + type: str + ipv6: + description: Internet V6 Protocol. + type: dict + suboptions: + nexthop_group: + description: Nexthop-group name. + type: str + source: + description: The packet's source address + type: dict + suboptions: + address: + description: dotted decimal notation of IP address + type: str + wildcard_bits: + description: Source wildcard bits + type: str + subnet_address: + description: A subnet address + type: str + host: + description: Host IP address + type: str + any: + description: Rule matches all source addresses + type: bool + port_protocol: + description: Specify source port/protocoli, along with operator. + (comes with tcp/udp). + type: dict + destination: + description: The packet's destination address + type: dict + suboptions: + address: + description: dotted decimal notation of IP address + type: str + wildcard_bits: + description: Source wildcard bits + type: str + subnet_address: + description: A subnet address + type: str + host: + description: Host IP address + type: str + any: + description: Rule matches all source addresses + type: bool + port_protocol: + description: Specify dest port/protocol, along with operator . + (comes with tcp/udp). + type: dict + ttl: + description: Compares the TTL (time-to-live) value in the packet to + a specified value + type: dict + suboptions: + eq: + description: Match a single TTL value + type: int + lt: + description: Match TTL lesser than this number + type: int + gt: + description: Match TTL greater than this number + type: int + neq: + description: Match TTL not equal to this value + type: int + fragments: + description: Match non-head fragment packets + type: bool + log: + description: Log matches against this rule + type: bool + tracked: + description: Match packets in existing ICMP/UDP/TCP connections + type: bool + hop_limit: + description: Hop limit value. + type: dict + running_config: + description: + - This option is used only with state I(parsed). + - The value of this option should be the output received from the EOS device by + executing the command B(show running-config | section access-list). + - The state I(parsed) reads the configuration from C(running_config) option and + transforms it into Ansible structured data as per the resource module's argspec + and the value is then returned in the I(parsed) key within the result. + type: str + state: + description: + - The state the configuration should be left in. + type: str + choices: + - deleted + - merged + - overridden + - replaced + - gathered + - rendered + - parsed + default: merged + +""" +EXAMPLES = """ +# Using merged + +# Before state: +# ------------- +# show running-config | section access-list +# ip access-list test1 +# 10 permit ip 10.10.10.0/24 any ttl eq 200 +# 20 permit ip 10.30.10.0/24 host 10.20.10.1 +# 30 deny tcp host 10.10.20.1 eq finger www any syn log +# 40 permit ip any any +# ipv6 access-list test2 +# 10 deny icmpv6 any any reject-route hop-limit eq 20 + +- name: Merge provided configuration with device configuration + arista.eos.eos_acls: + config: + - afi: ipv4 + acls: + - name: test1 + aces: + - sequence: 35 + grant: deny + protocol: ospf + source: + subnet_address: 20.0.0.0/8 + destnation: + any: true + state: merged + +# After state: +# ------------ +# +# show running-config | section access-list +# ip access-list test1 +# 10 permit ip 10.10.10.0/24 any ttl eq 200 +# 20 permit ip 10.30.10.0/24 host 10.20.10.1 +# 30 deny tcp host 10.10.20.1 eq finger www any syn log +# 35 deny ospf 20.0.0.0/8 any +# 40 permit ip any any +# ipv6 access-list test2 +# 10 deny icmpv6 any any reject-route hop-limit eq 20 + +# Using merged + +# Before state: +# ------------- +# show running-config | section access-list +# ip access-list test1 +# 10 permit ip 10.10.10.0/24 any ttl eq 200 +# 20 permit ip 10.30.10.0/24 host 10.20.10.1 +# 30 deny tcp host 10.10.20.1 eq finger www any syn log +# 40 permit ip any any +# ipv6 access-list test2 +# 10 deny icmpv6 any any reject-route hop-limit eq 20 + +- name: Merge to update the given configuration with an existing ace + arista.eos.eos_acls: + config: + - afi: ipv4 + acls: + - name: test1 + aces: + - sequence: 35 + log: true + ttl: + eq: 33 + state: merged + +# After state: +# ------------ +# +# show running-config | section access-list +# ip access-list test1 +# 10 permit ip 10.10.10.0/24 any ttl eq 200 +# 20 permit ip 10.30.10.0/24 host 10.20.10.1 +# 30 deny tcp host 10.10.20.1 eq finger www any syn log +# 35 deny ospf 20.0.0.0/8 any ttl eq 33 log +# 40 permit ip any any +# ipv6 access-list test2 +# 10 deny icmpv6 any any reject-route hop-limit eq 20 + +# Using replaced + +# Before state: +# ------------- +# show running-config | section access-list +# ip access-list test1 +# 10 permit ip 10.10.10.0/24 any ttl eq 200 +# 20 permit ip 10.30.10.0/24 host 10.20.10.1 +# 30 deny tcp host 10.10.20.1 eq finger www any syn log +# 40 permit ip any any +# ! +# ip access-list test3 +# 10 permit ip 35.33.0.0/16 any log +# ! +# ipv6 access-list test2 +# 10 deny icmpv6 any any reject-route hop-limit eq 20 + + + +- name: Replace device configuration with provided configuration + arista.eos.eos_acls: + config: + - afi: ipv4 + acls: + - name: test1 + aces: + - sequence: 35 + grant: permit + protocol: ospf + source: + subnet_address: 20.0.0.0/8 + destination: + any: true + state: replaced + +# After state: +# ------------ +# +# show running-config | section access-list +# ip access-list test1 +# 35 permit ospf 20.0.0.0/8 any +# ! +# ip access-list test3 +# 10 permit ip 35.33.0.0/16 any log +# ! +# ipv6 access-list test2 +# 10 deny icmpv6 any any reject-route hop-limit eq 20 + + +# Using overridden + +# Before state: +# ------------- +# show running-config | section access-list +# ip access-list test1 +# 10 permit ip 10.10.10.0/24 any ttl eq 200 +# 20 permit ip 10.30.10.0/24 host 10.20.10.1 +# 30 deny tcp host 10.10.20.1 eq finger www any syn log +# 40 permit ip any any +# ! +# ip access-list test3 +# 10 permit ip 35.33.0.0/16 any log +# ! +# ipv6 access-list test2 +# 10 deny icmpv6 any any reject-route hop-limit eq 20 + + + +- name: override device configuration with provided configuration + arista.eos.eos_acls: + config: + - afi: ipv4 + acls: + - name: test1 + aces: + - sequence: 35 + action: permit + protocol: ospf + source: + subnet_address: 20.0.0.0/8 + destination: + any: true + state: overridden + +# After state: +# ------------ +# +# show running-config | section access-list +# ip access-list test1 +# 35 permit ospf 20.0.0.0/8 any +# ! + +# Using deleted: + +# Before state: +# ------------- +# show running-config | section access-list +# ip access-list test1 +# 10 permit ip 10.10.10.0/24 any ttl eq 200 +# 20 permit ip 10.30.10.0/24 host 10.20.10.1 +# 30 deny tcp host 10.10.20.1 eq finger www any syn log +# 40 permit ip any any +# ipv6 access-list test2 +# 10 deny icmpv6 any any reject-route hop-limit eq 20 + +# ! + +- name: Delete provided configuration + arista.eos.eos_acls: + config: + - afi: ipv4 + state: deleted + +# After state: +# ------------ +# +# show running-config | section access-list + +# ipv6 access-list test2 +# 10 deny icmpv6 any any reject-route hop-limit eq 20 + + + +# Before state: +# ------------- +# show running-config | section access-list +# ip access-list test1 +# 10 permit ip 10.10.10.0/24 any ttl eq 200 +# 20 permit ip 10.30.10.0/24 host 10.20.10.1 +# 30 deny tcp host 10.10.20.1 eq finger www any syn log +# 40 permit ip any any +# ipv6 access-list test2 +# 10 deny icmpv6 any any reject-route hop-limit eq 20 + +# ! + +- name: Delete provided configuration + arista.eos.eos_acls: + config: + - afi: ipv4 + acls: + - name: test1 + state: deleted + +# After state: +# ------------ +# +# show running-config | section access-list + +# ipv6 access-list test2 +# 10 deny icmpv6 any any reject-route hop-limit eq 20 + + +# using gathered + +# ip access-list test1 +# 35 deny ospf 20.0.0.0/8 any +# ip access-list test2 +# 40 permit vlan 55 0xE2 icmpv6 any any log + +- name: Gather the exisitng condiguration + arista.eos.eos_acls: + state: gathered + +# returns: + + +# arista.eos.eos_acls: +# config: +# - afi: "ipv4" +# acls: +# - name: test1 +# aces: +# - sequence: 35 +# grant: "deny" +# protocol: "ospf" +# source: +# subnet_address: 20.0.0.0/8 +# destination: +# any: true +# - afi: "ipv6" +# acls: +# - name: test2 +# aces: +# - sequence: 40 +# grant: "permit" +# vlan: "55 0xE2" +# protocol: "icmpv6" +# log: true +# source: +# any: true +# destination: +# any: true + + +# using rendered + +- name: Delete provided configuration + arista.eos.eos_acls: + config: + - afi: ipv4 + acls: + - name: test1 + aces: + - sequence: 35 + grant: deny + protocol: ospf + source: + subnet_address: 20.0.0.0/8 + destination: + any: true + - afi: ipv6 + acls: + - name: test2 + aces: + - sequence: 40 + grant: permit + vlan: 55 0xE2 + protocol: icmpv6 + log: true + source: + any: true + destination: + any: true + state: rendered + +# returns: + +# ip access-list test1 +# 35 deny ospf 20.0.0.0/8 any +# ip access-list test2 +# 40 permit vlan 55 0xE2 icmpv6 any any log + + +# Using Parsed + +# parsed_acls.cfg + +# ipv6 access-list standard test2 +# 10 permit any log +# ! +# ip access-list test1 +# 35 deny ospf 20.0.0.0/8 any +# 45 remark Run by ansible +# 55 permit tcp any any +# ! + +- name: parse configs + arista.eos.eos_acls: + running_config: "{{ lookup('file', './parsed_acls.cfg') }}" + state: parsed + +# returns +# "parsed": [ +# { +# "acls": [ +# { +# "aces": [ +# { +# "destination": { +# "any": true +# }, +# "grant": "deny", +# "protocol": "ospf", +# "sequence": 35, +# "source": { +# "subnet_address": "20.0.0.0/8" +# } +# }, +# { +# "remark": "Run by ansible", +# "sequence": 45 +# }, +# { +# "destination": { +# "any": true +# }, +# "grant": "permit", +# "protocol": "tcp", +# "sequence": 55, +# "source": { +# "any": true +# } +# } +# ], +# "name": "test1" +# } +# ], +# "afi": "ipv4" +# }, +# { +# "acls": [ +# { +# "aces": [ +# { +# "grant": "permit", +# "log": true, +# "sequence": 10, +# "source": { +# "any": true +# } +# } +# ], +# "name": "test2", +# "standard": true +# } +# ], +# "afi": "ipv6" +# } +# ] + +""" +RETURN = """ +before: + description: The configuration prior to the model invocation. + returned: always + type: list + sample: > + The configuration returned will always be in the same format + of the parameters above. +after: + description: The resulting configuration model invocation. + returned: when changed + type: list + sample: > + The configuration returned will always be in the same format + of the parameters above. +commands: + description: The set of commands pushed to the remote device. + returned: always + type: list + sample: + - ipv6 access-list standard test2 + - 10 permit any log + - ip access-list test1 + - 35 deny ospf 20.0.0.0/8 any + - 45 remark Run by ansible + - 55 permit tcp any any +""" + + +from ansible.module_utils.basic import AnsibleModule +from ansible_collections.arista.eos.plugins.module_utils.network.eos.argspec.acls.acls import ( + AclsArgs, +) +from ansible_collections.arista.eos.plugins.module_utils.network.eos.config.acls.acls import ( + Acls, +) + + +def main(): + """ + Main entry point for module execution + + :returns: the result form module invocation + """ + + required_if = [ + ("state", "merged", ("config",)), + ("state", "replaced", ("config",)), + ("state", "overridden", ("config",)), + ("state", "rendered", ("config",)), + ("state", "parsed", ("running_config",)), + ] + mutually_exclusive = [("config", "running_config")] + + module = AnsibleModule( + argument_spec=AclsArgs.argument_spec, + required_if=required_if, + supports_check_mode=True, + mutually_exclusive=mutually_exclusive, + ) + + result = Acls(module).execute_module() + module.exit_json(**result) + + +if __name__ == "__main__": + main() diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/modules/eos_banner.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/modules/eos_banner.py new file mode 100644 index 00000000..2a8077ef --- /dev/null +++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/modules/eos_banner.py @@ -0,0 +1,219 @@ +#!/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/>. +# +from __future__ import absolute_import, division, print_function + +__metaclass__ = type + + +DOCUMENTATION = """ +module: eos_banner +author: Peter Sprygada (@privateip) +short_description: Manage multiline banners on Arista EOS devices +description: +- This will configure both login and motd banners on remote devices running Arista + EOS. It allows playbooks to add or remote banner text from the active running configuration. +version_added: 1.0.0 +extends_documentation_fragment: +- arista.eos.eos +notes: +- Tested against EOS 4.15 +options: + banner: + description: + - Specifies which banner that should be configured on the remote device. + required: true + choices: + - login + - motd + type: str + text: + description: + - The banner text that should be present in the remote device running configuration. This + argument accepts a multiline string. Requires I(state=present). + type: str + state: + description: + - Specifies whether or not the configuration is present in the current devices + active running configuration. + default: present + type: str + choices: + - present + - absent +""" + +EXAMPLES = """ +- name: configure the login banner + arista.eos.eos_banner: + banner: login + text: | + this is my login banner + that contains a multiline + string + state: present + +- name: remove the motd banner + arista.eos.eos_banner: + banner: motd + state: absent +""" + +RETURN = """ +commands: + description: The list of configuration mode commands to send to the device + returned: always + type: list + sample: + - banner login + - this is my login banner + - that contains a multiline + - string + - EOF +session_name: + description: The EOS config session name used to load the configuration + returned: if changes + type: str + sample: ansible_1479315771 +""" +from ansible.module_utils.basic import AnsibleModule +from ansible_collections.arista.eos.plugins.module_utils.network.eos.eos import ( + load_config, + run_commands, +) +from ansible_collections.arista.eos.plugins.module_utils.network.eos.eos import ( + eos_argument_spec, + is_local_eapi, +) +from ansible.module_utils.six import string_types +from ansible.module_utils._text import to_text + + +def map_obj_to_commands(updates, module): + commands = list() + want, have = updates + state = module.params["state"] + + if state == "absent" and have.get("text"): + if isinstance(have["text"], string_types): + commands.append("no banner %s" % module.params["banner"]) + elif have["text"].get("loginBanner") or have["text"].get("motd"): + commands.append({"cmd": "no banner %s" % module.params["banner"]}) + + elif state == "present": + if isinstance(have["text"], string_types): + if want["text"] != have["text"]: + commands.append("banner %s" % module.params["banner"]) + commands.extend(want["text"].strip().split("\n")) + commands.append("EOF") + else: + have_text = have["text"].get("loginBanner") or have["text"].get( + "motd" + ) + if have_text: + have_text = have_text.strip() + + if to_text(want["text"]) != have_text or not have_text: + # For EAPI we need to construct a dict with cmd/input + # key/values for the banner + commands.append( + { + "cmd": "banner %s" % module.params["banner"], + "input": want["text"].strip("\n"), + } + ) + + return commands + + +def map_config_to_obj(module): + output = run_commands(module, ["show banner %s" % module.params["banner"]]) + obj = {"banner": module.params["banner"], "state": "absent"} + if output: + if is_local_eapi(module): + # On EAPI we need to extract the banner text from dict key + # 'loginBanner' + if module.params["banner"] == "login": + banner_response_key = "loginBanner" + else: + banner_response_key = "motd" + if ( + isinstance(output[0], dict) + and banner_response_key in output[0].keys() + ): + obj["text"] = output[0] + else: + obj["text"] = output[0] + obj["state"] = "present" + return obj + + +def map_params_to_obj(module): + text = module.params["text"] + if text: + text = to_text(text).strip() + + return { + "banner": module.params["banner"], + "text": text, + "state": module.params["state"], + } + + +def main(): + """ main entry point for module execution + """ + argument_spec = dict( + banner=dict(required=True, choices=["login", "motd"]), + text=dict(), + state=dict(default="present", choices=["present", "absent"]), + ) + + argument_spec.update(eos_argument_spec) + + required_if = [("state", "present", ("text",))] + + module = AnsibleModule( + argument_spec=argument_spec, + required_if=required_if, + supports_check_mode=True, + ) + + warnings = list() + + result = {"changed": False} + if warnings: + result["warnings"] = warnings + want = map_params_to_obj(module) + have = map_config_to_obj(module) + + commands = map_obj_to_commands((want, have), module) + result["commands"] = commands + + if commands: + commit = not module.check_mode + response = load_config(module, commands, commit=commit) + if response.get("diff") and module._diff: + result["diff"] = {"prepared": response.get("diff")} + result["session_name"] = response.get("session") + result["changed"] = True + + module.exit_json(**result) + + +if __name__ == "__main__": + main() diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/modules/eos_bgp.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/modules/eos_bgp.py new file mode 100644 index 00000000..be30dfd4 --- /dev/null +++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/modules/eos_bgp.py @@ -0,0 +1,465 @@ +#!/usr/bin/python +# -*- coding: utf-8 -*- +# +# (c) 2019, Ansible by Red Hat, inc +# 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 + + +DOCUMENTATION = """ +module: eos_bgp +author: Nilashish Chakraborty (@NilashishC) +short_description: (deprecated, removed after 2023-01-29) Configure global BGP protocol settings on Arista EOS. +description: +- This module provides configuration management of global BGP parameters on Arista + EOS devices. +version_added: 1.0.0 +deprecated: + alternative: eos_bgp_global + why: Updated module released with more functionality. + removed_at_date: '2023-01-29' +notes: +- Tested against Arista vEOS Version 4.15.9M. +options: + config: + description: + - Specifies the BGP related configuration. + type: dict + suboptions: + bgp_as: + description: + - Specifies the BGP Autonomous System (AS) number to configure on the device. + type: int + required: true + router_id: + description: + - Configures the BGP routing process router-id value. + type: str + log_neighbor_changes: + description: + - Enable/disable logging neighbor up/down and reset reason. + type: bool + neighbors: + description: + - Specifies BGP neighbor related configurations. + type: list + elements: dict + suboptions: + neighbor: + description: + - Neighbor router address. + required: true + type: str + remote_as: + description: + - Remote AS of the BGP neighbor to configure. + type: int + required: true + update_source: + description: + - Source of the routing updates. + type: str + password: + description: + - Password to authenticate the BGP peer connection. + type: str + description: + description: + - Neighbor specific description. + type: str + ebgp_multihop: + description: + - Specifies the maximum hop count for EBGP neighbors not on directly connected + networks. + - The range is from 1 to 255. + type: int + peer_group: + description: + - Name of the peer group that the neighbor is a member of. + type: str + timers: + description: + - Specifies BGP neighbor timer related configurations. + type: dict + suboptions: + keepalive: + description: + - Frequency (in seconds) with which the device sends keepalive messages + to its peer. + - The range is from 0 to 3600. + type: int + required: true + holdtime: + description: + - Interval (in seconds) after not receiving a keepalive message that + device declares a peer dead. + - The range is from 3 to 7200. + - Setting this value to 0 will not send keep-alives (hold forever). + type: int + required: true + route_reflector_client: + description: + - Specify a neighbor as a route reflector client. + type: int + remove_private_as: + description: + - Remove the private AS number from outbound updates. + type: bool + enabled: + description: + - Administratively shutdown or enable a neighbor. + type: bool + maximum_prefix: + description: + - Maximum number of prefixes to accept from this peer. + - The range is from 0 to 4294967294. + type: int + redistribute: + description: + - Specifies the redistribute information from another routing protocol. + type: list + elements: dict + suboptions: + protocol: + description: + - Specifies the protocol for configuring redistribute information. + required: true + type: str + choices: [ospf, ospf3, static, connected, rip, isis] + route_map: + description: + - Specifies the route map reference. + type: str + networks: + description: + - Specify Networks to announce via BGP. + - For operation replace, this option is mutually exclusive with networks option + under address_family. + - For operation replace, if the device already has an address family activated, + this option is not allowed. + type: list + elements: dict + suboptions: + prefix: + description: + - Network ID to announce via BGP. + required: true + type: str + masklen: + description: + - Subnet mask length for the Network to announce(e.g, 8, 16, 24, etc.). + type: int + route_map: + description: + - Route map to modify the attributes. + type: str + address_family: + description: + - Specifies BGP address family related configurations. + type: list + elements: dict + suboptions: + afi: + description: + - Type of address family to configure. + type: str + choices: + - ipv4 + - ipv6 + required: true + redistribute: + description: + - Specifies the redistribute information from another routing protocol. + type: list + elements: dict + suboptions: + protocol: + description: + - Specifies the protocol for configuring redistribute information. + required: true + type: str + choices: + - ospf3 + - ospf + - isis + - static + - connected + - rip + route_map: + description: + - Specifies the route map reference. + type: str + networks: + description: + - Specify Networks to announce via BGP. + - For operation replace, this option is mutually exclusive with root level + networks option. + type: list + elements: dict + suboptions: + prefix: + description: + - Network ID to announce via BGP. + required: true + type: str + masklen: + description: + - Subnet mask length for the Network to announce(e.g, 8, 16, 24, etc.). + type: int + route_map: + description: + - Route map to modify the attributes. + type: str + neighbors: + description: + - Specifies BGP neighbor related configurations in Address Family configuration + mode. + type: list + elements: dict + suboptions: + neighbor: + description: + - Neighbor router address. + required: true + type: str + activate: + description: + - Enable the Address Family for this Neighbor. + type: bool + default_originate: + description: + - Originate default route to this neighbor. + type: bool + graceful_restart: + description: + - Enable/disable graceful restart mode for this neighbor. + type: bool + weight: + description: + - Assign weight for routes learnt from this neighbor. + - The range is from 0 to 65535 + type: int + operation: + description: + - Specifies the operation to be performed on the BGP process configured on the + device. + - In case of merge, the input configuration will be merged with the existing BGP + configuration on the device. + - In case of replace, if there is a diff between the existing configuration and + the input configuration, the existing configuration will be replaced by the + input configuration for every option that has the diff. + - In case of override, all the existing BGP configuration will be removed from + the device and replaced with the input configuration. + - In case of delete the existing BGP configuration will be removed from the device. + type: str + default: merge + choices: + - merge + - replace + - override + - delete +""" + +EXAMPLES = """ +- name: configure global bgp as 64496 + arista.eos.eos_bgp: + config: + bgp_as: 64496 + router_id: 192.0.2.1 + log_neighbor_changes: true + neighbors: + - neighbor: 203.0.113.5 + remote_as: 64511 + timers: + keepalive: 300 + holdtime: 360 + - neighbor: 198.51.100.2 + remote_as: 64498 + networks: + - prefix: 198.51.100.0 + route_map: RMAP_1 + - prefix: 192.0.2.0 + masklen: 23 + address_family: + - afi: ipv4 + safi: unicast + redistribute: + - protocol: isis + route_map: RMAP_1 + operation: merge + +- name: Configure BGP neighbors + arista.eos.eos_bgp: + config: + bgp_as: 64496 + neighbors: + - neighbor: 192.0.2.10 + remote_as: 64496 + description: IBGP_NBR_1 + ebgp_multihop: 100 + timers: + keepalive: 300 + holdtime: 360 + + - neighbor: 192.0.2.15 + remote_as: 64496 + description: IBGP_NBR_2 + ebgp_multihop: 150 + operation: merge + +- name: Configure root-level networks for BGP + arista.eos.eos_bgp: + config: + bgp_as: 64496 + networks: + - prefix: 203.0.113.0 + masklen: 27 + route_map: RMAP_1 + + - prefix: 203.0.113.32 + masklen: 27 + route_map: RMAP_2 + operation: merge + +- name: Configure BGP neighbors under address family mode + arista.eos.eos_bgp: + config: + bgp_as: 64496 + address_family: + - afi: ipv4 + neighbors: + - neighbor: 203.0.113.10 + activate: yes + default_originate: true + + - neighbor: 192.0.2.15 + activate: yes + graceful_restart: true + operation: merge + +- name: remove bgp as 64496 from config + arista.eos.eos_bgp: + config: + bgp_as: 64496 + operation: delete +""" + +RETURN = """ +commands: + description: The list of configuration mode commands to send to the device + returned: always + type: list + sample: + - router bgp 64496 + - bgp router-id 192.0.2.1 + - bgp log-neighbor-changes + - neighbor 203.0.113.5 remote-as 64511 + - neighbor 203.0.113.5 timers 300 360 + - neighbor 198.51.100.2 remote-as 64498 + - network 198.51.100.0 route-map RMAP_1 + - network 192.0.2.0 mask 255.255.254.0 + - address-family ipv4 + - redistribute isis route-map RMAP_1 + - exit-address-family +""" +from ansible.module_utils._text import to_text +from ansible_collections.arista.eos.plugins.module_utils.network.eos.providers.module import ( + NetworkModule, +) +from ansible_collections.arista.eos.plugins.module_utils.network.eos.providers.cli.config.bgp.process import ( + REDISTRIBUTE_PROTOCOLS, +) + + +def main(): + """ main entry point for module execution + """ + network_spec = { + "prefix": dict(required=True), + "masklen": dict(type="int"), + "route_map": dict(), + } + + redistribute_spec = { + "protocol": dict(choices=list(REDISTRIBUTE_PROTOCOLS), required=True), + "route_map": dict(), + } + + timer_spec = { + "keepalive": dict(type="int", required=True), + "holdtime": dict(type="int", required=True), + } + + neighbor_spec = { + "neighbor": dict(required=True), + "remote_as": dict(type="int", required=True), + "update_source": dict(), + "password": dict(no_log=True), + "enabled": dict(type="bool"), + "description": dict(), + "ebgp_multihop": dict(type="int"), + "timers": dict(type="dict", options=timer_spec), + "peer_group": dict(), + "maximum_prefix": dict(type="int"), + "route_reflector_client": dict(type="int"), + "remove_private_as": dict(type="bool"), + } + + af_neighbor_spec = { + "neighbor": dict(required=True), + "activate": dict(type="bool"), + "default_originate": dict(type="bool"), + "graceful_restart": dict(type="bool"), + "weight": dict(type="int"), + } + + address_family_spec = { + "afi": dict(choices=["ipv4", "ipv6"], required=True), + "networks": dict(type="list", elements="dict", options=network_spec), + "redistribute": dict( + type="list", elements="dict", options=redistribute_spec + ), + "neighbors": dict( + type="list", elements="dict", options=af_neighbor_spec + ), + } + + config_spec = { + "bgp_as": dict(type="int", required=True), + "router_id": dict(), + "log_neighbor_changes": dict(type="bool"), + "neighbors": dict(type="list", elements="dict", options=neighbor_spec), + "address_family": dict( + type="list", elements="dict", options=address_family_spec + ), + "redistribute": dict( + type="list", elements="dict", options=redistribute_spec + ), + "networks": dict(type="list", elements="dict", options=network_spec), + } + + argument_spec = { + "config": dict(type="dict", options=config_spec), + "operation": dict( + default="merge", choices=["merge", "replace", "override", "delete"] + ), + } + + module = NetworkModule( + argument_spec=argument_spec, supports_check_mode=True + ) + + try: + result = module.edit_config(config_filter="| section bgp") + except Exception as exc: + module.fail_json(msg=to_text(exc)) + + module.exit_json(**result) + + +if __name__ == "__main__": + main() diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/modules/eos_bgp_address_family.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/modules/eos_bgp_address_family.py new file mode 100644 index 00000000..7e64fdef --- /dev/null +++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/modules/eos_bgp_address_family.py @@ -0,0 +1,1332 @@ +#!/usr/bin/python +# -*- coding: utf-8 -*- +# Copyright 2021 Red Hat +# GNU General Public License v3.0+ +# (see COPYING or https://www.gnu.org/licenses/gpl-3.0.txt) + +############################################# +# WARNING # +############################################# +# +# This file is auto generated by the resource +# module builder playbook. +# +# Do not edit this file manually. +# +# Changes to this file will be over written +# by the resource module builder. +# +# Changes should be made in the model used to +# generate this file or in the resource module +# builder template. +# +############################################# + +""" +The module file for eos_bgp_address_family +""" + +from __future__ import absolute_import, division, print_function + +__metaclass__ = type + +DOCUMENTATION = """ +module: eos_bgp_address_family +short_description: Manages BGP address family resource module +description: This module configures and manages the attributes of BGP AF on Arista + EOS platforms. +version_added: 1.4.0 +author: Gomathi Selvi Srinivasan (@GomathiselviS) +notes: +- Tested against Arista EOS 4.23.0F +- This module works with connection C(network_cli). See the L(EOS Platform Options,eos_platform_options). +options: + config: + description: Configurations for BGP address family. + type: dict + suboptions: + as_number: + description: Autonomous system number. + type: str + address_family: &address_family + description: Enable address family and enter its config mode + type: list + elements: dict + suboptions: + afi: + description: address family. + type: str + choices: ['ipv4', 'ipv6', 'evpn'] + safi: + description: Address family type for ipv4. + type: str + choices: ['labeled-unicast', 'multicast'] + bgp_params: + description: BGP parameters. + type: dict + suboptions: + additional_paths: + description: BGP additional-paths commands + type: str + choices: ['install', 'send', 'receive'] + next_hop_address_family: + description: Next-hop address-family configuration + type: str + choices: ['ipv6'] + next_hop_unchanged: + description: Preserve original nexthop while advertising routes to + eBGP peers. + type: bool + redistribute_internal: + description: Redistribute internal BGP routes. + type: bool + route: + description: Configure route-map for route installation. + type: str + graceful_restart: + description: Enable graceful restart mode. + type: bool + neighbor: + description: Configure routing for a network. + type: list + elements: dict + suboptions: + peer: + type: str + description: Neighbor address/ peer-group name. + activate: + description: Activate neighbor in the address family. + type: bool + additional_paths: + description: BGP additional-paths commands. + type: str + choices: ['send', 'receive'] + default_originate: + description: Originate default route to this neighbor. + type: dict + suboptions: + route_map: + description: Route map reference. + type: str + always: + description: Always originate default route to this neighbor. + type: bool + graceful_restart: + description: Enable graceful restart mode. + type: bool + next_hop_address_family: + description: Next-hop address-family configuration + type: str + choices: ['ipv6'] + next_hop_unchanged: + description: Preserve original nexthop while advertising routes to + eBGP peers. + type: bool + prefix_list: + description: Prefix list reference. + type: dict + suboptions: + direction: + description: Configure an inbound/outbound prefix-list. + type: str + choices: ['in', 'out'] + name: + description: prefix list name. + type: str + route_map: + description: Route map reference. + type: dict + suboptions: + direction: + description: Configure an inbound/outbound route-map. + type: str + choices: ['in', 'out'] + name: + description: Route map name. + type: str + weight: + description: Weight to assign. + type: int + encapsulation: + description: Default transport encapsulation for neighbor. Applicable for evpn address-family. + type: dict + suboptions: + transport: + description: MPLS/VXLAN transport. + type: str + choices: ['mpls', 'vxlan'] + source_interface: + description: Source interface to update BGP next hop address. Applicable for mpls transport. + type: str + network: + description: configure routing for network. + type: list + elements: dict + suboptions: + route_map: + description: Route map reference. + type: str + address: + description: network address. + type: str + redistribute: + description: Redistribute routes in to BGP. + type: list + elements: dict + suboptions: + protocol: + description: Routes to be redistributed. + type: str + choices: ['isis', 'ospf3', 'dhcp'] + route_map: + description: Route map reference. + type: str + isis_level: + description: Applicable for isis routes. Specify isis route level. + type: str + choices: ['level-1', 'level-2', 'level-1-2'] + ospf_route: + description: ospf route options. + type: str + choices: ['internal', 'external', 'nssa_external_1', 'nssa_external_2'] + route_target: + description: Route target + type: dict + suboptions: + mode: + description: route import or route export. + type: str + choices: ['both', 'import', 'export'] + target: + description: route target + type: str + vrf: + description: name of the VRF in which BGP will be configured. + type: str + running_config: + description: + - This option is used only with state I(parsed). + - The value of this option should be the output received from the EOS device by + executing the command B(show running-config | section bgp). + - The state I(parsed) reads the configuration from C(running_config) option and + transforms it into Ansible structured data as per the resource module's argspec + and the value is then returned in the I(parsed) key within the result. + type: str + state: + description: + - The state the configuration should be left in. + type: str + choices: [deleted, merged, overridden, replaced, gathered, rendered, parsed] + default: merged + +""" + +EXAMPLES = """ + +# Using merged + +# Before state + +# veos(config)#show running-config | section bgp +# veos(config)# + + - name: Merge provided configuration with device configuration + arista.eos.eos_bgp_address_family: + config: + as_number: "10" + address_family: + - afi: "ipv4" + redistribute: + - protocol: "ospf3" + ospf_route: "external" + network: + - address: "1.1.1.0/24" + - address: "1.5.1.0/24" + route_map: "MAP01" + - afi: "ipv6" + bgp_params: + additional_paths: "receive" + neighbor: + - peer: "peer2" + default_originate: + always: True + - afi: "ipv6" + redistribute: + - protocol: "isis" + isis_level: "level-2" + route_target: + mode: "export" + target: "33:11" + vrf: "vrft" + state: merged + +# After state: + +# veos(config-router-bgp)#show running-config | section bgp +# router bgp 10 +# neighbor peer2 peer-group +# neighbor peer2 maximum-routes 12000 +# neighbor 1.1.1.1 maximum-routes 12000 +# ! +# address-family ipv4 +# neighbor 1.1.1.1 activate +# network 1.1.1.0/24 +# network 1.5.1.0/24 route-map MAP01 +# redistribute ospf3 match external +# ! +# address-family ipv6 +# bgp additional-paths receive +# neighbor peer2 activate +# neighbor peer2 default-originate always +# ! +# vrf vrft +# address-family ipv6 +# route-target export 33:11 +# redistribute isis level-2 +# veos(config-router-bgp)# + +# Module Execution: + +# "after": { +# "address_family": [ +# { +# "afi": "ipv4", +# "redistribute": [ +# { +# "ospf_route": "external", +# "protocol": "ospf3" +# } +# ] +# }, +# { +# "afi": "ipv6", +# "bgp_params": { +# "additional_paths": "receive" +# }, +# "neighbor": [ +# { +# "default_originate": { +# "always": true +# }, +# "peer": "peer2" +# } +# ] +# }, +# { +# "afi": "ipv6", +# "redistribute": [ +# { +# "isis_level": "level-2", +# "protocol": "isis" +# } +# ], +# "route_target": { +# "mode": "export", +# "target": "33:11" +# }, +# "vrf": "vrft" +# } +# ], +# "as_number": "10" +# }, +# "before": {}, +# "changed": true, +# "commands": [ +# "router bgp 10", +# "address-family ipv4", +# "redistribute ospf3 match external", +# "network 1.1.1.0/24", +# "network 1.5.1.0/24 route-map MAP01", +# "exit", +# "address-family ipv6", +# "neighbor peer2 default-originate always", +# "bgp additional-paths receive", +# "exit", +# "vrf vrft", +# "address-family ipv6", +# "redistribute isis level-2", +# "route-target export 33:11", +# "exit", +# "exit" +# ], + +# Using replaced: + +# Before State: + +# veos(config-router-bgp)#show running-config | section bgp +# router bgp 10 +# neighbor peer2 peer-group +# neighbor peer2 maximum-routes 12000 +# neighbor 1.1.1.1 maximum-routes 12000 +# ! +# address-family ipv4 +# neighbor 1.1.1.1 activate +# network 1.1.1.0/24 +# network 1.5.1.0/24 route-map MAP01 +# redistribute ospf3 match external +# ! +# address-family ipv6 +# bgp additional-paths receive +# neighbor peer2 activate +# neighbor peer2 default-originate always +# ! +# vrf vrft +# address-family ipv6 +# route-target export 33:11 +# redistribute isis level-2 +# veos(config-router-bgp)# +# + + - name: Replace + arista.eos.eos_bgp_address_family: + config: + as_number: "10" + address_family: + - afi: "ipv6" + vrf: "vrft" + redistribute: + - protocol: "ospf3" + ospf_route: "external" + - afi: "ipv6" + redistribute: + - protocol: "isis" + isis_level: "level-2" + state: replaced + +# After State: + +# veos(config-router-bgp)#show running-config | section bgp +# router bgp 10 +# neighbor peer2 peer-group +# neighbor peer2 maximum-routes 12000 +# neighbor 1.1.1.1 maximum-routes 12000 +# ! +# address-family ipv4 +# neighbor 1.1.1.1 activate +# network 1.1.1.0/24 +# network 1.5.1.0/24 route-map MAP01 +# redistribute ospf3 match external +# ! +# address-family ipv6 +# neighbor peer2 default-originate always +# redistribute isis level-2 +# ! +# vrf vrft +# address-family ipv6 +# redistribute ospf3 match external +# veos(config-router-bgp)# +# +# +# # Module Execution: +# +# "after": { +# "address_family": [ +# { +# "afi": "ipv4", +# "neighbor": [ +# { +# "activate": true, +# "peer": "1.1.1.1" +# } +# ], +# "network": [ +# { +# "address": "1.1.1.0/24" +# }, +# { +# "address": "1.5.1.0/24", +# "route_map": "MAP01" +# } +# ], +# "redistribute": [ +# { +# "ospf_route": "external", +# "protocol": "ospf3" +# } +# ] +# }, +# { +# "afi": "ipv6", +# "neighbor": [ +# { +# "default_originate": { +# "always": true +# }, +# "peer": "peer2" +# } +# ], +# "redistribute": [ +# { +# "isis_level": "level-2", +# "protocol": "isis" +# } +# ] +# }, +# { +# "afi": "ipv6", +# "redistribute": [ +# { +# "ospf_route": "external", +# "protocol": "ospf3" +# } +# ], +# "vrf": "vrft" +# } +# ], +# "as_number": "10" +# }, +# "before": { +# "address_family": [ +# { +# "afi": "ipv4", +# "neighbor": [ +# { +# "activate": true, +# "peer": "1.1.1.1" +# } +# ], +# "network": [ +# { +# "address": "1.1.1.0/24" +# }, +# { +# "address": "1.5.1.0/24", +# "route_map": "MAP01" +# } +# ], +# "redistribute": [ +# { +# "ospf_route": "external", +# "protocol": "ospf3" +# } +# ] +# }, +# { +# "afi": "ipv6", +# "bgp_params": { +# "additional_paths": "receive" +# }, +# "neighbor": [ +# { +# "activate": true, +# "default_originate": { +# "always": true +# }, +# "peer": "peer2" +# } +# ] +# }, +# { +# "afi": "ipv6", +# "redistribute": [ +# { +# "isis_level": "level-2", +# "protocol": "isis" +# } +# ], +# "route_target": { +# "mode": "export", +# "target": "33:11" +# }, +# "vrf": "vrft" +# } +# ], +# "as_number": "10" +# }, +# "changed": true, +# "commands": [ +# "router bgp 10", +# "vrf vrft", +# "address-family ipv6", +# "redistribute ospf3 match external", +# "no redistribute isis level-2", +# "no route-target export 33:11", +# "exit", +# "exit", +# "address-family ipv6", +# "redistribute isis level-2", +# "no neighbor peer2 activate", +# "no bgp additional-paths receive", +# "exit" +# ], + +# Using overridden (overriding af at global context): +# Before state: + +# veos(config-router-bgp)#show running-config | section bgp +# router bgp 10 +# neighbor peer2 peer-group +# neighbor peer2 maximum-routes 12000 +# neighbor 1.1.1.1 maximum-routes 12000 +# ! +# address-family ipv4 +# neighbor 1.1.1.1 activate +# network 1.1.1.0/24 +# network 1.5.1.0/24 route-map MAP01 +# redistribute ospf3 match external +# ! +# address-family ipv6 +# neighbor peer2 default-originate always +# redistribute isis level-2 +# ! +# vrf vrft +# address-family ipv6 +# redistribute ospf3 match external +# veos(config-router-bgp)# + + - name: Overridden + arista.eos.eos_bgp_address_family: + config: + as_number: "10" + address_family: + - afi: "ipv4" + bgp_params: + additional_paths: "receive" + neighbor: + - peer: "peer2" + default_originate: + always: True + state: overridden + +# After State: +# veos(config-router-bgp)#show running-config | section bgp +# router bgp 10 +# neighbor peer2 peer-group +# neighbor peer2 maximum-routes 12000 +# neighbor 1.1.1.1 maximum-routes 12000 +# ! +# address-family ipv4 +# bgp additional-paths receive +# neighbor peer2 default-originate always +# ! +# vrf vrft +# address-family ipv6 +# redistribute ospf3 match external +# veos(config-router-bgp)# +# +# Module Execution: +# +# "after": { +# "address_family": [ +# { +# "afi": "ipv4", +# "bgp_params": { +# "additional_paths": "receive" +# }, +# "neighbor": [ +# { +# "default_originate": { +# "always": true +# }, +# "peer": "peer2" +# } +# ] +# }, +# { +# "afi": "ipv6", +# "redistribute": [ +# { +# "ospf_route": "external", +# "protocol": "ospf3" +# } +# ], +# "vrf": "vrft" +# } +# ], +# "as_number": "10" +# }, +# "before": { +# "address_family": [ +# { +# "afi": "ipv4", +# "neighbor": [ +# { +# "activate": true, +# "peer": "1.1.1.1" +# } +# ], +# "network": [ +# { +# "address": "1.1.1.0/24" +# }, +# { +# "address": "1.5.1.0/24", +# "route_map": "MAP01" +# } +# ], +# "redistribute": [ +# { +# "ospf_route": "external", +# "protocol": "ospf3" +# } +# ] +# }, +# { +# "afi": "ipv6", +# "neighbor": [ +# { +# "default_originate": { +# "always": true +# }, +# "peer": "peer2" +# } +# ], +# "redistribute": [ +# { +# "isis_level": "level-2", +# "protocol": "isis" +# } +# ] +# }, +# { +# "afi": "ipv6", +# "redistribute": [ +# { +# "ospf_route": "external", +# "protocol": "ospf3" +# } +# ], +# "vrf": "vrft" +# } +# ], +# "as_number": "10" +# }, +# "changed": true, +# "commands": [ +# "router bgp 10", +# "address-family ipv4", +# "no redistribute ospf3 match external", +# "no network 1.1.1.0/24", +# "no network 1.5.1.0/24 route-map MAP01", +# "neighbor peer2 default-originate always", +# "no neighbor 1.1.1.1 activate", +# "bgp additional-paths receive", +# "exit", +# "no address-family ipv6" +# ], + +# using Overridden (overridding af in vrf context): + +# Before State: + +# veos(config-router-bgp)#show running-config | section bgp +# router bgp 10 +# neighbor peer2 peer-group +# neighbor peer2 maximum-routes 12000 +# neighbor 1.1.1.1 maximum-routes 12000 +# ! +# address-family ipv4 +# bgp additional-paths receive +# neighbor peer2 default-originate always +# no neighbor 1.1.1.1 activate +# network 1.1.1.0/24 +# network 1.5.1.0/24 route-map MAP01 +# redistribute ospf3 match external +# ! +# address-family ipv6 +# bgp additional-paths receive +# neighbor peer2 default-originate always +# ! +# vrf vrft +# address-family ipv6 +# route-target export 33:11 +# redistribute isis level-2 +# redistribute ospf3 match external +# veos(config-router-bgp)# + + + - name: Overridden + arista.eos.eos_bgp_address_family: + config: + as_number: "10" + address_family: + - afi: "ipv4" + bgp_params: + additional_paths: "receive" + neighbor: + - peer: "peer2" + default_originate: + always: True + vrf: vrft + state: overridden + +# After State: + +# veos(config-router-bgp)#show running-config | section bgp +# router bgp 10 +# neighbor peer2 peer-group +# neighbor peer2 maximum-routes 12000 +# neighbor 1.1.1.1 maximum-routes 12000 +# ! +# address-family ipv4 +# bgp additional-paths receive +# neighbor peer2 default-originate always +# network 1.1.1.0/24 +# network 1.5.1.0/24 route-map MAP01 +# redistribute ospf3 match external +# ! +# address-family ipv6 +# bgp additional-paths receive +# neighbor peer2 default-originate always +# ! +# vrf vrft +# address-family ipv4 +# bgp additional-paths receive +# veos(config-router-bgp)# +# +# Module Execution: +# +# "after": { +# "address_family": [ +# { +# "afi": "ipv4", +# "bgp_params": { +# "additional_paths": "receive" +# }, +# "neighbor": [ +# { +# "default_originate": { +# "always": true +# }, +# "peer": "peer2" +# } +# ], +# "network": [ +# { +# "address": "1.1.1.0/24" +# }, +# { +# "address": "1.5.1.0/24", +# "route_map": "MAP01" +# } +# ], +# "redistribute": [ +# { +# "ospf_route": "external", +# "protocol": "ospf3" +# } +# ] +# }, +# { +# "afi": "ipv6", +# "bgp_params": { +# "additional_paths": "receive" +# }, +# "neighbor": [ +# { +# "default_originate": { +# "always": true +# }, +# "peer": "peer2" +# } +# ] +# }, +# { +# "afi": "ipv4", +# "bgp_params": { +# "additional_paths": "receive" +# }, +# "vrf": "vrft" +# } +# ], +# "as_number": "10" +# }, +# "before": { +# "address_family": [ +# { +# "afi": "ipv4", +# "bgp_params": { +# "additional_paths": "receive" +# }, +# "neighbor": [ +# { +# "default_originate": { +# "always": true +# }, +# "peer": "peer2" +# } +# ], +# "network": [ +# { +# "address": "1.1.1.0/24" +# }, +# { +# "address": "1.5.1.0/24", +# "route_map": "MAP01" +# } +# ], +# "redistribute": [ +# { +# "ospf_route": "external", +# "protocol": "ospf3" +# } +# ] +# }, +# { +# "afi": "ipv6", +# "bgp_params": { +# "additional_paths": "receive" +# }, +# "neighbor": [ +# { +# "default_originate": { +# "always": true +# }, +# "peer": "peer2" +# } +# ] +# }, +# { +# "afi": "ipv6", +# "redistribute": [ +# { +# "isis_level": "level-2", +# "protocol": "isis" +# }, +# { +# "ospf_route": "external", +# "protocol": "ospf3" +# } +# ], +# "route_target": { +# "mode": "export", +# "target": "33:11" +# }, +# "vrf": "vrft" +# } +# ], +# "as_number": "10" +# }, +# "changed": true, +# "commands": [ +# "router bgp 10", +# "vrf vrft", +# "address-family ipv4", +# "neighbor peer2 default-originate always", +# "bgp additional-paths receive", +# "exit", +# "exit", +# " vrf vrft", +# "no address-family ipv6" +# ], + +# Using Deleted: + +# veos(config-router-bgp)#show running-config | section bgp +# router bgp 10 +# neighbor peer2 peer-group +# neighbor peer2 maximum-routes 12000 +# neighbor 1.1.1.1 maximum-routes 12000 +# ! +# address-family ipv4 +# bgp additional-paths receive +# neighbor peer2 default-originate always +# no neighbor 1.1.1.1 activate +# network 1.1.1.0/24 +# network 1.5.1.0/24 route-map MAP01 +# redistribute ospf3 match external +# ! +# address-family ipv6 +# bgp additional-paths receive +# neighbor peer2 default-originate always +# ! +# vrf vrft +# address-family ipv4 +# bgp additional-paths receive +# veos(config-router-bgp)# + + - name: Delete + arista.eos.eos_bgp_address_family: + config: + as_number: "10" + address_family: + - afi: "ipv6" + vrf: "vrft" + - afi: "ipv6" + state: deleted + +# After State: + +# veos(config-router-bgp)#show running-config | section bgp +# router bgp 10 +# neighbor peer2 peer-group +# neighbor peer2 maximum-routes 12000 +# neighbor 1.1.1.1 maximum-routes 12000 +# ! +# address-family ipv4 +# bgp additional-paths receive +# neighbor peer2 default-originate always +# no neighbor 1.1.1.1 activate +# network 1.1.1.0/24 +# network 1.5.1.0/24 route-map MAP01 +# redistribute ospf3 match external +# ! +# vrf vrft +# address-family ipv4 +# bgp additional-paths receive +# veos(config-router-bgp)# +# +# Module Execution: +# +# "after": { +# "address_family": [ +# { +# "afi": "ipv4", +# "bgp_params": { +# "additional_paths": "receive" +# }, +# "neighbor": [ +# { +# "default_originate": { +# "always": true +# }, +# "peer": "peer2" +# } +# ], +# "network": [ +# { +# "address": "1.1.1.0/24" +# }, +# { +# "address": "1.5.1.0/24", +# "route_map": "MAP01" +# } +# ], +# "redistribute": [ +# { +# "ospf_route": "external", +# "protocol": "ospf3" +# } +# ] +# }, +# { +# "afi": "ipv4", +# "bgp_params": { +# "additional_paths": "receive" +# }, +# "vrf": "vrft" +# } +# ], +# "as_number": "10" +# }, +# "before": { +# "address_family": [ +# { +# "afi": "ipv4", +# "bgp_params": { +# "additional_paths": "receive" +# }, +# "neighbor": [ +# { +# "default_originate": { +# "always": true +# }, +# "peer": "peer2" +# } +# ], +# "network": [ +# { +# "address": "1.1.1.0/24" +# }, +# { +# "address": "1.5.1.0/24", +# "route_map": "MAP01" +# } +# ], +# "redistribute": [ +# { +# "ospf_route": "external", +# "protocol": "ospf3" +# } +# ] +# }, +# { +# "afi": "ipv6", +# "bgp_params": { +# "additional_paths": "receive" +# }, +# "neighbor": [ +# { +# "default_originate": { +# "always": true +# }, +# "peer": "peer2" +# } +# ] +# }, +# { +# "afi": "ipv4", +# "bgp_params": { +# "additional_paths": "receive" +# }, +# "vrf": "vrft" +# } +# ], +# "as_number": "10" +# }, + +# Using parsed: + +# parsed_bgp_address_family.cfg : + +# router bgp 10 +# neighbor n2 peer-group +# neighbor n2 next-hop-unchanged +# neighbor n2 maximum-routes 12000 +# neighbor peer2 peer-group +# neighbor peer2 maximum-routes 12000 +# network 1.1.1.0/24 +# network 1.5.1.0/24 route-map MAP01 +# ! +# address-family ipv4 +# bgp additional-paths receive +# neighbor peer2 default-originate always +# redistribute ospf3 match external +# ! +# address-family ipv6 +# no bgp additional-paths receive +# neighbor n2 next-hop-unchanged +# redistribute isis level-2 +# ! +# vrf bgp_10 +# ip access-group acl01 +# ucmp fec threshold trigger 33 clear 22 warning-only +# ! +# address-family ipv4 +# route-target import 20:11 +# ! +# vrf vrft +# address-family ipv4 +# bgp additional-paths receive +# ! +# address-family ipv6 +# redistribute ospf3 match external + + - name: parse configs + arista.eos.eos_bgp_address_family: + running_config: "{{ lookup('file', './parsed_bgp_address_family.cfg') }}" + state: parsed + +# Module Execution: +# "parsed": { +# "address_family": [ +# { +# "afi": "ipv4", +# "bgp_params": { +# "additional_paths": "receive" +# }, +# "neighbor": [ +# { +# "default_originate": { +# "always": true +# }, +# "peer": "peer2" +# } +# ], +# "redistribute": [ +# { +# "ospf_route": "external", +# "protocol": "ospf3" +# } +# ] +# }, +# { +# "afi": "ipv6", +# "neighbor": [ +# { +# "next_hop_unchanged": true, +# "peer": "n2" +# } +# ], +# "redistribute": [ +# { +# "isis_level": "level-2", +# "protocol": "isis" +# } +# ] +# }, +# { +# "afi": "ipv4", +# "route_target": { +# "mode": "import", +# "target": "20:11" +# }, +# "vrf": "bgp_10" +# }, +# { +# "afi": "ipv4", +# "bgp_params": { +# "additional_paths": "receive" +# }, +# "vrf": "vrft" +# }, +# { +# "afi": "ipv6", +# "redistribute": [ +# { +# "ospf_route": "external", +# "protocol": "ospf3" +# } +# ], +# "vrf": "vrft" +# } +# ], +# "as_number": "10" +# } +# } + +# Using gathered: + +# Device config: +# veos(config-router-bgp)#show running-config | section bgp +# router bgp 10 +# neighbor peer2 peer-group +# neighbor peer2 maximum-routes 12000 +# neighbor 1.1.1.1 maximum-routes 12000 +# ! +# address-family ipv4 +# bgp additional-paths receive +# neighbor peer2 default-originate always +# no neighbor 1.1.1.1 activate +# network 1.1.1.0/24 +# network 1.5.1.0/24 route-map MAP01 +# redistribute ospf3 match external +# ! +# vrf vrft +# address-family ipv4 +# bgp additional-paths receive +# veos(config-router-bgp)# + + - name: gather configs + arista.eos.eos_bgp_address_family: + state: gathered + +# Module Execution: +# "gathered": { +# "address_family": [ +# { +# "afi": "ipv4", +# "bgp_params": { +# "additional_paths": "receive" +# }, +# "neighbor": [ +# { +# "default_originate": { +# "always": true +# }, +# "peer": "peer2" +# } +# ], +# "network": [ +# { +# "address": "1.1.1.0/24" +# }, +# { +# "address": "1.5.1.0/24", +# "route_map": "MAP01" +# } +# ], +# "redistribute": [ +# { +# "ospf_route": "external", +# "protocol": "ospf3" +# } +# ] +# }, +# { +# "afi": "ipv4", +# "bgp_params": { +# "additional_paths": "receive" +# }, +# "vrf": "vrft" +# } +# ], +# "as_number": "10" +# }, + +# using rendered: + + - name: Render + arista.eos.eos_bgp_address_family: + config: + as_number: "10" + address_family: + - afi: "ipv4" + redistribute: + - protocol: "ospf3" + ospf_route: "external" + network: + - address: "1.1.1.0/24" + - address: "1.5.1.0/24" + route_map: "MAP01" + - afi: "ipv6" + bgp_params: + additional_paths: "receive" + neighbor: + - peer: "peer2" + default_originate: + always: True + - afi: "ipv6" + redistribute: + - protocol: "isis" + isis_level: "level-2" + route_target: + mode: "export" + target: "33:11" + vrf: "vrft" + + state: rendered + +# Module Execution: + +# "rendered": [ +# "router bgp 10", +# "address-family ipv4", +# "redistribute ospf3 match external", +# "network 1.1.1.0/24", +# "network 1.5.1.0/24 route-map MAP01", +# "exit", +# "address-family ipv6", +# "neighbor peer2 default-originate always", +# "bgp additional-paths receive", +# "exit", +# "vrf vrft", +# "address-family ipv6", +# "redistribute isis level-2", +# "route-target export 33:11", +# "exit", +# "exit" +# ] +# + +""" + +from ansible.module_utils.basic import AnsibleModule +from ansible_collections.arista.eos.plugins.module_utils.network.eos.argspec.bgp_address_family.bgp_address_family import ( + Bgp_afArgs, +) +from ansible_collections.arista.eos.plugins.module_utils.network.eos.config.bgp_address_family.bgp_address_family import ( + Bgp_af, +) + + +def main(): + """ + Main entry point for module execution + + :returns: the result form module invocation + """ + module = AnsibleModule( + argument_spec=Bgp_afArgs.argument_spec, + mutually_exclusive=[], + required_if=[], + supports_check_mode=False, + ) + + result = Bgp_af(module).execute_module() + module.exit_json(**result) + + +if __name__ == "__main__": + main() diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/modules/eos_bgp_global.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/modules/eos_bgp_global.py new file mode 100644 index 00000000..319d435d --- /dev/null +++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/modules/eos_bgp_global.py @@ -0,0 +1,2312 @@ +#!/usr/bin/python +# -*- coding: utf-8 -*- +# Copyright 2020 Red Hat +# GNU General Public License v3.0+ +# (see COPYING or https://www.gnu.org/licenses/gpl-3.0.txt) + +############################################# +# WARNING # +############################################# +# +# This file is auto generated by the resource +# module builder playbook. +# +# Do not edit this file manually. +# +# Changes to this file will be over written +# by the resource module builder. +# +# Changes should be made in the model used to +# generate this file or in the resource module +# builder template. +# +############################################# + +""" +The module file for eos_bgp_global +""" + +from __future__ import absolute_import, division, print_function + +__metaclass__ = type + +DOCUMENTATION = """ +module: eos_bgp_global +short_description: Manages BGP global resource module +description: This module configures and manages the attributes of BGP global on Arista + EOS platforms. +version_added: 1.4.0 +author: Gomathi Selvi Srinivasan (@GomathiselviS) +notes: +- Tested against Arista EOS 4.23.0F +- This module works with connection C(network_cli). See the L(EOS Platform Options,eos_platform_options). +options: + config: + description: A list of configurations for BGP global. + type: dict + suboptions: + as_number: + description: Autonomous system number. + type: str + aggregate_address: + description: Configure aggregate address. + type: list + elements: dict + suboptions: + address: + description: ipv4/ipv6 address prefix. + type: str + advertise_only: + description: Advertise without installing the generated blackhole route in + FIB. + type: bool + as_set: + description: Generate autonomous system set path information. + type: bool + attribute_map: + description: Name of the route map used to set the attribute of the + aggregate route. + type: str + match_map: + description: Name of the route map used to filter the contributors of the + aggregate route. + type: str + summary_only: + description: Filters all more-specific routes from updates. + type: bool + bgp_params: + description: BGP parameters. + type: dict + suboptions: + additional_paths: + description: BGP additional-paths commands + type: str + choices: ['install', 'send', 'receive'] + advertise_inactive: + description: Advertise BGP routes even if they are inactive in RIB. + type: bool + allowas_in: + description: Allow local-as in updates. + type: dict + suboptions: + set: + description: When True, it is set. + type: bool + count: + description: Number of local ASNs allowed in a BGP update. + type: int + always_compare_med: + description: BGP Always Compare MED + type: bool + asn: + description: AS Number notation. + type: str + choices: ['asdot', 'asplain'] + auto_local_addr: + description: Automatically determine the local address to be used + for the non-transport AF. + type: bool + bestpath: + description: Select the bestpath selection algorithim for BGP routes. + type: dict + suboptions: + as_path: + description: Select the bestpath selection based on as-path. + type: str + choices: ['ignore', 'multipath_relax'] + ecmp_fast: + description: Tie-break BGP paths in a ECMP group based on the order of arrival. + type: bool + med: + description: MED attribute + type: dict + suboptions: + confed: + description: MED Confed. + type: bool + missing_as_worst: + description: MED missing-as-worst. + type: bool + skip: + description: skip one of the tie breaking rules in the bestpath selection. + type: bool + tie_break: + description: Configure the tie-break option for BGP bestpath selection. + choices: ['cluster_list_length', 'router_id'] + type: str + client_to_client: + description: client to client configuration. + type: bool + cluster_id: + description: Cluster ID of this router acting as a route reflector. + type: str + confederation: + description: confederation. + type: dict + suboptions: + identifier: + description: Confederation identifier. + type: str + peers: + description: Confederation peers. + type: str + control_plan_filter: + description: Control plane filter for BGP. + type: bool + convergence: + description: Bgp convergence parameters. + type: dict + suboptions: + slow_peer: + description: Maximum amount of time to wait for slow peers to estabilsh session. + type: bool + time: + description: time in secs + type: int + default: + description: Default neighbor configuration commands. + type: str + choices: ['ipv4_unicast', 'ipv6_unicast'] + enforce_first_as: + description: Enforce the First AS for EBGP routes(default). + type: bool + host_routes: + description: BGP host routes configuration. + type: bool + labeled_unicast: + description: Labeled Unicast. + type: str + choices: ['ip', 'tunnel'] + listen: + description: BGP listen. + type: dict + suboptions: + limit: + description: Set limit on the number of dynamic BGP peers allowed. + type: int + range: + description: Subnet Range to be associated with the peer-group. + type: dict + suboptions: + address: + description: Address prefix + type: str + peer_group: + description: Name of peer group. + type: dict + suboptions: + name: + description: name. + type: str + peer_filter: + description: Name of peer filter. + type: str + remote_as: + description: Neighbor AS number + type: str + log_neighbor_changes: + description: Log neighbor up/down events. + type: bool + missing_policy: + description: Missing policy override configuration commands. + type: dict + suboptions: + direction: + description: Missing policy direction options. + type: str + choices: ['in', 'out'] + action: + description: Missing policy action options. + type: str + choices: ['deny', 'permit', 'deny-in-out'] + monitoring: + description: Enable Bgp monitoring for all/specified stations. + type: bool + next_hop_unchanged: + description: Preserve original nexthop while advertising routes to + eBGP peers. + type: bool + redistribute_internal: + description: Redistribute internal BGP routes. + type: bool + route: + description: Configure route-map for route installation. + type: str + route_reflector: + description: Configure route reflector options + type: dict + suboptions: + set: + description: When True route_reflector is set. + type: bool + preserve: + description: preserve route attributes, overwriting route-map changes + type: bool + transport: + description: Configure transport port for TCP session + type: int + default_metric: + description: Default metric. + type: int + distance: + description: Define an administrative distance. + type: dict + suboptions: + external: + description: distance for external routes. + type: int + internal: + description: distance for internal routes. + type: int + local: + description: distance for local routes. + type: int + graceful_restart: + description: Enable graceful restart mode. + type: dict + suboptions: + set: + description: When True, graceful restart is set. + type: bool + restart_time: + description: Set the max time needed to restart and come back up. + type: int + stalepath_time: + description: Set the max time to hold onto restarting peer stale paths. + type: int + graceful_restart_helper: + description: Enable graceful restart helper mode. + type: bool + access_group: + description: ip/ipv6 access list configuration. + type: dict + suboptions: + afi: + description: Specify ip/ipv6. + type: str + choices: ['ipv4', 'ipv6'] + acl_name: + description: access list name. + type: str + direction: + description: direction of packets. + type: str + maximum_paths: + description: Maximum number of equal cost paths. + type: dict + suboptions: + max_equal_cost_paths: + description: Value for maximum number of equal cost paths. + type: int + max_installed_ecmp_paths: + description: Value for maximum number of installed ECMP routes. + type: int + monitoring: + description: BGP monitoring protocol configuration. + type: dict + suboptions: + port: + description: Configure the BGP monitoring protocol port number <1024-65535>. + type: int + received: + description: BGP monitoring protocol received route selection. + type: str + choices: ['post_policy', 'pre_policy'] + station: + description: BGP monitoring station configuration. + type: str + timestamp: + description: BGP monitoring protocol Per-Peer Header timestamp behavior. + type: str + choices: ['none', 'send_time'] + neighbor: + description: Configure routing for a network. + type: list + elements: dict + suboptions: + peer: + type: str + description: Neighbor address or peer-group. + additional_paths: + description: BGP additional-paths commands. + type: str + choices: ['send', 'receive'] + allowas_in: + description: Allow local-as in updates. + type: dict + suboptions: + set: + description: When True, it is set. + type: bool + count: + description: Number of local ASNs allowed in a BGP update. + type: int + auto_local_addr: + description: Automatically determine the local address to be used + for the non-transport AF. + type: bool + default_originate: + description: Originate default route to this neighbor. + type: dict + suboptions: + route_map: + description: Route map reference. + type: str + always: + description: Always originate default route to this neighbor. + type: bool + description: + description: Text describing the neighbor. + type: str + dont_capability_negotiate: + description: Donot perform Capability Negotiation with this + neighbor. + type: bool + ebgp_multihop: + description: Allow BGP connections to indirectly connected + external peers. + type: dict + suboptions: + ttl: + description: Time-to-live in the range 1-255 hops. + type: int + set: + description: If True, ttl is not set. + type: bool + enforce_first_as: + description: Enforce the First AS for EBGP routes(default). + type: bool + export_localpref: + description: Override localpref when exporting to an internal + peer. + type: int + fall_over: + description: Configure BFD protocol options for this peer. + type: bool + graceful_restart: + description: Enable graceful restart mode. + type: bool + graceful_restart_helper: + description: Enable graceful restart helper mode. + type: bool + idle_restart_timer: + description: Neighbor idle restart timer. + type: int + import_localpref: + description: Override localpref when importing from an external + peer. + type: int + link_bandwidth: + description: Enable link bandwidth community for routes to this + peer. + type: dict + suboptions: + set: + description: If True, set link bandwidth + type: bool + auto: + description: Enable link bandwidth auto generation for routes from this peer. + type: bool + default: + description: Enable link bandwidth default generation for routes from this + peer. + type: str + update_delay: + description: Delay outbound route updates. + type: int + local_as: + description: Configure local AS number advertised to peer. + type: dict + suboptions: + as_number: + description: AS number. + type: str + fallback: + description: Prefer router AS Number over local AS Number. + type: bool + local_v6_addr: + description: The local IPv6 address of the neighbor in A:B:C:D:E:F:G:H format. + type: str + maximum_accepted_routes: + description: Maximum number of routes accepted from this peer. + type: dict + suboptions: + count: + description: Maximum number of accepted routes (0 means unlimited). + type: int + warning_limit: + description: Maximum number of accepted routes after which a warning is issued. + (0 means never warn) + type: int + maximum_received_routes: + description: Maximum number of routes received from this peer. + type: dict + suboptions: + count: + description: Maximum number of routes (0 means unlimited). + type: int + warning_limit: + description: Percentage of maximum-routes at which warning is to be issued. + type: dict + suboptions: + limit_count: + description: Number of routes at which to warn. + type: int + limit_percent: + description: Percentage of maximum number of routes at which to warn( 1-100). + type: int + warning_only: + description: Only warn, no restart, if max route limit exceeded. + type: bool + metric_out: + description: MED value to advertise to peer. + type: int + monitoring: + description: Enable BGP Monitoring Protocol for this peer. + type: bool + next_hop_self: + description: Always advertise this router address as the BGP + next hop + type: bool + next_hop_unchanged: + description: Preserve original nexthop while advertising routes to + eBGP peers. + type: bool + next_hop_v6_address: + description: IPv6 next-hop address for the neighbor + type: str + out_delay: + description: Delay outbound route updates. + type: int + encryption_password: + description: Password to use in computation of MD5 hash. + type: dict + suboptions: + type: + description: Encryption type. + type: int + choices: [0, 7] + password: + description: password (up to 80 chars). + type: str + remote_as: + description: Neighbor Autonomous System. + type: str + remove_private_as: + description: Remove private AS number from updates to this peer. + type: dict + suboptions: + set: + description: If True, set remove_private_as. + type: bool + all: + description: Remove private AS number. + type: bool + replace_as: + description: Replace private AS number with local AS number. + type: bool + peer_group: + description: Name of the peer-group. + type: str + + prefix_list: + description: Prefix list reference. + type: dict + suboptions: + direction: + description: Configure an inbound/outbound prefix-list. + type: str + choices: ['in', 'out'] + name: + description: prefix list name. + type: str + route_map: + description: Route map reference. + type: dict + suboptions: + direction: + description: Configure an inbound/outbound route-map. + type: str + choices: ['in', 'out'] + name: + description: Route map name. + type: str + route_reflector_client: + description: Configure peer as a route reflector client. + type: bool + route_to_peer: + description: Use routing table information to reach the peer. + type: bool + send_community: + description: Send community attribute to this neighbor. + type: dict + suboptions: + community_attribute: + description: Type of community attributes to send to this neighbor. + type: str + sub_attribute: + description: Attribute to be sent to the neighbor. + type: str + choices: ['extended', 'link-bandwidth', 'standard'] + link_bandwidth_attribute: + description: cumulative/aggregate attribute to be sent. + type: str + choices: ['aggregate', 'divide'] + speed: + description: Reference link speed in bits/second + type: str + divide: + description: link-bandwidth divide attribute. + type: str + choices: ['equal', 'ratio'] + shut_down: + description: Administratively shut down this neighbor. + type: bool + soft_recognition: + description: Configure how to handle routes that fail import. + type: str + choices: ['all', 'None'] + timers: + description: Timers. + type: dict + suboptions: + keepalive: + description: Keep Alive Interval in secs. + type: int + holdtime: + description: Hold time in secs. + type: int + transport: + description: Configure transport options for TCP session. + type: dict + suboptions: + connection_mode: + description: Configure connection-mode for TCP session. + type: str + remote_port: + description: Configure BGP peer TCP port to connect to. + type: int + ttl: + description: BGP ttl security check + type: int + update_source: + description: Specify the local source interface for peer BGP + sessions. + type: str + weight: + description: Weight to assign. + type: int + network: + description: Configure routing for a network. + type: list + elements: dict + suboptions: + address: + description: address prefix. + type: str + route_map: + description: Name of route map. + type: str + redistribute: + description: Redistribute routes in to BGP. + type: list + elements: dict + suboptions: + protocol: + description: Routes to be redistributed. + type: str + choices: ['isis', 'ospf3', 'ospf', 'attached-host', 'connected', 'rip', 'static'] + route_map: + description: Route map reference. + type: str + isis_level: + description: Applicable for isis routes. Specify isis route level. + type: str + choices: ['level-1', 'level-2', 'level-1-2'] + ospf_route: + description: ospf route options. + type: str + choices: ['internal', 'external', 'nssa_external_1', 'nssa_external_2'] + router_id: + description: Router id. + type: str + route_target: + description: Route target. + type: dict + suboptions: + action: + description: Route action. + type: str + choices: ['both', 'import', 'export'] + target: + description: Route Target. + type: str + shutdown: + description: When True, shut down BGP. + type: bool + timers: + description: Timers. + type: dict + suboptions: + keepalive: + description: Keep Alive Interval in secs. + type: int + holdtime: + description: Hold time in secs. + type: int + ucmp: + description: Configure unequal cost multipathing. + type: dict + suboptions: + fec: + description: Configure UCMP fec utilization threshold. + type: dict + suboptions: + trigger: + description: UCMP fec utilization too high threshold. + type: int + clear: + description: UCMP FEC utilization Clear thresholds. + type: int + link_bandwidth: + description: Configure link-bandwidth propagation delay. + type: dict + suboptions: + mode: + description: UCMP link bandwidth mode + type: str + choices: ['encoding_weighted', 'recursive'] + update_delay: + description: Link Bandwidth Advertisement delay. + type: int + mode: + description: UCMP mode. + type: dict + suboptions: + set: + description: If True, ucmp mode is set to 1. + type: bool + nexthops: + description: Value for total number UCMP nexthops. + type: int + update: + description: Configure BGP update generation. + type: dict + suboptions: + wait_for: + description: wait for options before converge or synchronize. + type: str + choices: ['wait_for_convergence', 'wait_install'] + batch_size: + description: batch size for FIB route acknowledgements. + type: int + vlan: + description: Configure MAC VRF BGP for single VLAN support. + type: int + vlan_aware_bundle: + description: Configure MAC VRF BGP for multiple VLAN support. + type: str + vrfs: + description: Configure BGP in a VRF. + type: list + elements: dict + suboptions: + vrf: + description: VRF name. + type: str + aggregate_address: + description: Configure aggregate address. + type: list + elements: dict + suboptions: + address: + description: ipv4/ipv6 address prefix. + type: str + advertise_only: + description: Advertise without installing the generated blackhole route in + FIB. + type: bool + as_set: + description: Generate autonomous system set path information. + type: bool + attribute_map: + description: Name of the route map used to set the attribute of the + aggregate route. + type: str + match_map: + description: Name of the route map used to filter the contributors of the + aggregate route. + type: str + summary_only: + description: Filters all more-specific routes from updates. + type: bool + bgp_params: + description: BGP parameters. + type: dict + suboptions: + additional_paths: + description: BGP additional-paths commands + type: str + choices: ['install', 'send', 'receive'] + advertise_inactive: + description: Advertise BGP routes even if they are inactive in RIB. + type: bool + allowas_in: + description: Allow local-as in updates. + type: dict + suboptions: + set: + description: When True, it is set. + type: bool + count: + description: Number of local ASNs allowed in a BGP update. + type: int + always_compare_med: + description: BGP Always Compare MED + type: bool + asn: + description: AS Number notation. + type: str + choices: ['asdot', 'asplain'] + auto_local_addr: + description: Automatically determine the local address to be used + for the non-transport AF. + type: bool + bestpath: + description: Select the bestpath selection algorithim for BGP routes. + type: dict + suboptions: + as_path: + description: Select the bestpath selection based on as-path. + type: str + choices: ['ignore', 'multipath_relax'] + ecmp_fast: + description: Tie-break BGP paths in a ECMP group based on the order of arrival. + type: bool + med: + description: MED attribute + type: dict + suboptions: + confed: + description: MED Confed. + type: bool + missing_as_worst: + description: MED missing-as-worst. + type: bool + skip: + description: skip one of the tie breaking rules in the bestpath selection. + type: bool + tie_break: + description: Configure the tie-break option for BGP bestpath selection. + choices: ['cluster_list_length', 'router_id'] + type: str + client_to_client: + description: client to client configuration. + type: bool + cluster_id: + description: Cluster ID of this router acting as a route reflector. + type: str + confederation: + description: confederation. + type: dict + suboptions: + identifier: + description: Confederation identifier. + type: str + peers: + description: Confederation peers. + type: str + control_plane_filter: + description: Control plane filter for BGP. + type: bool + convergence: + description: Bgp convergence parameters. + type: dict + suboptions: + slow_peer: + description: Maximum amount of time to wait for slow peers to estabilsh session. + type: bool + time: + description: time in secs + type: int + default: + description: Default neighbor configuration commands. + type: str + choices: ['ipv4_unicast', 'ipv6_unicast'] + enforce_first_as: + description: Enforce the First AS for EBGP routes(default). + type: bool + host_routes: + description: BGP host routes configuration. + type: bool + labeled_unicast: + description: Labeled Unicast. + type: str + choices: ['ip', 'tunnel'] + listen: + description: BGP listen. + type: dict + suboptions: + limit: + description: Set limit on the number of dynamic BGP peers allowed. + type: int + range: + description: Subnet Range to be associated with the peer-group. + type: dict + suboptions: + address: + description: Address prefix + type: str + peer_group: + description: Name of peer group. + type: dict + suboptions: + name: + description: name. + type: str + peer_filter: + description: Name of peer filter. + type: str + remote_as: + description: Neighbor AS number + type: str + log_neighbor_changes: + description: Log neighbor up/down events. + type: bool + missing_policy: + description: Missing policy override configuration commands. + type: dict + suboptions: + direction: + description: Missing policy direction options. + type: str + choices: ['in', 'out'] + action: + description: Missing policy action options. + type: str + choices: ['deny', 'permit', 'deny-in-out'] + monitoring: + description: Enable Bgp monitoring for all/specified stations. + type: bool + next_hop_unchanged: + description: Preserve original nexthop while advertising routes to + eBGP peers. + type: bool + redistribute_internal: + description: Redistribute internal BGP routes. + type: bool + route: + description: Configure route-map for route installation. + type: str + route_reflector: + description: Configure route reflector options + type: dict + suboptions: + set: + description: When True route_reflector is set. + type: bool + preserve: + description: preserve route attributes, overwriting route-map changes + type: bool + transport: + description: Configure transport port for TCP session + type: int + default_metric: + description: Default metric. + type: int + distance: + description: Define an administrative distance. + type: dict + suboptions: + external: + description: distance for external routes. + type: int + internal: + description: distance for internal routes. + type: int + local: + description: distance for local routes. + type: int + graceful_restart: + description: Enable graceful restart mode. + type: dict + suboptions: + set: + description: When True, graceful restart is set. + type: bool + restart_time: + description: Set the max time needed to restart and come back up. + type: int + stalepath_time: + description: Set the max time to hold onto restarting peer stale paths. + type: int + graceful_restart_helper: + description: Enable graceful restart helper mode. + type: bool + access_group: + description: ip/ipv6 access list configuration. + type: dict + suboptions: + afi: + description: Specify ip/ipv6. + type: str + choices: ['ip', 'ipv6'] + acl_name: + description: access list name. + type: str + direction: + description: direction of packets. + type: str + maximum_paths: + description: Maximum number of equal cost paths. + type: dict + suboptions: + max_equal_cost_paths: + description: Value for maximum number of equal cost paths. + type: int + max_installed_ecmp_paths: + description: Value for maximum number of installed ECMP routes. + type: int + neighbor: + description: Configure routing for a network. + type: list + elements: dict + suboptions: + peer: + type: str + description: Neighbor address or peer group. + additional_paths: + description: BGP additional-paths commands. + type: str + choices: ['send', 'receive'] + allowas_in: + description: Allow local-as in updates. + type: dict + suboptions: + set: + description: When True, it is set. + type: bool + count: + description: Number of local ASNs allowed in a BGP update. + type: int + auto_local_addr: + description: Automatically determine the local address to be used + for the non-transport AF. + type: bool + default_originate: + description: Originate default route to this neighbor. + type: dict + suboptions: + route_map: + description: Route map reference. + type: str + always: + description: Always originate default route to this neighbor. + type: bool + description: + description: Text describing the neighbor. + type: str + dont_capability_negotiate: + description: Donot perform Capability Negotiation with this + neighbor. + type: bool + ebgp_multihop: + description: Allow BGP connections to indirectly connected + external peers. + type: dict + suboptions: + ttl: + description: Time-to-live in the range 1-255 hops. + type: int + set: + description: If True, ttl is not set. + type: bool + enforce_first_as: + description: Enforce the First AS for EBGP routes(default). + type: bool + export_localpref: + description: Override localpref when exporting to an internal + peer. + type: int + fall_over: + description: Configure BFD protocol options for this peer. + type: bool + graceful_restart: + description: Enable graceful restart mode. + type: bool + graceful_restart_helper: + description: Enable graceful restart helper mode. + type: bool + idle_restart_timer: + description: Neighbor idle restart timer. + type: int + import_localpref: + description: Override localpref when importing from an external + peer. + type: int + link_bandwidth: + description: Enable link bandwidth community for routes to this + peer. + type: dict + suboptions: + set: + description: If True, set link bandwidth + type: bool + auto: + description: Enable link bandwidth auto generation for routes from this peer. + type: bool + default: + description: Enable link bandwidth default generation for routes from this + peer. + type: str + update_delay: + description: Delay outbound route updates. + type: int + local_as: + description: Configure local AS number advertised to peer. + type: dict + suboptions: + as_number: + description: AS number. + type: str + fallback: + description: Prefer router AS Number over local AS Number. + type: bool + local_v6_addr: + description: The local IPv6 address of the neighbor in A:B:C:D:E:F:G:H format. + type: str + maximum_accepted_routes: + description: Maximum number of routes accepted from this peer. + type: dict + suboptions: + count: + description: Maximum number of accepted routes (0 means unlimited). + type: int + warning_limit: + description: Maximum number of accepted routes after which a warning is issued. + (0 means never warn) + type: int + maximum_received_routes: + description: Maximum number of routes received from this peer. + type: dict + suboptions: + count: + description: Maximum number of routes (0 means unlimited). + type: int + warning_limit: + description: Percentage of maximum-routes at which warning is to be issued. + type: dict + suboptions: + limit_count: + description: Number of routes at which to warn. + type: int + limit_percent: + description: Percentage of maximum number of routes at which to warn( 1-100). + type: int + warning_only: + description: Only warn, no restart, if max route limit exceeded. + type: bool + metric_out: + description: MED value to advertise to peer. + type: int + monitoring: + description: Enable BGP Monitoring Protocol for this peer. + type: bool + next_hop_self: + description: Always advertise this router address as the BGP + next hop + type: bool + next_hop_unchanged: + description: Preserve original nexthop while advertising routes to + eBGP peers. + type: bool + next_hop_v6_address: + description: IPv6 next-hop address for the neighbor + type: str + out_delay: + description: Delay outbound route updates. + type: int + encryption_password: + description: Password to use in computation of MD5 hash. + type: dict + suboptions: + type: + description: Encryption type. + type: int + choices: [0, 7] + password: + description: password (up to 80 chars). + type: str + remote_as: + description: Neighbor Autonomous System. + type: str + remove_private_as: + description: Remove private AS number from updates to this peer. + type: dict + suboptions: + set: + description: If True, set remove_private_as. + type: bool + all: + description: Remove private AS number. + type: bool + replace_as: + description: Replace private AS number with local AS number. + type: bool + peer_group: + description: Name of the peer-group. + type: str + + prefix_list: + description: Prefix list reference. + type: dict + suboptions: + direction: + description: Configure an inbound/outbound prefix-list. + type: str + choices: ['in', 'out'] + name: + description: prefix list name. + type: str + route_map: + description: Route map reference. + type: dict + suboptions: + direction: + description: Configure an inbound/outbound route-map. + type: str + choices: ['in', 'out'] + name: + description: Route map name. + type: str + route_reflector_client: + description: Configure peer as a route reflector client. + type: bool + route_to_peer: + description: Use routing table information to reach the peer. + type: bool + send_community: + description: Send community attribute to this neighbor. + type: dict + suboptions: + community_attribute: + description: Type of community attributes to send to this neighbor. + type: str + sub_attribute: + description: Attribute to be sent to the neighbor. + type: str + choices: ['extended', 'link-bandwidth', 'standard'] + link_bandwidth_attribute: + description: cumulative/aggregate attribute to be sent. + type: str + choices: ['aggregate', 'divide'] + speed: + description: Reference link speed in bits/second + type: str + divide: + description: link-bandwidth divide attribute. + type: str + choices: ['equal', 'ratio'] + shut_down: + description: Administratively shut down this neighbor. + type: bool + soft_recognition: + description: Configure how to handle routes that fail import. + type: str + choices: ['all', 'None'] + timers: + description: Timers. + type: dict + suboptions: + keepalive: + description: Keep Alive Interval in secs. + type: int + holdtime: + description: Hold time in secs. + type: int + transport: + description: Configure transport options for TCP session. + type: dict + suboptions: + connection_mode: + description: Configure connection-mode for TCP session. + type: str + remote_port: + description: Configure BGP peer TCP port to connect to. + type: int + ttl: + description: BGP ttl security check + type: int + update_source: + description: Specify the local source interface for peer BGP + sessions. + type: str + weight: + description: Weight to assign. + type: int + network: + description: Configure routing for a network. + type: list + elements: dict + suboptions: + address: + description: address prefix. + type: str + route_map: + description: Name of route map. + type: str + redistribute: + description: Redistribute routes in to BGP. + type: list + elements: dict + suboptions: + protocol: + description: Routes to be redistributed. + type: str + choices: ['isis', 'ospf3', 'ospf', 'attached-host', 'connected', 'rip', 'static'] + route_map: + description: Route map reference. + type: str + isis_level: + description: Applicable for isis routes. Specify isis route level. + type: str + choices: ['level-1', 'level-2', 'level-1-2'] + ospf_route: + description: ospf route options. + type: str + choices: ['internal', 'external', 'nssa_external_1', 'nssa_external_2'] + route_target: + description: Route target. + type: dict + suboptions: + action: + description: Route action. + type: str + choices: ['both', 'import', 'export'] + target: + description: Route Target. + type: str + router_id: + description: Router id. + type: str + shutdown: + description: When True, shut down BGP. + type: bool + timers: + description: Timers. + type: dict + suboptions: + keepalive: + description: Keep Alive Interval in secs. + type: int + holdtime: + description: Hold time in secs. + type: int + ucmp: + description: Configure unequal cost multipathing. + type: dict + suboptions: + fec: + description: Configure UCMP fec utilization threshold. + type: dict + suboptions: + trigger: + description: UCMP fec utilization too high threshold. + type: int + clear: + description: UCMP FEC utilization Clear thresholds. + type: int + link_bandwidth: + description: Configure link-bandwidth propagation delay. + type: dict + suboptions: + mode: + description: UCMP link bandwidth mode + type: str + choices: ['encoding_weighted', 'recursive'] + update_delay: + description: Link Bandwidth Advertisement delay. + type: int + mode: + description: UCMP mode. + type: dict + suboptions: + set: + description: If True, ucmp mode is set to 1. + type: bool + nexthops: + description: Value for total number UCMP nexthops. + type: int + update: + description: Configure BGP update generation. + type: dict + suboptions: + wait_for: + description: wait for options before converge or synchronize. + type: str + choices: ['wait_for_convergence', 'wait_install'] + batch_size: + description: batch size for FIB route acknowledgements. + type: int + running_config: + description: + - This option is used only with state I(parsed). + - The value of this option should be the output received from the EOS device by + executing the command B(show running-config | section bgp). + - The state I(parsed) reads the configuration from C(running_config) option and + transforms it into Ansible structured data as per the resource module's argspec + and the value is then returned in the I(parsed) key within the result. + type: str + state: + description: + - The state the configuration should be left in. + - State I(purged) removes all the BGP configurations from the + target device. Use caution with this state.('no router bgp <x>') + - State I(deleted) only removes BGP attributes that this modules + manages and does not negate the BGP process completely. Thereby, preserving + address-family related configurations under BGP context. + - Running states I(deleted) and I(replaced) will result in an error if there + are address-family configuration lines present under vrf context that is + is to be removed. Please use the M(arista.eos.eos_bgp_address_family) + module for prior cleanup. + - Refer to examples for more details. + type: str + choices: [deleted, merged, purged, replaced, gathered, rendered, parsed] + default: merged +""" +EXAMPLES = """ +# Using merged +# Before state + +# veos(config)#show running-config | section bgp +# veos(config)# + + - name: Merge provided configuration with device configuration + arista.eos.eos_bgp_global: + config: + as_number: "100" + bgp_params: + host_routes: True + convergence: + slow_peer: True + time: 6 + additional_paths: "send" + log_neighbor_changes: True + maximum_paths: + max_equal_cost_paths: 55 + aggregate_address: + - address: "1.2.1.0/24" + as_set: true + match_map: "match01" + - address: "5.2.1.0/24" + attribute_map: "attrmatch01" + advertise_only: true + redistribute: + - protocol: "static" + route_map: "map_static" + - protocol: "attached-host" + distance: + internal: 50 + neighbor: + - peer: "10.1.3.2" + allowas_in: + set: true + default_originate: + always: true + dont_capability_negotiate: true + export_localpref: 4000 + maximum_received_routes: + count: 500 + warning_limit: + limit_percent: 5 + next_hop_unchanged: true + prefix_list: + name: "prefix01" + direction: "out" + - peer: "peer1" + fall_over: true + link_bandwidth: + update_delay: 5 + monitoring: True + send_community: + community_attribute: "extended" + sub_attribute: "link-bandwidth" + link_bandwidth_attribute: "aggregate" + speed: "600" + vlan: 5 + state: merged + +# After State: +# veos(config)#show running-config | section bgp +# router bgp 100 +# bgp convergence slow-peer time 6 +# distance bgp 50 50 50 +# maximum-paths 55 +# bgp additional-paths send any +# neighbor peer1 peer-group +# neighbor peer1 link-bandwidth update-delay 5 +# neighbor peer1 fall-over bfd +# neighbor peer1 monitoring +# neighbor peer1 send-community extended link-bandwidth aggregate 600 +# neighbor peer1 maximum-routes 12000 +# neighbor 10.1.3.2 export-localpref 4000 +# neighbor 10.1.3.2 next-hop-unchanged +# neighbor 10.1.3.2 dont-capability-negotiate +# neighbor 10.1.3.2 allowas-in 3 +# neighbor 10.1.3.2 default-originate always +# neighbor 10.1.3.2 maximum-routes 500 warning-limit 5 percent +# aggregate-address 1.2.1.0/24 as-set match-map match01 +# aggregate-address 5.2.1.0/24 attribute-map attrmatch01 advertise-only +# redistribute static route-map map_static +# redistribute attached-host +# ! +# vlan 5 +# ! +# address-family ipv4 +# neighbor 10.1.3.2 prefix-list prefix01 out +# veos(config)# +# +# Module Execution: +# +# "after": { +# "aggregate_address": [ +# { +# "address": "1.2.1.0/24", +# "as_set": true, +# "match_map": "match01" +# }, +# { +# "address": "5.2.1.0/24", +# "advertise_only": true, +# "attribute_map": "attrmatch01" +# } +# ], +# "as_number": "100", +# "bgp_params": { +# "additional_paths": "send", +# "convergence": { +# "slow_peer": true, +# "time": 6 +# } +# }, +# "distance": { +# "external": 50, +# "internal": 50, +# "local": 50 +# }, +# "maximum_paths": { +# "max_equal_cost_paths": 55 +# }, +# "neighbor": [ +# { +# "fall_over": true, +# "link_bandwidth": { +# "set": true, +# "update_delay": 5 +# }, +# "maximum_received_routes": { +# "count": 12000 +# }, +# "monitoring": true, +# "peer": "peer1", +# "peer_group": "peer1", +# "send_community": { +# "community_attribute": "extended", +# "link_bandwidth_attribute": "aggregate", +# "speed": "600", +# "sub_attribute": "link-bandwidth" +# } +# }, +# { +# "allowas_in": { +# "count": 3 +# }, +# "default_originate": { +# "always": true +# }, +# "dont_capability_negotiate": true, +# "export_localpref": 4000, +# "maximum_received_routes": { +# "count": 500, +# "warning_limit": { +# "limit_percent": 5 +# } +# }, +# "next_hop_unchanged": true, +# "peer": "10.1.3.2" +# } +# ], +# "redistribute": [ +# { +# "protocol": "static", +# "route_map": "map_static" +# }, +# { +# "protocol": "attached-host" +# } +# ], +# "vlan": 5 +# }, +# "before": {}, +# "changed": true, +# "commands": [ +# "router bgp 100", +# "neighbor 10.1.3.2 allowas-in", +# "neighbor 10.1.3.2 default-originate always", +# "neighbor 10.1.3.2 dont-capability-negotiate", +# "neighbor 10.1.3.2 export-localpref 4000", +# "neighbor 10.1.3.2 maximum-routes 500 warning-limit 5 percent", +# "neighbor 10.1.3.2 next-hop-unchanged", +# "neighbor 10.1.3.2 prefix-list prefix01 out", +# "neighbor peer1 fall-over bfd", +# "neighbor peer1 link-bandwidth update-delay 5", +# "neighbor peer1 monitoring", +# "neighbor peer1 send-community extended link-bandwidth aggregate 600", +# "redistribute static route-map map_static", +# "redistribute attached-host", +# "aggregate-address 1.2.1.0/24 as-set match-map match01", +# "aggregate-address 5.2.1.0/24 attribute-map attrmatch01 advertise-only", +# "bgp host-routes fib direct-install", +# "bgp convergence slow-peer time 6", +# "bgp additional-paths send any", +# "bgp log-neighbor-changes", +# "maximum-paths 55", +# "distance bgp 50", +# "vlan 5" +# ], + +# Using replaced: + +# Before state: +# veos(config)#show running-config | section bgp +# router bgp 100 +# bgp convergence slow-peer time 6 +# distance bgp 50 50 50 +# maximum-paths 55 +# bgp additional-paths send any +# neighbor peer1 peer-group +# neighbor peer1 link-bandwidth update-delay 5 +# neighbor peer1 fall-over bfd +# neighbor peer1 monitoring +# neighbor peer1 send-community extended link-bandwidth aggregate 600 +# neighbor peer1 maximum-routes 12000 +# neighbor 10.1.3.2 export-localpref 4000 +# neighbor 10.1.3.2 next-hop-unchanged +# neighbor 10.1.3.2 dont-capability-negotiate +# neighbor 10.1.3.2 allowas-in 3 +# neighbor 10.1.3.2 default-originate always +# neighbor 10.1.3.2 maximum-routes 500 warning-limit 5 percent +# aggregate-address 1.2.1.0/24 as-set match-map match01 +# aggregate-address 5.2.1.0/24 attribute-map attrmatch01 advertise-only +# redistribute static route-map map_static +# redistribute attached-host +# ! +# vlan 5 +# ! +# address-family ipv4 +# neighbor 10.1.3.2 prefix-list prefix01 out +# ! +# vrf vrf01 +# route-target import 54:11 +# neighbor 12.1.3.2 dont-capability-negotiate +# neighbor 12.1.3.2 allowas-in 3 +# neighbor 12.1.3.2 default-originate always +# neighbor 12.1.3.2 maximum-routes 12000 +# veos(config)# + + - name: replace provided configuration with device configuration + arista.eos.eos_bgp_global: + config: + as_number: "100" + bgp_params: + host_routes: True + convergence: + slow_peer: True + time: 6 + additional_paths: "send" + log_neighbor_changes: True + vrfs: + - vrf: "vrf01" + maximum_paths: + max_equal_cost_paths: 55 + aggregate_address: + - address: "1.2.1.0/24" + as_set: true + match_map: "match01" + - address: "5.2.1.0/24" + attribute_map: "attrmatch01" + advertise_only: true + redistribute: + - protocol: "static" + route_map: "map_static" + - protocol: "attached-host" + distance: + internal: 50 + neighbor: + - peer: "10.1.3.2" + allowas_in: + set: true + default_originate: + always: true + dont_capability_negotiate: true + export_localpref: 4000 + maximum_received_routes: + count: 500 + warning_limit: + limit_percent: 5 + next_hop_unchanged: true + prefix_list: + name: "prefix01" + direction: "out" + - peer: "peer1" + fall_over: true + link_bandwidth: + update_delay: 5 + monitoring: True + send_community: + community_attribute: "extended" + sub_attribute: "link-bandwidth" + link_bandwidth_attribute: "aggregate" + speed: "600" + state: replaced + +# After State: + +# veos(config)#show running-config | section bgp +# router bgp 100 +# bgp convergence slow-peer time 6 +# bgp additional-paths send any +# ! +# vrf vrf01 +# distance bgp 50 50 50 +# maximum-paths 55 +# neighbor 10.1.3.2 export-localpref 4000 +# neighbor 10.1.3.2 next-hop-unchanged +# neighbor 10.1.3.2 dont-capability-negotiate +# neighbor 10.1.3.2 allowas-in 3 +# neighbor 10.1.3.2 default-originate always +# neighbor 10.1.3.2 maximum-routes 500 warning-limit 5 percent +# aggregate-address 1.2.1.0/24 as-set match-map match01 +# aggregate-address 5.2.1.0/24 attribute-map attrmatch01 advertise-only +# redistribute static route-map map_static +# redistribute attached-host +# ! +# address-family ipv4 +# neighbor 10.1.3.2 prefix-list prefix01 out +# veos(config)# +# +# +# Module Execution: +# +# "after": { +# "as_number": "100", +# "bgp_params": { +# "additional_paths": "send", +# "convergence": { +# "slow_peer": true, +# "time": 6 +# } +# }, +# "vrfs": [ +# { +# "aggregate_address": [ +# { +# "address": "1.2.1.0/24", +# "as_set": true, +# "match_map": "match01" +# }, +# { +# "address": "5.2.1.0/24", +# "advertise_only": true, +# "attribute_map": "attrmatch01" +# } +# ], +# "distance": { +# "external": 50, +# "internal": 50, +# "local": 50 +# }, +# "maximum_paths": { +# "max_equal_cost_paths": 55 +# }, +# "neighbor": [ +# { +# "allowas_in": { +# "count": 3 +# }, +# "default_originate": { +# "always": true +# }, +# "dont_capability_negotiate": true, +# "export_localpref": 4000, +# "maximum_received_routes": { +# "count": 500, +# "warning_limit": { +# "limit_percent": 5 +# } +# }, +# "next_hop_unchanged": true, +# "peer": "10.1.3.2" +# } +# ], +# "redistribute": [ +# { +# "protocol": "static", +# "route_map": "map_static" +# }, +# { +# "protocol": "attached-host" +# } +# ], +# "vrf": "vrf01" +# } +# ] +# }, +# "before": { +# "aggregate_address": [ +# { +# "address": "1.2.1.0/24", +# "as_set": true, +# "match_map": "match01" +# }, +# { +# "address": "5.2.1.0/24", +# "advertise_only": true, +# "attribute_map": "attrmatch01" +# } +# ], +# "as_number": "100", +# "bgp_params": { +# "additional_paths": "send", +# "convergence": { +# "slow_peer": true, +# "time": 6 +# } +# }, +# "distance": { +# "external": 50, +# "internal": 50, +# "local": 50 +# }, +# "maximum_paths": { +# "max_equal_cost_paths": 55 +# }, +# "neighbor": [ +# { +# "fall_over": true, +# "link_bandwidth": { +# "set": true, +# "update_delay": 5 +# }, +# "maximum_received_routes": { +# "count": 12000 +# }, +# "monitoring": true, +# "peer": "peer1", +# "peer_group": "peer1", +# "send_community": { +# "community_attribute": "extended", +# "link_bandwidth_attribute": "aggregate", +# "speed": "600", +# "sub_attribute": "link-bandwidth" +# } +# }, +# { +# "allowas_in": { +# "count": 3 +# }, +# "default_originate": { +# "always": true +# }, +# "dont_capability_negotiate": true, +# "export_localpref": 4000, +# "maximum_received_routes": { +# "count": 500, +# "warning_limit": { +# "limit_percent": 5 +# } +# }, +# "next_hop_unchanged": true, +# "peer": "10.1.3.2" +# } +# ], +# "redistribute": [ +# { +# "protocol": "static", +# "route_map": "map_static" +# }, +# { +# "protocol": "attached-host" +# } +# ], +# "vlan": 5, +# "vrfs": [ +# { +# "neighbor": [ +# { +# "allowas_in": { +# "count": 3 +# }, +# "default_originate": { +# "always": true +# }, +# "dont_capability_negotiate": true, +# "maximum_received_routes": { +# "count": 12000 +# }, +# "peer": "12.1.3.2" +# } +# ], +# "route_target": { +# "action": "import", +# "target": "54:11" +# }, +# "vrf": "vrf01" +# } +# ] +# }, +# "changed": true, +# "commands": [ +# "router bgp 100", +# "vrf vrf01", +# "no route-target import 54:11", +# "neighbor 10.1.3.2 allowas-in", +# "neighbor 10.1.3.2 default-originate always", +# "neighbor 10.1.3.2 dont-capability-negotiate", +# "neighbor 10.1.3.2 export-localpref 4000", +# "neighbor 10.1.3.2 maximum-routes 500 warning-limit 5 percent", +# "neighbor 10.1.3.2 next-hop-unchanged", +# "neighbor 10.1.3.2 prefix-list prefix01 out", +# "neighbor peer1 fall-over bfd", +# "neighbor peer1 link-bandwidth update-delay 5", +# "neighbor peer1 monitoring", +# "neighbor peer1 send-community extended link-bandwidth aggregate 600", +# "no neighbor 12.1.3.2", +# "redistribute static route-map map_static", +# "redistribute attached-host", +# "aggregate-address 1.2.1.0/24 as-set match-map match01", +# "aggregate-address 5.2.1.0/24 attribute-map attrmatch01 advertise-only", +# "maximum-paths 55", +# "distance bgp 50", +# "exit", +# "no neighbor peer1 peer-group", +# "no neighbor peer1 link-bandwidth update-delay 5", +# "no neighbor peer1 fall-over bfd", +# "no neighbor peer1 monitoring", +# "no neighbor peer1 send-community extended link-bandwidth aggregate 600", +# "no neighbor peer1 maximum-routes 12000", +# "no neighbor 10.1.3.2", +# "no redistribute static route-map map_static", +# "no redistribute attached-host", +# "no aggregate-address 1.2.1.0/24 as-set match-map match01", +# "no aggregate-address 5.2.1.0/24 attribute-map attrmatch01 advertise-only", +# "bgp host-routes fib direct-install", +# "bgp log-neighbor-changes", +# "no distance bgp 50 50 50", +# "no maximum-paths 55", +# "no vlan 5" +# ], +# + +# Using replaced (in presence of address_family under vrf): +# Before State: + +#veos(config)#show running-config | section bgp +# router bgp 100 +# bgp convergence slow-peer time 6 +# bgp additional-paths send any +# ! +# vrf vrf01 +# distance bgp 50 50 50 +# maximum-paths 55 +# neighbor 10.1.3.2 export-localpref 4000 +# neighbor 10.1.3.2 next-hop-unchanged +# neighbor 10.1.3.2 dont-capability-negotiate +# neighbor 10.1.3.2 allowas-in 3 +# neighbor 10.1.3.2 default-originate always +# neighbor 10.1.3.2 maximum-routes 500 warning-limit 5 percent +# aggregate-address 1.2.1.0/24 as-set match-map match01 +# aggregate-address 5.2.1.0/24 attribute-map attrmatch01 advertise-only +# redistribute static route-map map_static +# redistribute attached-host +# ! +# address-family ipv4 +# neighbor 10.1.3.2 prefix-list prefix01 out +# ! +# address-family ipv6 +# redistribute dhcp +# veos(config)# + + - name: Replace + arista.eos.eos_bgp_global: + config: + as_number: "100" + graceful_restart: + set: True + router_id: "1.1.1.1" + timers: + keepalive: 2 + holdtime: 5 + ucmp: + mode: + set: True + vlan_aware_bundle: "bundle1 bundle2 bundle3" + state: replaced + +# Module Execution: + +# fatal: [192.168.122.113]: FAILED! => { +# "changed": false, +# "invocation": { +# "module_args": { +# "config": { +# "access_group": null, +# "aggregate_address": null, +# "as_number": "100", +# "bgp_params": null, +# "default_metric": null, +# "distance": null, +# "graceful_restart": { +# "restart_time": null, +# "set": true, +# "stalepath_time": null +# }, +# "graceful_restart_helper": null, +# "maximum_paths": null, +# "monitoring": null, +# "neighbor": null, +# "network": null, +# "redistribute": null, +# "route_target": null, +# "router_id": "1.1.1.1", +# "shutdown": null, +# "timers": { +# "holdtime": 5, +# "keepalive": 2 +# }, +# "ucmp": { +# "fec": null, +# "link_bandwidth": null, +# "mode": { +# "nexthops": null, +# "set": true +# } +# }, +# "update": null, +# "vlan": null, +# "vlan_aware_bundle": "bundle1 bundle2 bundle3", +# "vrfs": null +# }, +# "running_config": null, +# "state": "replaced" +# } +# }, +# "msg": "Use the _bgp_af module to delete the address_family under vrf, before replacing/deleting the vrf." +# } + +# Using deleted: + +# Before state: + +# veos(config)#show running-config | section bgp +# router bgp 100 +# bgp convergence slow-peer time 6 +# bgp additional-paths send any +# ! +# vrf vrf01 +# distance bgp 50 50 50 +# maximum-paths 55 +# neighbor 10.1.3.2 export-localpref 4000 +# neighbor 10.1.3.2 next-hop-unchanged +# neighbor 10.1.3.2 dont-capability-negotiate +# neighbor 10.1.3.2 allowas-in 3 +# neighbor 10.1.3.2 default-originate always +# neighbor 10.1.3.2 maximum-routes 500 warning-limit 5 percent +# aggregate-address 1.2.1.0/24 as-set match-map match01 +# aggregate-address 5.2.1.0/24 attribute-map attrmatch01 advertise-only +# redistribute static route-map map_static +# redistribute attached-host +# ! + + - name: Delete configuration + arista.eos.eos_bgp_global: + config: + as_number: "100" + state: deleted + +# After State: + +# veos(config)#show running-config | section bgp +# router bgp 100 +# +# +# Module Execution: +# +# "after": { +# "as_number": "100" +# }, +# "before": { +# "as_number": "100", +# "bgp_params": { +# "additional_paths": "send", +# "convergence": { +# "slow_peer": true, +# "time": 6 +# } +# }, +# "vrfs": [ +# { +# "aggregate_address": [ +# { +# "address": "1.2.1.0/24", +# "as_set": true, +# "match_map": "match01" +# }, +# { +# "address": "5.2.1.0/24", +# "advertise_only": true, +# "attribute_map": "attrmatch01" +# } +# ], +# "distance": { +# "external": 50, +# "internal": 50, +# "local": 50 +# }, +# "maximum_paths": { +# "max_equal_cost_paths": 55 +# }, +# "neighbor": [ +# { +# "allowas_in": { +# "count": 3 +# }, +# "default_originate": { +# "always": true +# }, +# "dont_capability_negotiate": true, +# "export_localpref": 4000, +# "maximum_received_routes": { +# "count": 500, +# "warning_limit": { +# "limit_percent": 5 +# } +# }, +# "next_hop_unchanged": true, +# "peer": "10.1.3.2" +# } +# ], +# "redistribute": [ +# { +# "protocol": "static", +# "route_map": "map_static" +# }, +# { +# "protocol": "attached-host" +# } +# ], +# "vrf": "vrf01" +# } +# ] +# }, +# "changed": true, +# "commands": [ +# "router bgp 100", +# "no vrf vrf01", +# "no bgp convergence slow-peer time 6", +# "no bgp additional-paths send any" +# ], +# + +# Using purged: + +# Before state: + +# veos(config)#show running-config | section bgp +# router bgp 100 +# bgp convergence slow-peer time 6 +# distance bgp 50 50 50 +# maximum-paths 55 +# bgp additional-paths send any +# neighbor peer1 peer-group +# neighbor peer1 link-bandwidth update-delay 5 +# neighbor peer1 fall-over bfd +# neighbor peer1 monitoring +# neighbor peer1 send-community extended link-bandwidth aggregate 600 +# neighbor peer1 maximum-routes 12000 +# neighbor 10.1.3.2 export-localpref 4000 +# neighbor 10.1.3.2 next-hop-unchanged +# neighbor 10.1.3.2 dont-capability-negotiate +# neighbor 10.1.3.2 allowas-in 3 +# neighbor 10.1.3.2 default-originate always +# neighbor 10.1.3.2 maximum-routes 500 warning-limit 5 percent +# aggregate-address 1.2.1.0/24 as-set match-map match01 +# aggregate-address 5.2.1.0/24 attribute-map attrmatch01 advertise-only +# redistribute static route-map map_static +# redistribute attached-host +# ! +# vlan 5 +# ! +# address-family ipv4 +# neighbor 10.1.3.2 prefix-list prefix01 out +# ! +# vrf vrf01 +# route-target import 54:11 +# neighbor 12.1.3.2 dont-capability-negotiate +# neighbor 12.1.3.2 allowas-in 3 +# neighbor 12.1.3.2 default-originate always +# neighbor 12.1.3.2 maximum-routes 12000 +# veos(config)# + + - name: Purge configuration + arista.eos.eos_bgp_global: + config: + as_number: "100" + state: purged + +# After State: + +# veos(config)#show running-config | section bgp +# veos(config)# + +# Module Execution: + +# "after": {}, +# "before": { +# "aggregate_address": [ +# { +# "address": "1.2.1.0/24", +# "as_set": true, +# "match_map": "match01" +# }, +# { +# "address": "5.2.1.0/24", +# "advertise_only": true, +# "attribute_map": "attrmatch01" +# } +# ], +# "as_number": "100", +# "bgp_params": { +# "additional_paths": "send", +# "convergence": { +# "slow_peer": true, +# "time": 6 +# } +# }, +# "distance": { +# "external": 50, +# "internal": 50, +# "local": 50 +# }, +# "maximum_paths": { +# "max_equal_cost_paths": 55 +# }, +# "neighbor": [ +# { +# "fall_over": true, +# "link_bandwidth": { +# "set": true, +# "update_delay": 5 +# }, +# "maximum_received_routes": { +# "count": 12000 +# }, +# "monitoring": true, +# "peer": "peer1", +# "peer_group": "peer1", +# "send_community": { +# "community_attribute": "extended", +# "link_bandwidth_attribute": "aggregate", +# "speed": "600", +# "sub_attribute": "link-bandwidth" +# } +# }, +# { +# "allowas_in": { +# "count": 3 +# }, +# "default_originate": { +# "always": true +# }, +# "dont_capability_negotiate": true, +# "export_localpref": 4000, +# "maximum_received_routes": { +# "count": 500, +# "warning_limit": { +# "limit_percent": 5 +# } +# }, +# "next_hop_unchanged": true, +# "peer": "10.1.3.2" +# } +# ], +# "redistribute": [ +# { +# "protocol": "static", +# "route_map": "map_static" +# }, +# { +# "protocol": "attached-host" +# } +# ], +# "vlan": 5, +# "vrfs": [ +# { +# "neighbor": [ +# { +# "allowas_in": { +# "count": 3 +# }, +# "default_originate": { +# "always": true +# }, +# "dont_capability_negotiate": true, +# "maximum_received_routes": { +# "count": 12000 +# }, +# "peer": "12.1.3.2" +# } +# ], +# "route_target": { +# "action": "import", +# "target": "54:11" +# }, +# "vrf": "vrf01" +# } +# ] +# }, +# "changed": true, +# "commands": [ +# "no router bgp 100" +# ], + +""" + +from ansible.module_utils.basic import AnsibleModule +from ansible_collections.arista.eos.plugins.module_utils.network.eos.argspec.bgp_global.bgp_global import ( + Bgp_globalArgs, +) +from ansible_collections.arista.eos.plugins.module_utils.network.eos.config.bgp_global.bgp_global import ( + Bgp_global, +) + + +def main(): + """ + Main entry point for module execution + + :returns: the result form module invocation + """ + module = AnsibleModule( + argument_spec=Bgp_globalArgs.argument_spec, + mutually_exclusive=[], + required_if=[], + supports_check_mode=False, + ) + + result = Bgp_global(module).execute_module() + module.exit_json(**result) + + +if __name__ == "__main__": + main() diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/modules/eos_command.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/modules/eos_command.py new file mode 100644 index 00000000..c4f43c77 --- /dev/null +++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/modules/eos_command.py @@ -0,0 +1,260 @@ +#!/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/>. +# +from __future__ import absolute_import, division, print_function + +__metaclass__ = type + + +DOCUMENTATION = """ +module: eos_command +author: Peter Sprygada (@privateip) +short_description: Run arbitrary commands on an Arista EOS device +description: +- Sends an arbitrary set of commands to an EOS node and returns the results read from + the device. This module includes an argument that will cause the module to wait + for a specific condition before returning or timing out if the condition is not + met. +version_added: 1.0.0 +extends_documentation_fragment: +- arista.eos.eos +notes: +- Tested against EOS 4.15 +options: + commands: + description: + - The commands to send to the remote EOS device over the configured provider. The + resulting output from the command is returned. If the I(wait_for) argument + is provided, the module is not returned until the condition is satisfied or + the number of I(retries) has been exceeded. + - If a command sent to the device requires answering a prompt, it is possible to pass + a dict containing command, answer and prompt. Common answers are 'y' or "\\r" + (carriage return, must be double quotes). Refer below examples. + required: true + type: list + elements: raw + wait_for: + description: + - Specifies what to evaluate from the output of the command and what conditionals + to apply. This argument will cause the task to wait for a particular conditional + to be true before moving forward. If the conditional is not true by the configured + retries, the task fails. Note - With I(wait_for) the value in C(result['stdout']) + can be accessed using C(result), that is to access C(result['stdout'][0]) use + C(result[0]) See examples. + type: list + elements: str + aliases: + - waitfor + match: + description: + - The I(match) argument is used in conjunction with the I(wait_for) argument to + specify the match policy. Valid values are C(all) or C(any). If the value + is set to C(all) then all conditionals in the I(wait_for) must be satisfied. If + the value is set to C(any) then only one of the values must be satisfied. + type: str + default: all + choices: + - any + - all + retries: + description: + - Specifies the number of retries a command should be tried before it is considered + failed. The command is run on the target device every retry and evaluated against + the I(wait_for) conditionals. + default: 10 + type: int + interval: + description: + - Configures the interval in seconds to wait between retries of the command. If + the command does not pass the specified conditional, the interval indicates + how to long to wait before trying the command again. + default: 1 + type: int +""" + +EXAMPLES = """ +- name: run show version on remote devices + arista.eos.eos_command: + commands: show version + +- name: run show version and check to see if output contains Arista + arista.eos.eos_command: + commands: show version + wait_for: result[0] contains Arista + +- name: run multiple commands on remote nodes + arista.eos.eos_command: + commands: + - show version + - show interfaces + +- name: run multiple commands and evaluate the output + arista.eos.eos_command: + commands: + - show version + - show interfaces + wait_for: + - result[0] contains Arista + - result[1] contains Loopback0 + +- name: run commands and specify the output format + arista.eos.eos_command: + commands: + - command: show version + output: json + +- name: using cli transport, check whether the switch is in maintenance mode + arista.eos.eos_command: + commands: show maintenance + wait_for: result[0] contains 'Under Maintenance' + +- name: using cli transport, check whether the switch is in maintenance mode using + json output + arista.eos.eos_command: + commands: show maintenance | json + wait_for: result[0].units.System.state eq 'underMaintenance' + +- name: using eapi transport check whether the switch is in maintenance, with 8 retries + and 2 second interval between retries + arista.eos.eos_command: + commands: show maintenance + wait_for: result[0]['units']['System']['state'] eq 'underMaintenance' + interval: 2 + retries: 8 + provider: + transport: eapi +""" + +RETURN = """ +stdout: + description: The set of responses from the commands + returned: always apart from low level errors (such as action plugin) + type: list + sample: ['...', '...'] +stdout_lines: + description: The value of stdout split into a list + returned: always apart from low level errors (such as action plugin) + type: list + sample: [['...', '...'], ['...'], ['...']] +failed_conditions: + description: The list of conditionals that have failed + returned: failed + type: list + sample: ['...', '...'] +""" +import time + +from ansible.module_utils._text import to_text +from ansible.module_utils.basic import AnsibleModule +from ansible_collections.ansible.netcommon.plugins.module_utils.network.common.parsing import ( + Conditional, +) +from ansible_collections.ansible.netcommon.plugins.module_utils.network.common.utils import ( + transform_commands, + to_lines, +) +from ansible_collections.arista.eos.plugins.module_utils.network.eos.eos import ( + run_commands, +) +from ansible_collections.arista.eos.plugins.module_utils.network.eos.eos import ( + eos_argument_spec, +) + + +def parse_commands(module, warnings): + commands = transform_commands(module) + + if module.check_mode: + for item in list(commands): + if not item["command"].startswith("show"): + warnings.append( + "Only show commands are supported when using check mode, not " + "executing %s" % item["command"] + ) + commands.remove(item) + + return commands + + +def to_cli(obj): + cmd = obj["command"] + if obj.get("output") == "json": + cmd += " | json" + return cmd + + +def main(): + """entry point for module execution + """ + argument_spec = dict( + commands=dict(type="list", required=True, elements="raw"), + wait_for=dict(type="list", aliases=["waitfor"], elements="str"), + match=dict(default="all", choices=["all", "any"]), + retries=dict(default=10, type="int"), + interval=dict(default=1, type="int"), + ) + + argument_spec.update(eos_argument_spec) + + module = AnsibleModule( + argument_spec=argument_spec, supports_check_mode=True + ) + + warnings = list() + result = {"changed": False, "warnings": warnings} + commands = parse_commands(module, warnings) + wait_for = module.params["wait_for"] or list() + + try: + conditionals = [Conditional(c) for c in wait_for] + except AttributeError as exc: + module.fail_json(msg=to_text(exc)) + + retries = module.params["retries"] + interval = module.params["interval"] + match = module.params["match"] + + while retries > 0: + responses = run_commands(module, commands) + + for item in list(conditionals): + if item(responses): + if match == "any": + conditionals = list() + break + conditionals.remove(item) + + if not conditionals: + break + + time.sleep(interval) + retries -= 1 + + if conditionals: + failed_conditions = [item.raw for item in conditionals] + msg = "One or more conditional statements have not been satisfied" + module.fail_json(msg=msg, failed_conditions=failed_conditions) + + result.update( + {"stdout": responses, "stdout_lines": list(to_lines(responses))} + ) + + module.exit_json(**result) + + +if __name__ == "__main__": + main() diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/modules/eos_config.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/modules/eos_config.py new file mode 100644 index 00000000..fe53b502 --- /dev/null +++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/modules/eos_config.py @@ -0,0 +1,593 @@ +#!/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/>. +# +from __future__ import absolute_import, division, print_function + +__metaclass__ = type + + +DOCUMENTATION = """ +module: eos_config +author: Peter Sprygada (@privateip) +short_description: Manage Arista EOS configuration sections +description: +- Arista EOS configurations use a simple block indent file syntax for segmenting configuration + into sections. This module provides an implementation for working with EOS configuration + sections in a deterministic way. This module works with either CLI or eAPI transports. +version_added: 1.0.0 +extends_documentation_fragment: +- arista.eos.eos +notes: +- Tested against EOS 4.15 +- Abbreviated commands are NOT idempotent, see + L(Network FAQ,../network/user_guide/faq.html#why-do-the-config-modules-always-return-changed-true-with-abbreviated-commands). +- To ensure idempotency and correct diff the configuration lines in the relevant module options should be similar to how they + appear if present in the running configuration on device including the indentation. +options: + lines: + description: + - The ordered set of commands that should be configured in the section. The commands + must be the exact same commands as found in the device running-config as found in the + device running-config to ensure idempotency and correct diff. Be sure + to note the configuration command syntax as some commands are automatically + modified by the device config parser. + aliases: + - commands + type: list + elements: str + parents: + description: + - The ordered set of parents that uniquely identify the section or hierarchy the + commands should be checked against. If the parents argument is omitted, the + commands are checked against the set of top level or global commands. + type: list + elements: str + src: + description: + - The I(src) argument provides a path to the configuration file to load into the + remote system. The path can either be a full system path to the configuration + file if the value starts with / or relative to the root of the implemented role + or playbook. This argument is mutually exclusive with the I(lines) and I(parents) + arguments. It can be a Jinja2 template as well. The configuration lines in the source + file should be similar to how it will appear if present in the running-configuration + (live switch config) of the device i ncluding the indentation to ensure idempotency + and correct diff. Arista EOS device config has 3 spaces indentation. + type: path + before: + description: + - The ordered set of commands to push on to the command stack if a change needs + to be made. This allows the playbook designer the opportunity to perform configuration + commands prior to pushing any changes without affecting how the set of commands + are matched against the system. + type: list + elements: str + after: + description: + - The ordered set of commands to append to the end of the command stack if a change + needs to be made. Just like with I(before) this allows the playbook designer + to append a set of commands to be executed after the command set. + type: list + elements: str + match: + description: + - Instructs the module on the way to perform the matching of 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(strict), command lines are matched + with respect to position. If match is set to I(exact), command lines 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. + type: str + default: line + choices: + - line + - strict + - exact + - none + replace: + description: + - Instructs the module on the way to perform the configuration on the device. If + the replace argument is set to I(line) then the modified lines are pushed to + the device in configuration mode. If the replace argument is set to I(block) + then the entire command block is pushed to the device in configuration mode + if any line is not correct. + type: str + default: line + choices: + - line + - block + - config + 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. If the C(backup_options) + value is not given, the backup file is written to the C(backup) folder in the + playbook root directory or role root directory, if playbook is part of an ansible + role. If the directory does not exist, it is created. + type: bool + default: no + running_config: + description: + - The module, by default, will connect to the remote device and retrieve the current + running-config to use as a base for comparing against the contents of source. There + are times when it is not desirable to have the task get the current running-config + for every task in a playbook. The I(running_config) argument allows the implementer + to pass in the configuration to use as the base config for this module. + The configuration lines for this option should be similar to how it will appear if present + in the running-configuration of the device including the indentation to ensure idempotency + and correct diff. + type: str + aliases: + - config + defaults: + description: + - The I(defaults) argument will influence how the running-config is collected + from the device. When the value is set to true, the command used to collect + the running-config is append with the all keyword. When the value is set to + false, the command is issued without the all keyword + type: bool + default: no + save_when: + description: + - When changes are made to the device running-configuration, the changes are not + copied to non-volatile storage by default. Using this argument will change + that before. If the argument is set to I(always), then the running-config will + always be copied to the startup-config and the I(modified) flag will always + be set to True. If the argument is set to I(modified), then the running-config + will only be copied to the startup-config if it has changed since the last save + to startup-config. If the argument is set to I(never), the running-config will + never be copied to the startup-config. If the argument is set to I(changed), + then the running-config will only be copied to the startup-config if the task + has made a change. I(changed) was added in Ansible 2.5. + default: never + type: str + choices: + - always + - never + - modified + - changed + diff_against: + description: + - When using the C(ansible-playbook --diff) command line argument the module can + generate diffs against different sources. + - When this option is configure as I(startup), the module will return the diff + of the running-config against the startup-config. + - When this option is configured as I(intended), the module will return the diff + of the running-config against the configuration provided in the C(intended_config) + argument. + - When this option is configured as I(running), the module will return the before + and after diff of the running-config with respect to any changes made to the + device configuration. + - When this option is configured as C(session), the diff returned will be based + on the configuration session. + default: session + type: str + choices: + - startup + - running + - intended + - session + diff_ignore_lines: + description: + - Use this argument to specify one or more lines that should be ignored during + the diff. This is used for lines in the configuration that are automatically + updated by the system. This argument takes a list of regular expressions or + exact line matches. + type: list + elements: str + intended_config: + description: + - The C(intended_config) provides the master configuration that the node should + conform to and is used to check the final running-config against. This argument + will not modify any settings on the remote device and is strictly used to check + the compliance of the current device's configuration against. When specifying + this argument, the task should also modify the C(diff_against) value and set + it to I(intended). The configuration lines for this value should be similar to how it + will appear if present in the running-configuration of the device including the indentation + to ensure correct diff. + type: str + backup_options: + description: + - This is a dict object containing configurable options related to backup file + path. The value of this option is read only when C(backup) is set to I(yes), + if C(backup) is set to I(no) this option will be silently ignored. + suboptions: + filename: + description: + - The filename to be used to store the backup configuration. If the filename + is not given it will be generated based on the hostname, current time and + date in format defined by <hostname>_config.<current-date>@<current-time> + type: str + dir_path: + description: + - This option provides the path ending with directory name in which the backup + configuration file will be stored. If the directory does not exist it will + be first created and the filename is either the value of C(filename) or + default filename as described in C(filename) options description. If the + path value is not given in that case a I(backup) directory will be created + in the current working directory and backup configuration will be copied + in C(filename) within I(backup) directory. + type: path + type: dict +""" +# noqa: E501 + +EXAMPLES = """ +- name: configure top level settings + arista.eos.eos_config: + lines: hostname {{ inventory_hostname }} + +- name: load an acl into the device + arista.eos.eos_config: + lines: + - 10 permit ip host 192.0.2.1 any log + - 20 permit ip host 192.0.2.2 any log + - 30 permit ip host 192.0.2.3 any log + - 40 permit ip host 192.0.2.4 any log + parents: ip access-list test + before: no ip access-list test + replace: block + +- name: load configuration from file + arista.eos.eos_config: + src: eos.cfg + +- name: render a Jinja2 template onto an Arista switch + arista.eos.eos_config: + backup: yes + src: eos_template.j2 + +- name: diff the running config against a master config + arista.eos.eos_config: + diff_against: intended + intended_config: "{{ lookup('file', 'master.cfg') }}" + +- name: for idempotency, use full-form commands + arista.eos.eos_config: + lines: + # - shut + - shutdown + # parents: int eth1 + parents: interface Ethernet1 + +- name: configurable backup path + arista.eos.eos_config: + src: eos_template.j2 + backup: yes + backup_options: + filename: backup.cfg + dir_path: /home/user +""" + +RETURN = """ +commands: + description: The set of commands that will be pushed to the remote device + returned: always + type: list + sample: ['hostname switch01', 'interface Ethernet1', 'no shutdown'] +updates: + description: The set of commands that will be pushed to the remote device + returned: always + type: list + sample: ['hostname switch01', 'interface Ethernet1', 'no shutdown'] +backup_path: + description: The full path to the backup file + returned: when backup is yes + type: str + sample: /playbooks/ansible/backup/eos_config.2016-07-16@22:28:34 +filename: + description: The name of the backup file + returned: when backup is yes and filename is not specified in backup options + type: str + sample: eos_config.2016-07-16@22:28:34 +shortname: + description: The full path to the backup file excluding the timestamp + returned: when backup is yes and filename is not specified in backup options + type: str + sample: /playbooks/ansible/backup/eos_config +date: + description: The date extracted from the backup file name + returned: when backup is yes + type: str + sample: "2016-07-16" +time: + description: The time extracted from the backup file name + returned: when backup is yes + type: str + sample: "22:28:34" +""" +from ansible.module_utils._text import to_text +from ansible.module_utils.basic import AnsibleModule +from ansible.module_utils.connection import ConnectionError +from ansible_collections.ansible.netcommon.plugins.module_utils.network.common.config import ( + NetworkConfig, + dumps, +) +from ansible_collections.arista.eos.plugins.module_utils.network.eos.eos import ( + get_config, + load_config, + get_connection, +) +from ansible_collections.arista.eos.plugins.module_utils.network.eos.eos import ( + run_commands, +) +from ansible_collections.arista.eos.plugins.module_utils.network.eos.eos import ( + eos_argument_spec, +) + + +def get_candidate(module): + candidate = "" + if module.params["src"]: + candidate = module.params["src"] + elif module.params["lines"]: + candidate_obj = NetworkConfig(indent=3) + parents = module.params["parents"] or list() + candidate_obj.add(module.params["lines"], parents=parents) + candidate = dumps(candidate_obj, "raw") + return candidate + + +def get_running_config(module, config=None, flags=None): + contents = module.params["running_config"] + if not contents: + if config: + contents = config + else: + contents = get_config(module, flags=flags) + return contents + + +def save_config(module, result): + result["changed"] = True + if not module.check_mode: + cmd = { + "command": "copy running-config startup-config", + "output": "text", + } + run_commands(module, [cmd]) + else: + module.warn( + "Skipping command `copy running-config startup-config` " + "due to check_mode. Configuration not copied to " + "non-volatile storage" + ) + + +def main(): + """ main entry point for module execution + """ + backup_spec = dict(filename=dict(), dir_path=dict(type="path")) + argument_spec = dict( + src=dict(type="path"), + lines=dict(aliases=["commands"], type="list", elements="str"), + parents=dict(type="list", elements="str"), + before=dict(type="list", elements="str"), + after=dict(type="list", elements="str"), + match=dict( + default="line", choices=["line", "strict", "exact", "none"] + ), + replace=dict(default="line", choices=["line", "block", "config"]), + defaults=dict(type="bool", default=False), + backup=dict(type="bool", default=False), + backup_options=dict(type="dict", options=backup_spec), + save_when=dict( + choices=["always", "never", "modified", "changed"], default="never" + ), + diff_against=dict( + choices=["startup", "session", "intended", "running"], + default="session", + ), + diff_ignore_lines=dict(type="list", elements="str"), + running_config=dict(aliases=["config"]), + intended_config=dict(), + ) + + argument_spec.update(eos_argument_spec) + + mutually_exclusive = [("lines", "src"), ("parents", "src")] + + required_if = [ + ("match", "strict", ["lines"]), + ("match", "exact", ["lines"]), + ("replace", "block", ["lines"]), + ("replace", "config", ["src"]), + ("diff_against", "intended", ["intended_config"]), + ] + + module = AnsibleModule( + argument_spec=argument_spec, + mutually_exclusive=mutually_exclusive, + required_if=required_if, + supports_check_mode=True, + ) + + warnings = list() + + result = {"changed": False} + if warnings: + result["warnings"] = warnings + + diff_ignore_lines = module.params["diff_ignore_lines"] + config = None + contents = None + flags = ["all"] if module.params["defaults"] else [] + connection = get_connection(module) + + # Refuse to diff_against: session if sessions are disabled + if ( + module.params["diff_against"] == "session" + and not connection.supports_sessions + ): + module.fail_json( + msg="Cannot diff against sessions when sessions are disabled. Please change diff_against to another value" + ) + + if module.params["backup"] or ( + module._diff and module.params["diff_against"] == "running" + ): + contents = get_config(module, flags=flags) + config = NetworkConfig(indent=1, contents=contents) + if module.params["backup"]: + result["__backup__"] = contents + + if any((module.params["src"], module.params["lines"])): + msg = ( + "To ensure idempotency and correct diff the input configuration lines should be" + " similar to how they appear if present in the running configuration on device" + ) + if module.params["src"]: + msg += " including the indentation" + warnings.append(msg) + + match = module.params["match"] + replace = module.params["replace"] + path = module.params["parents"] + + candidate = get_candidate(module) + running = get_running_config(module, contents, flags=flags) + + try: + response = connection.get_diff( + candidate=candidate, + running=running, + diff_match=match, + diff_ignore_lines=diff_ignore_lines, + path=path, + diff_replace=replace, + ) + except ConnectionError as exc: + module.fail_json(msg=to_text(exc, errors="surrogate_then_replace")) + + config_diff = response["config_diff"] + if config_diff: + commands = config_diff.split("\n") + if module.params["before"]: + commands[:0] = module.params["before"] + + if module.params["after"]: + commands.extend(module.params["after"]) + + result["commands"] = commands + result["updates"] = commands + + replace = module.params["replace"] == "config" + commit = not module.check_mode + + response = load_config( + module, commands, replace=replace, commit=commit + ) + + result["changed"] = True + + if module.params["diff_against"] == "session": + if "diff" in response: + result["diff"] = {"prepared": response["diff"]} + else: + result["changed"] = False + + if "session" in response: + result["session"] = response["session"] + + running_config = module.params["running_config"] + startup_config = None + + if module.params["save_when"] == "always": + save_config(module, result) + elif module.params["save_when"] == "modified": + output = run_commands( + module, + [ + {"command": "show running-config", "output": "text"}, + {"command": "show startup-config", "output": "text"}, + ], + ) + + running_config = NetworkConfig( + indent=3, contents=output[0], ignore_lines=diff_ignore_lines + ) + startup_config = NetworkConfig( + indent=3, contents=output[1], ignore_lines=diff_ignore_lines + ) + + if running_config.sha1 != startup_config.sha1: + save_config(module, result) + + elif module.params["save_when"] == "changed" and result["changed"]: + save_config(module, result) + + if module._diff: + if not running_config: + output = run_commands( + module, {"command": "show running-config", "output": "text"} + ) + contents = output[0] + else: + contents = running_config + + # recreate the object in order to process diff_ignore_lines + running_config = NetworkConfig( + indent=3, contents=contents, ignore_lines=diff_ignore_lines + ) + + if module.params["diff_against"] == "running": + if module.check_mode: + module.warn( + "unable to perform diff against running-config due to check mode" + ) + contents = None + else: + contents = config.config_text + + elif module.params["diff_against"] == "startup": + if not startup_config: + output = run_commands( + module, + {"command": "show startup-config", "output": "text"}, + ) + contents = output[0] + else: + contents = startup_config.config_text + + elif module.params["diff_against"] == "intended": + contents = module.params["intended_config"] + + if contents is not None: + base_config = NetworkConfig( + indent=3, contents=contents, ignore_lines=diff_ignore_lines + ) + + if running_config.sha1 != base_config.sha1: + if module.params["diff_against"] == "intended": + before = running_config + after = base_config + elif module.params["diff_against"] in ("startup", "running"): + before = base_config + after = running_config + + result.update( + { + "changed": True, + "diff": {"before": str(before), "after": str(after)}, + } + ) + + module.exit_json(**result) + + +if __name__ == "__main__": + main() diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/modules/eos_eapi.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/modules/eos_eapi.py new file mode 100644 index 00000000..24cbd9c8 --- /dev/null +++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/modules/eos_eapi.py @@ -0,0 +1,452 @@ +#!/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/>. +# +from __future__ import absolute_import, division, print_function + +__metaclass__ = type + + +DOCUMENTATION = """ +module: eos_eapi +author: Peter Sprygada (@privateip) +short_description: Manage and configure Arista EOS eAPI. +requirements: +- EOS v4.12 or greater +description: +- Use to enable or disable eAPI access, and set the port and state of http, https, + local_http and unix-socket servers. +- When enabling eAPI access the default is to enable HTTP on port 80, enable HTTPS + on port 443, disable local HTTP, and disable Unix socket server. Use the options + listed below to override the default configuration. +- Requires EOS v4.12 or greater. +version_added: 1.0.0 +extends_documentation_fragment: +- arista.eos.eos +options: + http: + description: + - The C(http) argument controls the operating state of the HTTP transport protocol + when eAPI is present in the running-config. When the value is set to True, the + HTTP protocol is enabled and when the value is set to False, the HTTP protocol + is disabled. By default, when eAPI is first configured, the HTTP protocol is + disabled. + type: bool + aliases: + - enable_http + http_port: + description: + - Configures the HTTP port that will listen for connections when the HTTP transport + protocol is enabled. This argument accepts integer values in the valid range + of 1 to 65535. + type: int + https: + description: + - The C(https) argument controls the operating state of the HTTPS transport protocol + when eAPI is present in the running-config. When the value is set to True, the + HTTPS protocol is enabled and when the value is set to False, the HTTPS protocol + is disabled. By default, when eAPI is first configured, the HTTPS protocol is + enabled. + type: bool + aliases: + - enable_https + https_port: + description: + - Configures the HTTP port that will listen for connections when the HTTP transport + protocol is enabled. This argument accepts integer values in the valid range + of 1 to 65535. + type: int + local_http: + description: + - The C(local_http) argument controls the operating state of the local HTTP transport + protocol when eAPI is present in the running-config. When the value is set + to True, the HTTP protocol is enabled and restricted to connections from localhost + only. When the value is set to False, the HTTP local protocol is disabled. + - Note is value is independent of the C(http) argument + type: bool + aliases: + - enable_local_http + local_http_port: + description: + - Configures the HTTP port that will listen for connections when the HTTP transport + protocol is enabled. This argument accepts integer values in the valid range + of 1 to 65535. + type: int + socket: + description: + - The C(socket) argument controls the operating state of the UNIX Domain Socket + used to receive eAPI requests. When the value of this argument is set to True, + the UDS will listen for eAPI requests. When the value is set to False, the + UDS will not be available to handle requests. By default when eAPI is first + configured, the UDS is disabled. + type: bool + aliases: + - enable_socket + timeout: + description: + - The time (in seconds) to wait for the eAPI configuration to be reflected in + the running-config. + type: int + default: 30 + vrf: + description: + - The C(vrf) argument will configure eAPI to listen for connections in the specified + VRF. By default, eAPI transports will listen for connections in the global + table. This value requires the VRF to already be created otherwise the task + will fail. + default: default + type: str + config: + description: + - The module, by default, will connect to the remote device and retrieve the current + running-config to use as a base for comparing against the contents of source. There + are times when it is not desirable to have the task get the current running-config + for every task in a playbook. The I(config) argument allows the implementer + to pass in the configuration to use as the base config for comparison. + type: str + state: + description: + - The C(state) argument controls the operational state of eAPI on the remote device. When + this argument is set to C(started), eAPI is enabled to receive requests and + when this argument is C(stopped), eAPI is disabled and will not receive requests. + type: str + default: started + choices: + - started + - stopped +""" + +EXAMPLES = """ +- name: Enable eAPI access with default configuration + arista.eos.eos_eapi: + state: started + +- name: Enable eAPI with no HTTP, HTTPS at port 9443, local HTTP at port 80, and socket + enabled + arista.eos.eos_eapi: + state: started + http: false + https_port: 9443 + local_http: yes + local_http_port: 80 + socket: yes + +- name: Shutdown eAPI access + arista.eos.eos_eapi: + state: stopped +""" + +RETURN = """ +commands: + description: The list of configuration mode commands to send to the device + returned: always + type: list + sample: + - management api http-commands + - protocol http port 81 + - no protocol https +urls: + description: Hash of URL endpoints eAPI is listening on per interface + returned: when eAPI is started + type: dict + sample: {'Management1': ['http://172.26.10.1:80']} +session_name: + description: The EOS config session name used to load the configuration + returned: when changed is True + type: str + sample: ansible_1479315771 +""" +import re +import time + +from ansible.module_utils.basic import AnsibleModule +from ansible_collections.arista.eos.plugins.module_utils.network.eos.eos import ( + run_commands, + load_config, +) +from ansible.module_utils.six import iteritems +from ansible_collections.arista.eos.plugins.module_utils.network.eos.eos import ( + eos_argument_spec, +) + + +def check_transport(module): + transport = (module.params["provider"] or {}).get("transport") + + if transport == "eapi": + module.fail_json( + msg="eos_eapi module is only supported over cli transport" + ) + + +def validate_http_port(value, module): + if not 1 <= value <= 65535: + module.fail_json(msg="http_port must be between 1 and 65535") + + +def validate_https_port(value, module): + if not 1 <= value <= 65535: + module.fail_json(msg="http_port must be between 1 and 65535") + + +def validate_local_http_port(value, module): + if not 1 <= value <= 65535: + module.fail_json(msg="http_port must be between 1 and 65535") + + +def validate_vrf(value, module): + out = run_commands(module, ["show vrf"]) + configured_vrfs = [] + lines = out[0].strip().splitlines()[3:] + for line in lines: + if not line: + continue + splitted_line = re.split(r"\s{2,}", line.strip()) + if len(splitted_line) > 2: + configured_vrfs.append(splitted_line[0]) + + configured_vrfs.append("default") + if value not in configured_vrfs: + module.fail_json( + msg="vrf `%s` is not configured on the system" % value + ) + + +def map_obj_to_commands(updates, module, warnings): + commands = list() + want, have = updates + + def needs_update(x): + return want.get(x) is not None and (want.get(x) != have.get(x)) + + def add(cmd): + if "management api http-commands" not in commands: + commands.insert(0, "management api http-commands") + commands.append(cmd) + + if any((needs_update("http"), needs_update("http_port"))): + if want["http"] is False: + add("no protocol http") + else: + if have["http"] is False and want["http"] in (False, None): + warnings.append( + "protocol http is not enabled, not configuring http port value" + ) + else: + port = want["http_port"] or 80 + add("protocol http port %s" % port) + + if any((needs_update("https"), needs_update("https_port"))): + if want["https"] is False: + add("no protocol https") + else: + if have["https"] is False and want["https"] in (False, None): + warnings.append( + "protocol https is not enabled, not configuring https port value" + ) + else: + port = want["https_port"] or 443 + add("protocol https port %s" % port) + + if any((needs_update("local_http"), needs_update("local_http_port"))): + if want["local_http"] is False: + add("no protocol http localhost") + else: + if have["local_http"] is False and want["local_http"] in ( + False, + None, + ): + warnings.append( + "protocol local_http is not enabled, not configuring local_http port value" + ) + else: + port = want["local_http_port"] or 8080 + add("protocol http localhost port %s" % port) + + if any((needs_update("socket"), needs_update("socket"))): + if want["socket"] is False: + add("no protocol unix-socket") + else: + add("protocol unix-socket") + if needs_update("state"): + if want["state"] == "stopped": + add("shutdown") + elif want["state"] == "started": + add("no shutdown") + + if needs_update("vrf"): + add("vrf %s" % want["vrf"]) + # switching operational vrfs here + # need to add the desired state as well + if want["state"] == "stopped": + add("shutdown") + elif want["state"] == "started": + add("no shutdown") + + return commands + + +def parse_state(data): + if data[0]["enabled"]: + return "started" + else: + return "stopped" + + +def map_config_to_obj(module): + out = run_commands(module, ["show management api http-commands | json"]) + return { + "http": out[0]["httpServer"]["configured"], + "http_port": out[0]["httpServer"]["port"], + "https": out[0]["httpsServer"]["configured"], + "https_port": out[0]["httpsServer"]["port"], + "local_http": out[0]["localHttpServer"]["configured"], + "local_http_port": out[0]["localHttpServer"]["port"], + "socket": out[0]["unixSocketServer"]["configured"], + "vrf": out[0]["vrf"] or "default", + "state": parse_state(out), + } + + +def map_params_to_obj(module): + obj = { + "http": module.params["http"], + "http_port": module.params["http_port"], + "https": module.params["https"], + "https_port": module.params["https_port"], + "local_http": module.params["local_http"], + "local_http_port": module.params["local_http_port"], + "socket": module.params["socket"], + "vrf": module.params["vrf"], + "state": module.params["state"], + } + + for key, value in iteritems(obj): + if value: + validator = globals().get("validate_%s" % key) + if validator: + validator(value, module) + + return obj + + +def verify_state(updates, module): + want, have = updates + + invalid_state = [ + ("http", "httpServer"), + ("https", "httpsServer"), + ("local_http", "localHttpServer"), + ("socket", "unixSocketServer"), + ] + + timeout = module.params["timeout"] + state = module.params["state"] + + while invalid_state: + out = run_commands( + module, ["show management api http-commands | json"] + ) + for index, item in enumerate(invalid_state): + want_key, eapi_key = item + if want[want_key] is not None: + if want[want_key] == out[0][eapi_key]["running"]: + del invalid_state[index] + elif state == "stopped": + if not out[0][eapi_key]["running"]: + del invalid_state[index] + else: + del invalid_state[index] + time.sleep(1) + timeout -= 1 + if timeout == 0: + module.fail_json( + msg="timeout expired before eapi running state changed" + ) + + +def collect_facts(module, result): + out = run_commands(module, ["show management api http-commands | json"]) + facts = dict(eos_eapi_urls=dict()) + for each in out[0]["urls"]: + intf, url = each.split(" : ") + key = str(intf).strip() + if key not in facts["eos_eapi_urls"]: + facts["eos_eapi_urls"][key] = list() + facts["eos_eapi_urls"][key].append(str(url).strip()) + result["ansible_facts"] = facts + + +def main(): + """ main entry point for module execution + """ + argument_spec = dict( + http=dict(aliases=["enable_http"], type="bool"), + http_port=dict(type="int"), + https=dict(aliases=["enable_https"], type="bool"), + https_port=dict(type="int"), + local_http=dict(aliases=["enable_local_http"], type="bool"), + local_http_port=dict(type="int"), + socket=dict(aliases=["enable_socket"], type="bool"), + timeout=dict(type="int", default=30), + vrf=dict(default="default"), + config=dict(), + state=dict(default="started", choices=["stopped", "started"]), + ) + + argument_spec.update(eos_argument_spec) + + module = AnsibleModule( + argument_spec=argument_spec, supports_check_mode=True + ) + + check_transport(module) + + result = {"changed": False} + + warnings = list() + if module.params["config"]: + warnings.append( + "config parameter is no longer necessary and will be ignored" + ) + + want = map_params_to_obj(module) + have = map_config_to_obj(module) + + commands = map_obj_to_commands((want, have), module, warnings) + result["commands"] = commands + + if commands: + commit = not module.check_mode + response = load_config(module, commands, commit=commit) + if response.get("diff") and module._diff: + result["diff"] = {"prepared": response.get("diff")} + result["session_name"] = response.get("session") + result["changed"] = True + + if result["changed"]: + verify_state((want, have), module) + + collect_facts(module, result) + + if warnings: + result["warnings"] = warnings + + module.exit_json(**result) + + +if __name__ == "__main__": + main() diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/modules/eos_facts.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/modules/eos_facts.py new file mode 100644 index 00000000..24e61ea7 --- /dev/null +++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/modules/eos_facts.py @@ -0,0 +1,207 @@ +#!/usr/bin/python +# -*- coding: utf-8 -*- +# Copyright 2019 Red Hat +# 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 + + +DOCUMENTATION = """ +module: eos_facts +author: +- Peter Sprygada (@privateip) +- Nathaniel Case (@Qalthos) +short_description: Collect facts from remote devices running Arista EOS +description: +- Collects facts from Arista devices running the EOS operating system. This module + places the facts gathered in the fact tree keyed by the respective resource name. The + facts module will always collect a base set of facts from the device and can enable + or disable collection of additional facts. +version_added: 1.0.0 +extends_documentation_fragment: +- arista.eos.eos +options: + gather_subset: + description: + - When supplied, this argument will restrict the facts collected to a given subset. Possible + values for this argument include all, hardware, config, and interfaces. Can + specify a list of values to include a larger subset. Values can also be used + with an initial C(M(!)) to specify that a specific subset should not be collected. + required: false + type: list + elements: str + default: '!config' + gather_network_resources: + description: + - When supplied, this argument will restrict the facts collected to a given subset. + Possible values for this argument include all and the resources like interfaces, + vlans etc. Can specify a list of values to include a larger subset. Values can + also be used with an initial C(M(!)) to specify that a specific subset should + not be collected. Values can also be used with an initial C(M(!)) to specify + that a specific subset should not be collected. Valid subsets are 'all', 'interfaces', + 'l2_interfaces', 'l3_interfaces', 'lacp', 'lacp_interfaces', 'lag_interfaces', + 'lldp_global', 'lldp_interfaces', 'vlans', 'acls'. + required: false + type: list + elements: str +""" + +EXAMPLES = """ +- name: Gather all legacy facts +- arista.eos.eos_facts: + gather_subset: all + +- name: Gather only the config and default facts + arista.eos.eos_facts: + gather_subset: + - config + +- name: Do not gather hardware facts + arista.eos.eos_facts: + gather_subset: + - '!hardware' + +- name: Gather legacy and resource facts + arista.eos.eos_facts: + gather_subset: all + gather_network_resources: all + +- name: Gather only the interfaces resource facts and no legacy facts +- arista.eos.eos_facts: + gather_subset: + - '!all' + - '!min' + gather_network_resources: + - interfaces + +- name: Gather all resource facts and minimal legacy facts + arista.eos.eos_facts: + gather_subset: min + gather_network_resources: all +""" + +RETURN = """ +ansible_net_gather_subset: + description: The list of fact subsets collected from the device + returned: always + type: list + +ansible_net_gather_network_resources: + description: The list of fact for network resource subsets collected from the device + returned: when the resource is configured + type: list + +# default +ansible_net_model: + description: The model name returned from the device + returned: always + type: str +ansible_net_serialnum: + description: The serial number of the remote device + returned: always + type: str +ansible_net_version: + description: The operating system version running on the remote device + returned: always + type: str +ansible_net_hostname: + description: The configured hostname of the device + returned: always + type: str +ansible_net_image: + description: The image file the device is running + returned: always + type: str +ansible_net_fqdn: + description: The fully qualified domain name of the device + returned: always + type: str +ansible_net_api: + description: The name of the transport + returned: always + type: str +ansible_net_python_version: + description: The Python version Ansible controller is using + returned: always + type: str + +# hardware +ansible_net_filesystems: + description: All file system names available on the device + returned: when hardware is configured + type: list +ansible_net_memfree_mb: + description: The available free memory on the remote device in Mb + returned: when hardware is configured + type: int +ansible_net_memtotal_mb: + description: The total memory on the remote device in Mb + returned: when hardware is configured + type: int + +# config +ansible_net_config: + description: The current active config from the device + returned: when config is configured + type: str + +# interfaces +ansible_net_all_ipv4_addresses: + description: All IPv4 addresses configured on the device + returned: when interfaces is configured + type: list +ansible_net_all_ipv6_addresses: + description: All IPv6 addresses configured on the device + returned: when interfaces is configured + type: list +ansible_net_interfaces: + description: A hash of all interfaces running on the system + returned: when interfaces is configured + type: dict +ansible_net_neighbors: + description: The list of LLDP neighbors from the remote device + returned: when interfaces is configured + type: dict +""" + +from ansible.module_utils.basic import AnsibleModule +from ansible_collections.arista.eos.plugins.module_utils.network.eos.argspec.facts.facts import ( + FactsArgs, +) +from ansible_collections.arista.eos.plugins.module_utils.network.eos.facts.facts import ( + Facts, +) +from ansible_collections.arista.eos.plugins.module_utils.network.eos.eos import ( + eos_argument_spec, +) + + +def main(): + """ + Main entry point for module execution + + :returns: ansible_facts + """ + argument_spec = FactsArgs.argument_spec + argument_spec.update(eos_argument_spec) + + module = AnsibleModule( + argument_spec=argument_spec, supports_check_mode=True + ) + warnings = [ + "default value for `gather_subset` " + "will be changed to `min` from `!config` v2.11 onwards" + ] + + result = Facts(module).get_facts() + + ansible_facts, additional_warnings = result + warnings.extend(additional_warnings) + + module.exit_json(ansible_facts=ansible_facts, warnings=warnings) + + +if __name__ == "__main__": + main() diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/modules/eos_interface.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/modules/eos_interface.py new file mode 100644 index 00000000..b90903ee --- /dev/null +++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/modules/eos_interface.py @@ -0,0 +1,601 @@ +#!/usr/bin/python +# -*- coding: utf-8 -*- + +# Copyright: (c) 2017, Ansible by Red Hat, inc +# 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 + + +DOCUMENTATION = """ +module: eos_interface +author: Ganesh Nalawade (@ganeshrn) +short_description: (deprecated, removed after 2022-06-01) Manage Interface on Arista + EOS network devices +description: +- This module provides declarative management of Interfaces on Arista EOS network + devices. +version_added: 1.0.0 +deprecated: + alternative: eos_interfaces + why: Updated modules released with more functionality + removed_at_date: '2022-06-01' +notes: +- Tested against EOS 4.15 +options: + name: + description: + - Name of the Interface to be configured on remote device. The name of interface + should be in expanded format and not abbreviated. + type: str + description: + description: + - Description of Interface upto 240 characters. + type: str + enabled: + description: + - Interface link status. If the value is I(True) the interface state will be enabled, + else if value is I(False) interface will be in disable (shutdown) state. + default: true + type: bool + speed: + description: + - This option configures autoneg and speed/duplex/flowcontrol for the interface + given in C(name) option. + type: str + mtu: + description: + - Set maximum transmission unit size in bytes of transmit packet for the interface + given in C(name) option. + type: str + tx_rate: + description: + - Transmit rate in bits per second (bps) for the interface given in C(name) option. + - This is state check parameter only. + - Supports conditionals, see L(Conditionals in Networking Modules,../network/user_guide/network_working_with_command_output.html) + type: str + rx_rate: + description: + - Receiver rate in bits per second (bps) for the interface given in C(name) option. + - This is state check parameter only. + - Supports conditionals, see L(Conditionals in Networking Modules,../network/user_guide/network_working_with_command_output.html) + type: str + neighbors: + description: + - Check the operational state of given interface C(name) for LLDP neighbor. + - The following suboptions are available. + type: list + elements: dict + suboptions: + host: + description: + - LLDP neighbor host for given interface C(name). + type: str + port: + description: + - LLDP neighbor port to which given interface C(name) is connected. + type: str + aggregate: + description: + - List of Interfaces definitions. Each of the entry in aggregate list should define + name of interface C(name) and other options as required. + type: list + elements: dict + suboptions: + name: + description: + - Name of the Interface to be configured on remote device. The name of interface + should be in expanded format and not abbreviated. + type: str + required: true + description: + description: + - Description of Interface upto 240 characters. + type: str + enabled: + description: + - Interface link status. If the value is I(True) the interface state will be enabled, + else if value is I(False) interface will be in disable (shutdown) state. + type: bool + speed: + description: + - This option configures autoneg and speed/duplex/flowcontrol for the interface + given in C(name) option. + type: str + mtu: + description: + - Set maximum transmission unit size in bytes of transmit packet for the interface + given in C(name) option. + type: str + tx_rate: + description: + - Transmit rate in bits per second (bps) for the interface given in C(name) option. + - This is state check parameter only. + - Supports conditionals, see L(Conditionals in Networking Modules,../network/user_guide/network_working_with_command_output.html) + type: str + rx_rate: + description: + - Receiver rate in bits per second (bps) for the interface given in C(name) option. + - This is state check parameter only. + - Supports conditionals, see L(Conditionals in Networking Modules,../network/user_guide/network_working_with_command_output.html) + type: str + neighbors: + description: + - Check the operational state of given interface C(name) for LLDP neighbor. + - The following suboptions are available. + type: list + elements: dict + suboptions: + host: + description: + - LLDP neighbor host for given interface C(name). + type: str + port: + description: + - LLDP neighbor port to which given interface C(name) is connected. + type: str + delay: + description: + - Time in seconds to wait before checking for the operational state on remote + device. This wait is applicable for operational state argument which are I(state) + with values C(up)/C(down), I(tx_rate) and I(rx_rate). + default: 10 + type: int + state: + description: + - State of the Interface configuration, C(up) means present and operationally + up and C(down) means present and operationally C(down) + type: str + choices: + - present + - absent + - up + - down + delay: + description: + - Time in seconds to wait before checking for the operational state on remote + device. This wait is applicable for operational state argument which are I(state) + with values C(up)/C(down), I(tx_rate) and I(rx_rate). + default: 10 + type: int + state: + description: + - State of the Interface configuration, C(up) means present and operationally + up and C(down) means present and operationally C(down) + default: present + type: str + choices: + - present + - absent + - up + - down +extends_documentation_fragment: +- arista.eos.eos + +""" + +EXAMPLES = """ +- name: configure interface + arista.eos.eos_interface: + name: ethernet1 + description: test-interface + speed: 100full + mtu: 512 + +- name: remove interface + arista.eos.eos_interface: + name: ethernet1 + state: absent + +- name: make interface up + arista.eos.eos_interface: + name: ethernet1 + enabled: true + +- name: make interface down + arista.eos.eos_interface: + name: ethernet1 + enabled: false + +- name: Check intent arguments + arista.eos.eos_interface: + name: ethernet1 + state: up + tx_rate: ge(0) + rx_rate: le(0) + +- name: Check neighbors intent arguments + arista.eos.eos_interface: + name: ethernet1 + neighbors: + - port: eth0 + host: netdev + +- name: Configure interface in disabled state and check if the operational state is + disabled or not + arista.eos.eos_interface: + name: ethernet1 + enabled: false + state: down + +- name: Add interface using aggregate + arista.eos.eos_interface: + aggregate: + - {name: ethernet1, mtu: 256, description: test-interface-1} + - {name: ethernet2, mtu: 516, description: test-interface-2} + speed: 100full + state: present + +- name: Delete interface using aggregate + arista.eos.eos_interface: + aggregate: + - name: loopback9 + - name: loopback10 + state: absent +""" + +RETURN = """ +commands: + description: The list of configuration mode commands to send to the device. + returned: always, except for the platforms that use Netconf transport to manage the device. + type: list + sample: + - interface ethernet1 + - description test-interface + - speed 100full + - mtu 512 +""" +import re +from copy import deepcopy +from time import sleep + +from ansible.module_utils.basic import AnsibleModule +from ansible_collections.ansible.netcommon.plugins.module_utils.network.common.config import ( + NetworkConfig, +) +from ansible_collections.ansible.netcommon.plugins.module_utils.network.common.utils import ( + conditional, + remove_default_spec, +) +from ansible_collections.arista.eos.plugins.module_utils.network.eos.eos import ( + get_config, + load_config, + run_commands, +) +from ansible_collections.arista.eos.plugins.module_utils.network.eos.eos import ( + eos_argument_spec, +) + + +def validate_mtu(value, module): + if value and not 68 <= int(value) <= 65535: + module.fail_json(msg="mtu must be between 68 and 65535") + + +def validate_param_values(module, obj, param=None): + if param is None: + param = module.params + for key in obj: + # validate the param value (if validator func exists) + validator = globals().get("validate_%s" % key) + if callable(validator): + validator(param.get(key), module) + + +def parse_shutdown(configobj, name): + cfg = configobj["interface %s" % name] + cfg = "\n".join(cfg.children) + match = re.search(r"shutdown", cfg, re.M) + return bool(match) + + +def parse_config_argument(configobj, name, arg=None): + cfg = configobj["interface %s" % name] + cfg = "\n".join(cfg.children) + match = re.search(r"%s (.+)$" % arg, cfg, re.M) + if match: + return match.group(1) + + +def search_obj_in_list(name, lst): + for o in lst: + if o["name"] == name: + return o + + return None + + +def add_command_to_interface(interface, cmd, commands): + if interface not in commands: + commands.append(interface) + commands.append(cmd) + + +def map_config_to_obj(module): + config = get_config(module) + configobj = NetworkConfig(indent=3, contents=config) + + match = re.findall(r"^interface (\S+)", config, re.M) + if not match: + return list() + + instances = list() + + for item in set(match): + obj = { + "name": item.lower(), + "description": parse_config_argument( + configobj, item, "description" + ), + "speed": parse_config_argument(configobj, item, "speed"), + "mtu": parse_config_argument(configobj, item, "mtu"), + "disable": parse_shutdown(configobj, item), + "state": "present", + } + instances.append(obj) + return instances + + +def map_params_to_obj(module): + obj = [] + aggregate = module.params.get("aggregate") + if aggregate: + for item in aggregate: + for key in item: + if item.get(key) is None: + item[key] = module.params[key] + + item["name"] = item["name"].lower() + validate_param_values(module, item, item) + d = item.copy() + + if d["enabled"]: + d["disable"] = False + else: + d["disable"] = True + + obj.append(d) + + else: + params = { + "name": module.params["name"].lower(), + "description": module.params["description"], + "speed": module.params["speed"], + "mtu": module.params["mtu"], + "state": module.params["state"], + "delay": module.params["delay"], + "tx_rate": module.params["tx_rate"], + "rx_rate": module.params["rx_rate"], + "neighbors": module.params["neighbors"], + } + + validate_param_values(module, params) + if module.params["enabled"]: + params.update({"disable": False}) + else: + params.update({"disable": True}) + + obj.append(params) + return obj + + +def map_obj_to_commands(updates, modules): + commands = list() + want, have = updates + + args = ("speed", "description", "mtu") + for w in want: + name = w["name"] + disable = w["disable"] + state = w["state"] + + obj_in_have = search_obj_in_list(name, have) + interface = "interface " + name + + if state == "absent" and obj_in_have: + commands.append("no " + interface) + + elif state in ("present", "up", "down"): + if obj_in_have: + for item in args: + candidate = w.get(item) + running = obj_in_have.get(item) + if candidate != running: + if candidate: + cmd = "{0} {1}".format(item, candidate) + add_command_to_interface(interface, cmd, commands) + + if disable and not obj_in_have.get("disable", False): + add_command_to_interface(interface, "shutdown", commands) + elif not disable and obj_in_have.get("disable", False): + add_command_to_interface( + interface, "no shutdown", commands + ) + else: + commands.append(interface) + for item in args: + value = w.get(item) + if value: + commands.append("{0} {1}".format(item, value)) + + if disable: + commands.append("no shutdown") + return commands + + +def check_declarative_intent_params(module, want, result): + failed_conditions = [] + have_neighbors = None + for w in want: + want_state = w.get("state") + want_tx_rate = w.get("tx_rate") + want_rx_rate = w.get("rx_rate") + want_neighbors = w.get("neighbors") + + if ( + want_state not in ("up", "down") + and not want_tx_rate + and not want_rx_rate + and not want_neighbors + ): + continue + + if result["changed"]: + sleep(w["delay"]) + + command = { + "command": "show interfaces %s" % w["name"], + "output": "text", + } + output = run_commands(module, [command]) + + if want_state in ("up", "down"): + match = re.search( + r"%s (\w+)" % "line protocol is", output[0], re.M + ) + have_state = None + if match: + have_state = match.group(1) + if have_state is None or not conditional( + want_state, have_state.strip() + ): + failed_conditions.append("state " + "eq(%s)" % want_state) + + if want_tx_rate: + match = re.search(r"%s (\d+)" % "output rate", output[0], re.M) + have_tx_rate = None + if match: + have_tx_rate = match.group(1) + + if have_tx_rate is None or not conditional( + want_tx_rate, have_tx_rate.strip(), cast=int + ): + failed_conditions.append("tx_rate " + want_tx_rate) + + if want_rx_rate: + match = re.search(r"%s (\d+)" % "input rate", output[0], re.M) + have_rx_rate = None + if match: + have_rx_rate = match.group(1) + + if have_rx_rate is None or not conditional( + want_rx_rate, have_rx_rate.strip(), cast=int + ): + failed_conditions.append("rx_rate " + want_rx_rate) + + if want_neighbors: + have_host = [] + have_port = [] + if have_neighbors is None: + command = { + "command": "show lldp neighbors {0}".format(w["name"]), + "output": "text", + } + have_neighbors = run_commands(module, [command]) + + if have_neighbors[0]: + lines = have_neighbors[0].strip().split("\n") + col = None + for index, line in enumerate(lines): + if re.search( + r"^Port\s+Neighbor Device ID\s+Neighbor Port ID\s+TTL", + line, + ): + col = index + break + + if col and col < len(lines) - 1: + idx = col + 1 + for items in lines[idx:]: + value = re.split(r"\s+", items) + try: + have_port.append(value[2]) + have_host.append(value[1]) + except IndexError: + pass + + for item in want_neighbors: + host = item.get("host") + port = item.get("port") + if host and host not in have_host: + failed_conditions.append("host " + host) + if port and port not in have_port: + failed_conditions.append("port " + port) + return failed_conditions + + +def main(): + """ main entry point for module execution + """ + neighbors_spec = dict(host=dict(), port=dict()) + + element_spec = dict( + name=dict(), + description=dict(), + speed=dict(), + mtu=dict(), + enabled=dict(default=True, type="bool"), + tx_rate=dict(), + rx_rate=dict(), + neighbors=dict(type="list", elements="dict", options=neighbors_spec), + delay=dict(default=10, type="int"), + state=dict( + default="present", choices=["present", "absent", "up", "down"] + ), + ) + + aggregate_spec = deepcopy(element_spec) + aggregate_spec["name"] = dict(required=True) + + # remove default in aggregate spec, to handle common arguments + remove_default_spec(aggregate_spec) + aggregate_spec["delay"].update(default=10) + + argument_spec = dict( + aggregate=dict(type="list", elements="dict", options=aggregate_spec) + ) + + argument_spec.update(element_spec) + argument_spec.update(eos_argument_spec) + + required_one_of = [["name", "aggregate"]] + mutually_exclusive = [["name", "aggregate"]] + + module = AnsibleModule( + argument_spec=argument_spec, + required_one_of=required_one_of, + mutually_exclusive=mutually_exclusive, + supports_check_mode=True, + ) + + warnings = list() + result = {"changed": False} + if warnings: + result["warnings"] = warnings + + want = map_params_to_obj(module) + have = map_config_to_obj(module) + commands = map_obj_to_commands((want, have), module) + result["commands"] = commands + + if commands: + commit = not module.check_mode + response = load_config(module, commands, commit=commit) + if response.get("diff") and module._diff: + result["diff"] = {"prepared": response.get("diff")} + result["session_name"] = response.get("session") + result["changed"] = True + + failed_conditions = check_declarative_intent_params(module, want, result) + + if failed_conditions: + msg = "One or more conditional statements have not been satisfied" + module.fail_json(msg=msg, failed_conditions=failed_conditions) + + module.exit_json(**result) + + +if __name__ == "__main__": + main() diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/modules/eos_interfaces.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/modules/eos_interfaces.py new file mode 100644 index 00000000..b844816a --- /dev/null +++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/modules/eos_interfaces.py @@ -0,0 +1,413 @@ +#!/usr/bin/python +# -*- coding: utf-8 -*- +# Copyright 2019 Red Hat +# GNU General Public License v3.0+ +# (see COPYING or https://www.gnu.org/licenses/gpl-3.0.txt) + +############################################## +# WARNING # +############################################## +# +# This file is auto generated by the resource +# module builder playbook. +# +# Do not edit this file manually. +# +# Changes to this file will be over written +# by the resource module builder. +# +# Changes should be made in the model used to +# generate this file or in the resource module +# builder template. +# +############################################## + +""" +The module file for eos_interfaces +""" + +from __future__ import absolute_import, division, print_function + +__metaclass__ = type + + +DOCUMENTATION = """ +module: eos_interfaces +short_description: Interfaces resource module +description: +- This module manages the interface attributes of Arista EOS interfaces. +version_added: 1.0.0 +author: +- Nathaniel Case (@qalthos) +notes: +- Tested against Arista EOS 4.20.10M +- This module works with connection C(network_cli). See the L(EOS Platform Options,../network/user_guide/platform_eos.html). +options: + config: + description: The provided configuration + type: list + elements: dict + suboptions: + name: + description: + - Full name of the interface, e.g. GigabitEthernet1. + type: str + required: True + description: + description: + - Interface description + type: str + duplex: + description: + - Interface link status. Applicable for Ethernet interfaces only. + - Values other than C(auto) must also set I(speed). + - Ignored when I(speed) is set above C(1000). + type: str + enabled: + default: true + description: + - Administrative state of the interface. + - Set the value to C(true) to administratively enable the interface or C(false) + to disable it. + type: bool + mtu: + description: + - MTU for a specific interface. Must be an even number between 576 and 9216. + Applicable for Ethernet interfaces only. + type: int + speed: + description: + - Interface link speed. Applicable for Ethernet interfaces only. + type: str + mode: + description: + - Manage Layer2 or Layer3 state of the interface. Applicable for Ethernet + and port channel interfaces only. + choices: + - layer2 + - layer3 + type: str + running_config: + description: + - This option is used only with state I(parsed). + - The value of this option should be the output received from the EOS device by + executing the command B(show running-config | section ^interface). + - The state I(parsed) reads the configuration from C(running_config) option and + transforms it into Ansible structured data as per the resource module's argspec + and the value is then returned in the I(parsed) key within the result. + type: str + state: + choices: + - merged + - replaced + - overridden + - deleted + - parsed + - rendered + - gathered + default: merged + description: + - The state of the configuration after module completion. + type: str + +""" + +EXAMPLES = """ + +# Using merged + +# Before state: +# ------------- +# +# veos#show running-config | section interface +# interface Ethernet1 +# description "Interface 1" +# ! +# interface Ethernet2 +# ! +# interface Management1 +# description "Management interface" +# ip address dhcp +# ! + +- name: Merge provided configuration with device configuration + arista.eos.eos_interfaces: + config: + - name: Ethernet1 + enabled: true + mode: layer3 + - name: Ethernet2 + description: Configured by Ansible + enabled: false + state: merged + +# After state: +# ------------ +# +# veos#show running-config | section interface +# interface Ethernet1 +# description "Interface 1" +# no switchport +# ! +# interface Ethernet2 +# description "Configured by Ansible" +# shutdown +# ! +# interface Management1 +# description "Management interface" +# ip address dhcp +# ! + +# Using replaced + +# Before state: +# ------------- +# +# veos#show running-config | section interface +# interface Ethernet1 +# description "Interface 1" +# ! +# interface Ethernet2 +# ! +# interface Management1 +# description "Management interface" +# ip address dhcp +# ! + +- name: Replaces device configuration of listed interfaces with provided configuration + arista.eos.eos_interfaces: + config: + - name: Ethernet1 + enabled: true + - name: Ethernet2 + description: Configured by Ansible + enabled: false + state: replaced + +# After state: +# ------------ +# +# veos#show running-config | section interface +# interface Ethernet1 +# ! +# interface Ethernet2 +# description "Configured by Ansible" +# shutdown +# ! +# interface Management1 +# description "Management interface" +# ip address dhcp +# ! + +# Using overridden + +# Before state: +# ------------- +# +# veos#show running-config | section interface +# interface Ethernet1 +# description "Interface 1" +# ! +# interface Ethernet2 +# ! +# interface Management1 +# description "Management interface" +# ip address dhcp +# ! + +- name: Overrides all device configuration with provided configuration + arista.eos.eos_interfaces: + config: + - name: Ethernet1 + enabled: true + - name: Ethernet2 + description: Configured by Ansible + enabled: false + state: overridden + +# After state: +# ------------ +# +# veos#show running-config | section interface +# interface Ethernet1 +# ! +# interface Ethernet2 +# description "Configured by Ansible" +# shutdown +# ! +# interface Management1 +# ip address dhcp +# ! + +# Using deleted + +# Before state: +# ------------- +# +# veos#show running-config | section interface +# interface Ethernet1 +# description "Interface 1" +# no switchport +# ! +# interface Ethernet2 +# ! +# interface Management1 +# description "Management interface" +# ip address dhcp +# ! + +- name: Delete or return interface parameters to default settings + arista.eos.eos_interfaces: + config: + - name: Ethernet1 + state: deleted + +# After state: +# ------------ +# +# veos#show running-config | section interface +# interface Ethernet1 +# ! +# interface Ethernet2 +# ! +# interface Management1 +# description "Management interface" +# ip address dhcp +# ! + +# Using rendered + +- name: Use Rendered to convert the structured data to native config + arista.eos.eos_interfaces: + config: + - name: Ethernet1 + enabled: true + mode: layer3 + - name: Ethernet2 + description: Configured by Ansible + enabled: false + state: merged + +# Output: +# ------------ + +# - "interface Ethernet1" +# - "description "Interface 1"" +# - "no swithcport" +# - "interface Ethernet2" +# - "description "Configured by Ansible"" +# - "shutdown" +# - "interface Management1" +# - "description "Management interface"" +# - "ip address dhcp" + +# Using parsed +# parsed.cfg + +# interface Ethernet1 +# description "Interface 1" +# ! +# interface Ethernet2 +# description "Configured by Ansible" +# shutdown +# ! + +- name: Use parsed to convert native configs to structured data + arista.eos.interfaces: + running_config: "{{ lookup('file', 'parsed.cfg') }}" + state: parsed + +# Output +# parsed: +# - name: Ethernet1 +# enabled: True +# mode: layer2 +# - name: Ethernet2 +# description: 'Configured by Ansible' +# enabled: False +# mode: layer2 + +# Using gathered: + +# Existing config on the device +# ----------------------------- +# interface Ethernet1 +# description "Interface 1" +# ! +# interface Ethernet2 +# description "Configured by Ansible" +# shutdown +# ! + +- name: Gather interfaces facts from the device + arista.eos.interfaces: + state: gathered + +# output +# gathered: +# - name: Ethernet1 +# enabled: True +# mode: layer2 +# - name: Ethernet2 +# description: 'Configured by Ansible' +# enabled: False +# mode: layer2 +""" + +RETURN = """ +before: + description: The configuration as structured data prior to module invocation. + returned: always + type: dict + sample: The configuration returned will always be in the same format of the parameters above. +after: + description: The configuration as structured data after module completion. + returned: when changed + type: dict + sample: The configuration returned will always be in the same format of the parameters above. +commands: + description: The set of commands pushed to the remote device. + returned: always + type: list + sample: ['interface Ethernet2', 'shutdown', 'speed 10full'] +""" + + +from ansible.module_utils.basic import AnsibleModule +from ansible_collections.arista.eos.plugins.module_utils.network.eos.argspec.interfaces.interfaces import ( + InterfacesArgs, +) +from ansible_collections.arista.eos.plugins.module_utils.network.eos.config.interfaces.interfaces import ( + Interfaces, +) + + +def main(): + """ + Main entry point for module execution + + :returns: the result form module invocation + """ + + required_if = [ + ("state", "merged", ("config",)), + ("state", "replaced", ("config",)), + ("state", "overridden", ("config",)), + ("state", "rendered", ("config",)), + ("state", "parsed", ("running_config",)), + ] + mutually_exclusive = [("config", "running_config")] + + module = AnsibleModule( + argument_spec=InterfacesArgs.argument_spec, + required_if=required_if, + supports_check_mode=True, + mutually_exclusive=mutually_exclusive, + ) + + result = Interfaces(module).execute_module() + module.exit_json(**result) + + +if __name__ == "__main__": + main() diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/modules/eos_l2_interface.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/modules/eos_l2_interface.py new file mode 100644 index 00000000..50edb8d8 --- /dev/null +++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/modules/eos_l2_interface.py @@ -0,0 +1,461 @@ +#!/usr/bin/python +# -*- coding: utf-8 -*- + +# (c) 2017, Ansible by Red Hat, inc +# 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 + + +DOCUMENTATION = """ +module: eos_l2_interface +author: Ricardo Carrillo Cruz (@rcarrillocruz) +short_description: (deprecated, removed after 2022-06-01) Manage L2 + interfaces on Arista EOS network devices. +description: +- This module provides declarative management of L2 interfaces on Arista EOS network + devices. +version_added: 1.0.0 +deprecated: + alternative: eos_l2_interfaces + why: Updated modules released with more functionality + removed_at_date: '2022-06-01' +notes: +- Tested against EOS 4.15 +options: + name: + description: + - Name of the interface + type: str + aliases: + - interface + mode: + description: + - Mode in which interface needs to be configured. + type: str + choices: + - access + - trunk + access_vlan: + description: + - Configure given VLAN in access port. If C(mode=access), used as the access VLAN + ID. + type: str + native_vlan: + description: + - Native VLAN to be configured in trunk port. If C(mode=trunk), used as the trunk + native VLAN ID. + type: str + trunk_allowed_vlans: + description: + - List of allowed VLANs in a given trunk port. If C(mode=trunk), these are the + ONLY VLANs that will be configured on the trunk, i.e. C(2-10,15). + type: str + aliases: + - trunk_vlans + aggregate: + description: + - List of Layer-2 interface definitions. + type: list + elements: dict + suboptions: + name: + description: + - Name of the interface + type: str + required: true + access_vlan: + description: + - Configure given VLAN in access port. If C(mode=access), used as the access VLAN + ID. + type: str + native_vlan: + description: + - Native VLAN to be configured in trunk port. If C(mode=trunk), used as the trunk + native VLAN ID. + type: str + trunk_allowed_vlans: + description: + - List of allowed VLANs in a given trunk port. If C(mode=trunk), these are the + ONLY VLANs that will be configured on the trunk, i.e. C(2-10,15). + type: str + aliases: + - trunk_vlans + mode: + description: + - Mode in which interface needs to be configured. + type: str + choices: + - access + - trunk + state: + description: + - Manage the state of the Layer-2 Interface configuration. + type: str + choices: + - present + - absent + state: + description: + - Manage the state of the Layer-2 Interface configuration. + type: str + default: present + choices: + - present + - absent +extends_documentation_fragment: +- arista.eos.eos + +""" + +EXAMPLES = """ +- name: Ensure Ethernet1 does not have any switchport + arista.eos.eos_l2_interface: + name: Ethernet1 + state: absent + +- name: Ensure Ethernet1 is configured for access vlan 20 + arista.eos.eos_l2_interface: + name: Ethernet1 + mode: access + access_vlan: 20 + +- name: Ensure Ethernet1 is a trunk port and ensure 2-50 are being tagged (doesn't + mean others aren't also being tagged) + arista.eos.eos_l2_interface: + name: Ethernet1 + mode: trunk + native_vlan: 10 + trunk_allowed_vlans: 2-50 + +- name: Set switchports on aggregate + arista.eos.eos_l2_interface: + aggregate: + - {name: ethernet1, mode: access, access_vlan: 20} + - {name: ethernet2, mode: trunk, native_vlan: 10} +""" + +RETURN = """ +commands: + description: The list of configuration mode commands to send to the device + returned: always. + type: list + sample: + - interface ethernet1 + - switchport access vlan 20 +""" +import re +from copy import deepcopy + +from ansible.module_utils.basic import AnsibleModule +from ansible_collections.ansible.netcommon.plugins.module_utils.network.common.config import ( + NetworkConfig, +) +from ansible_collections.ansible.netcommon.plugins.module_utils.network.common.utils import ( + remove_default_spec, +) +from ansible_collections.arista.eos.plugins.module_utils.network.eos.eos import ( + get_config, + load_config, + run_commands, +) +from ansible_collections.arista.eos.plugins.module_utils.network.eos.eos import ( + eos_argument_spec, +) + + +def parse_config_argument(configobj, name, arg=None): + cfg = configobj["interface %s" % name] + cfg = "\n".join(cfg.children) + match = re.search(r"%s (.+)$" % arg, cfg, re.M) + if match: + return match.group(1).strip() + + +def search_obj_in_list(name, lst): + for o in lst: + if o["name"] == name: + return o + + return None + + +def map_obj_to_commands(want, have, module): + commands = list() + + for w in want: + name = w["name"] + state = w["state"] + mode = w["mode"] + access_vlan = w["access_vlan"] + native_vlan = w["native_vlan"] + trunk_allowed_vlans = w["trunk_allowed_vlans"] + + interface = "interface " + name + commands.append(interface) + + obj_in_have = search_obj_in_list(name, have) + if not obj_in_have: + module.fail_json(msg="invalid interface {0}".format(name)) + + if state == "absent": + if obj_in_have["mode"] == "access": + commands.append( + "no switchport access vlan {0}".format( + obj_in_have["access_vlan"] + ) + ) + + if obj_in_have["mode"] == "trunk": + commands.append("no switchport mode trunk") + + if obj_in_have["native_vlan"]: + commands.append( + "no switchport trunk native vlan {0}".format( + obj_in_have["native_vlan"] + ) + ) + + if obj_in_have["trunk_allowed_vlans"]: + commands.append( + "no switchport trunk allowed vlan {0}".format( + obj_in_have["trunk_allowed_vlans"] + ) + ) + + if obj_in_have["state"] == "present": + commands.append("no switchport") + else: + if obj_in_have["state"] == "absent": + commands.append("switchport") + commands.append("switchport mode {0}".format(mode)) + + if access_vlan: + commands.append( + "switchport access vlan {0}".format(access_vlan) + ) + + if native_vlan: + commands.append( + "switchport trunk native vlan {0}".format(native_vlan) + ) + + if trunk_allowed_vlans: + commands.append( + "switchport trunk allowed vlan {0}".format( + trunk_allowed_vlans + ) + ) + else: + if mode != obj_in_have["mode"]: + if obj_in_have["mode"] == "access": + commands.append( + "no switchport access vlan {0}".format( + obj_in_have["access_vlan"] + ) + ) + commands.append("switchport mode trunk") + if native_vlan: + commands.append( + "switchport trunk native vlan {0}".format( + native_vlan + ) + ) + if trunk_allowed_vlans: + commands.append( + "switchport trunk allowed vlan {0}".format( + trunk_allowed_vlans + ) + ) + else: + if obj_in_have["native_vlan"]: + commands.append( + "no switchport trunk native vlan {0}".format( + obj_in_have["native_vlan"] + ) + ) + commands.append("no switchport mode trunk") + if obj_in_have["trunk_allowed_vlans"]: + commands.append( + "no switchport trunk allowed vlan {0}".format( + obj_in_have["trunk_allowed_vlans"] + ) + ) + commands.append("no switchport mode trunk") + commands.append( + "switchport access vlan {0}".format(access_vlan) + ) + else: + if mode == "access": + if access_vlan != obj_in_have["access_vlan"]: + commands.append( + "switchport access vlan {0}".format( + access_vlan + ) + ) + else: + if ( + native_vlan != obj_in_have["native_vlan"] + and native_vlan + ): + commands.append( + "switchport trunk native vlan {0}".format( + native_vlan + ) + ) + if ( + trunk_allowed_vlans + != obj_in_have["trunk_allowed_vlans"] + and trunk_allowed_vlans + ): + commands.append( + "switchport trunk allowed vlan {0}".format( + trunk_allowed_vlans + ) + ) + + if commands[-1] == interface: + commands.pop(-1) + + return commands + + +def map_config_to_obj(module, warnings): + config = get_config(module, flags=["| section interface"]) + configobj = NetworkConfig(indent=3, contents=config) + + match = re.findall(r"^interface (\S+)", config, re.M) + if not match: + return list() + + instances = list() + + for item in set(match): + command = { + "command": "show interfaces {0} switchport | include Switchport".format( + item + ), + "output": "text", + } + command_result = run_commands(module, command, check_rc=False) + if "Interface does not exist" in command_result[0]: + warnings.append( + "Could not gather switchport information for {0}: {1}".format( + item, command_result[0] + ) + ) + continue + + if command_result[0]: + switchport_cfg = command_result[0].split(":")[1].strip() + + if switchport_cfg == "Enabled": + state = "present" + else: + state = "absent" + + obj = { + "name": item.lower(), + "state": state, + "access_vlan": parse_config_argument( + configobj, item, "switchport access vlan" + ), + "native_vlan": parse_config_argument( + configobj, item, "switchport trunk native vlan" + ), + "trunk_allowed_vlans": parse_config_argument( + configobj, item, "switchport trunk allowed vlan" + ), + } + if obj["access_vlan"]: + obj["mode"] = "access" + else: + obj["mode"] = "trunk" + + instances.append(obj) + + return instances + + +def map_params_to_obj(module): + obj = [] + + aggregate = module.params.get("aggregate") + if aggregate: + for item in aggregate: + for key in item: + if item.get(key) is None: + item[key] = module.params[key] + + item["name"] = item["name"].lower() + obj.append(item.copy()) + else: + obj.append( + { + "name": module.params["name"].lower(), + "mode": module.params["mode"], + "access_vlan": module.params["access_vlan"], + "native_vlan": module.params["native_vlan"], + "trunk_allowed_vlans": module.params["trunk_allowed_vlans"], + "state": module.params["state"], + } + ) + + return obj + + +def main(): + """ main entry point for module execution + """ + element_spec = dict( + name=dict(type="str", aliases=["interface"]), + mode=dict(choices=["access", "trunk"]), + access_vlan=dict(type="str"), + native_vlan=dict(type="str"), + trunk_allowed_vlans=dict(type="str", aliases=["trunk_vlans"]), + state=dict(default="present", choices=["present", "absent"]), + ) + + aggregate_spec = deepcopy(element_spec) + aggregate_spec["name"] = dict(required=True) + + # remove default in aggregate spec, to handle common arguments + remove_default_spec(aggregate_spec) + + argument_spec = dict( + aggregate=dict(type="list", elements="dict", options=aggregate_spec) + ) + argument_spec.update(element_spec) + argument_spec.update(eos_argument_spec) + module = AnsibleModule( + argument_spec=argument_spec, + mutually_exclusive=[ + ["access_vlan", "native_vlan"], + ["access_vlan", "trunk_allowed_vlans"], + ], + supports_check_mode=True, + ) + + warnings = list() + result = {"changed": False, "warnings": warnings} + + want = map_params_to_obj(module) + have = map_config_to_obj(module, warnings) + commands = map_obj_to_commands(want, have, module) + result["commands"] = commands + + if commands: + commit = not module.check_mode + response = load_config(module, commands, commit=commit) + if response.get("diff") and module._diff: + result["diff"] = {"prepared": response.get("diff")} + result["session_name"] = response.get("session") + result["changed"] = True + + module.exit_json(**result) + + +if __name__ == "__main__": + main() diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/modules/eos_l2_interfaces.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/modules/eos_l2_interfaces.py new file mode 100644 index 00000000..0250dd16 --- /dev/null +++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/modules/eos_l2_interfaces.py @@ -0,0 +1,428 @@ +#!/usr/bin/python +# -*- coding: utf-8 -*- +# Copyright 2019 Red Hat +# GNU General Public License v3.0+ (see COPYING or https://www.gnu.org/licenses/gpl-3.0.txt) + +############################################## +# WARNING # +############################################## +# +# This file is auto generated by the resource +# module builder playbook. +# +# Do not edit this file manually. +# +# Changes to this file will be over written +# by the resource module builder. +# +# Changes should be made in the model used to +# generate this file or in the resource module +# builder template. +# +############################################## + +""" +The module file for eos_l2_interfaces +""" + +from __future__ import absolute_import, division, print_function + +__metaclass__ = type + + +DOCUMENTATION = """ +module: eos_l2_interfaces +short_description: L2 interfaces resource module +description: This module provides declarative management of Layer-2 interface on Arista + EOS devices. +version_added: 1.0.0 +author: Nathaniel Case (@qalthos) +notes: +- Tested against Arista EOS 4.20.10M +- This module works with connection C(network_cli). See the L(EOS Platform Options,../network/user_guide/platform_eos.html). +options: + config: + description: A dictionary of Layer-2 interface options + type: list + elements: dict + suboptions: + name: + description: + - Full name of interface, e.g. Ethernet1. + type: str + required: true + access: + description: + - Switchport mode access command to configure the interface as a layer 2 access. + type: dict + suboptions: + vlan: + description: + - Configure given VLAN in access port. It's used as the access VLAN ID. + type: int + trunk: + description: + - Switchport mode trunk command to configure the interface as a Layer 2 trunk. + type: dict + suboptions: + native_vlan: + description: + - Native VLAN to be configured in trunk port. It is used as the trunk + native VLAN ID. + type: int + trunk_allowed_vlans: + description: + - List of allowed VLANs in a given trunk port. These are the only VLANs + that will be configured on the trunk. + type: list + elements: str + mode: + description: + - Mode in which interface needs to be configured. + - Access mode is not shown in interface facts, so idempotency will not be + maintained for switchport mode access and every time the output will come + as changed=True. + type: str + choices: + - access + - trunk + running_config: + description: + - This option is used only with state I(parsed). + - The value of this option should be the output received from the EOS device by + executing the command B(show running-config | section ^interface). + - The state I(parsed) reads the configuration from C(running_config) option and + transforms it into Ansible structured data as per the resource module's argspec + and the value is then returned in the I(parsed) key within the result. + type: str + state: + choices: + - merged + - replaced + - overridden + - deleted + - parsed + - rendered + - gathered + default: merged + description: + - The state of the configuration after module completion + type: str + +""" + +EXAMPLES = """ + +# Using merged + +# Before state: +# ------------- +# +# veos#show running-config | section interface +# interface Ethernet1 +# switchport access vlan 20 +# ! +# interface Ethernet2 +# switchport trunk native vlan 20 +# switchport mode trunk +# ! +# interface Management1 +# ip address dhcp +# ipv6 address auto-config +# ! + +- name: Merge provided configuration with device configuration. + arista.eos.eos_l2_interfaces: + config: + - name: Ethernet1 + mode: trunk + trunk: + native_vlan: 10 + - name: Ethernet2 + mode: access + access: + vlan: 30 + state: merged + +# After state: +# ------------ +# +# veos#show running-config | section interface +# interface Ethernet1 +# switchport trunk native vlan 10 +# switchport mode trunk +# ! +# interface Ethernet2 +# switchport access vlan 30 +# ! +# interface Management1 +# ip address dhcp +# ipv6 address auto-config +# ! + +# Using replaced + +# Before state: +# ------------- +# +# veos2#show running-config | s int +# interface Ethernet1 +# switchport access vlan 20 +# ! +# interface Ethernet2 +# switchport trunk native vlan 20 +# switchport mode trunk +# ! +# interface Management1 +# ip address dhcp +# ipv6 address auto-config +# ! + +- name: Replace device configuration of specified L2 interfaces with provided configuration. + arista.eos.eos_l2_interfaces: + config: + - name: Ethernet1 + mode: trunk + trunk: + native_vlan: 20 + trunk_vlans: 5-10, 15 + state: replaced + +# After state: +# ------------ +# +# veos#show running-config | section interface +# interface Ethernet1 +# switchport trunk native vlan 20 +# switchport trunk allowed vlan 5-10,15 +# switchport mode trunk +# ! +# interface Ethernet2 +# switchport trunk native vlan 20 +# switchport mode trunk +# ! +# interface Management1 +# ip address dhcp +# ipv6 address auto-config +# ! + +# Using overridden + +# Before state: +# ------------- +# +# veos#show running-config | section interface +# interface Ethernet1 +# switchport access vlan 20 +# ! +# interface Ethernet2 +# switchport trunk native vlan 20 +# switchport mode trunk +# ! +# interface Management1 +# ip address dhcp +# ipv6 address auto-config +# ! + +- name: Override device configuration of all L2 interfaces on device with provided + configuration. + arista.eos.eos_l2_interfaces: + config: + - name: Ethernet2 + mode: access + access: + vlan: 30 + state: overridden + +# After state: +# ------------ +# +# veos#show running-config | section interface +# interface Ethernet1 +# ! +# interface Ethernet2 +# switchport access vlan 30 +# ! +# interface Management1 +# ip address dhcp +# ipv6 address auto-config +# ! + +# Using deleted + +# Before state: +# ------------- +# +# veos#show running-config | section interface +# interface Ethernet1 +# switchport access vlan 20 +# ! +# interface Ethernet2 +# switchport trunk native vlan 20 +# switchport mode trunk +# ! +# interface Management1 +# ip address dhcp +# ipv6 address auto-config +# ! + +- name: Delete EOS L2 interfaces as in given arguments. + arista.eos.eos_l2_interfaces: + config: + - name: Ethernet1 + - name: Ethernet2 + state: deleted + +# After state: +# ------------ +# +# veos#show running-config | section interface +# interface Ethernet1 +# ! +# interface Ethernet2 +# ! +# interface Management1 +# ip address dhcp +# ipv6 address auto-config + +# using rendered + +- name: Use Rendered to convert the structured data to native config + arista.eos.eos_l2_interfaces: + config: + - name: Ethernet1 + mode: trunk + trunk: + native_vlan: 10 + - name: Ethernet2 + mode: access + access: + vlan: 30 + state: merged + +# Output : +# ------------ +# +# - "interface Ethernet1" +# - "switchport trunk native vlan 10" +# - "switchport mode trunk" +# - "interface Ethernet2" +# - "switchport access vlan 30" +# - "interface Management1" +# - "ip address dhcp" +# - "ipv6 address auto-config" + + +# using parsed + +# parsed.cfg + +# interface Ethernet1 +# switchport trunk native vlan 10 +# switchport mode trunk +# ! +# interface Ethernet2 +# switchport access vlan 30 +# ! + +- name: Use parsed to convert native configs to structured data + arista.eos.l2_interfaces: + running_config: "{{ lookup('file', 'parsed.cfg') }}" + state: parsed + +# Output: +# parsed: +# - name: Ethernet1 +# mode: trunk +# trunk: +# native_vlan: 10 +# - name: Ethernet2 +# mode: access +# access: +# vlan: 30 + + +# Using gathered: +# Existing config on the device: +# +# veos#show running-config | section interface +# interface Ethernet1 +# switchport trunk native vlan 10 +# switchport mode trunk +# ! +# interface Ethernet2 +# switchport access vlan 30 +# ! + +- name: Gather interfaces facts from the device + arista.eos.l2_interfaces: + state: gathered +# output: +# gathered: +# - name: Ethernet1 +# mode: trunk +# trunk: +# native_vlan: 10 +# - name: Ethernet2 +# mode: access +# access: +# vlan: 30 + +""" + +RETURN = """ +before: + description: The configuration as structured data prior to module invocation. + returned: always + type: list + sample: The configuration returned will always be in the same format of the parameters above. +after: + description: The configuration as structured data after module completion. + returned: when changed + type: list + sample: The configuration returned will always be in the same format of the parameters above. +commands: + description: The set of commands pushed to the remote device. + returned: always + type: list + sample: ['interface Ethernet2', 'switchport access vlan 20'] +""" + + +from ansible.module_utils.basic import AnsibleModule +from ansible_collections.arista.eos.plugins.module_utils.network.eos.argspec.l2_interfaces.l2_interfaces import ( + L2_interfacesArgs, +) +from ansible_collections.arista.eos.plugins.module_utils.network.eos.config.l2_interfaces.l2_interfaces import ( + L2_interfaces, +) + + +def main(): + """ + Main entry point for module execution + + :returns: the result form module invocation + """ + required_if = [ + ("state", "merged", ("config",)), + ("state", "replaced", ("config",)), + ("state", "overridden", ("config",)), + ("state", "rendered", ("config",)), + ("state", "parsed", ("running_config",)), + ] + mutually_exclusive = [("config", "running_config")] + + module = AnsibleModule( + argument_spec=L2_interfacesArgs.argument_spec, + required_if=required_if, + supports_check_mode=True, + mutually_exclusive=mutually_exclusive, + ) + + result = L2_interfaces(module).execute_module() + module.exit_json(**result) + + +if __name__ == "__main__": + main() diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/modules/eos_l3_interface.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/modules/eos_l3_interface.py new file mode 100644 index 00000000..4fb9c249 --- /dev/null +++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/modules/eos_l3_interface.py @@ -0,0 +1,374 @@ +#!/usr/bin/python +# -*- coding: utf-8 -*- + +# Copyright: (c) 2017, Ansible by Red Hat, inc +# 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 + + +DOCUMENTATION = """ +module: eos_l3_interface +author: Ganesh Nalawade (@ganeshrn) +short_description: (deprecated, removed after 2022-06-01) Manage L3 interfaces on + Arista EOS network devices. +description: +- This module provides declarative management of L3 interfaces on Arista EOS network + devices. +version_added: 1.0.0 +deprecated: + alternative: eos_l3_interfaces + why: Updated modules released with more functionality + removed_at_date: '2022-06-01' +notes: +- Tested against EOS 4.15 +options: + name: + description: + - Name of the L3 interface to be configured eg. ethernet1 + type: str + ipv4: + description: + - IPv4 address to be set for the L3 interface mentioned in I(name) option. The + address format is <ipv4 address>/<mask>, the mask is number in range 0-32 eg. + 192.168.0.1/24 + type: str + ipv6: + description: + - IPv6 address to be set for the L3 interface mentioned in I(name) option. The + address format is <ipv6 address>/<mask>, the mask is number in range 0-128 eg. + fd5d:12c9:2201:1::1/64 + type: str + aggregate: + description: + - List of L3 interfaces definitions. Each of the entry in aggregate list should + define name of interface C(name) and a optional C(ipv4) or C(ipv6) address. + type: list + elements: dict + suboptions: + name: + description: + - Name of the L3 interface to be configured eg. ethernet1 + type: str + required: True + ipv4: + description: + - IPv4 address to be set for the L3 interface mentioned in I(name) option. The + address format is <ipv4 address>/<mask>, the mask is number in range 0-32 eg. + 192.168.0.1/24 + type: str + ipv6: + description: + - IPv6 address to be set for the L3 interface mentioned in I(name) option. The + address format is <ipv6 address>/<mask>, the mask is number in range 0-128 eg. + fd5d:12c9:2201:1::1/64 + type: str + state: + description: + - State of the L3 interface configuration. It indicates if the configuration should + be present or absent on remote device. + type: str + choices: + - present + - absent + state: + description: + - State of the L3 interface configuration. It indicates if the configuration should + be present or absent on remote device. + type: str + default: present + choices: + - present + - absent +extends_documentation_fragment: +- arista.eos.eos + +""" + +EXAMPLES = """ +- name: Remove ethernet1 IPv4 and IPv6 address + arista.eos.eos_l3_interface: + name: ethernet1 + state: absent + +- name: Set ethernet1 IPv4 address + arista.eos.eos_l3_interface: + name: ethernet1 + ipv4: 192.168.0.1/24 + +- name: Set ethernet1 IPv6 address + arista.eos.eos_l3_interface: + name: ethernet1 + ipv6: fd5d:12c9:2201:1::1/64 + +- name: Set interface Vlan1 (SVI) IPv4 address + arista.eos.eos_l3_interface: + name: Vlan1 + ipv4: 192.168.0.5/24 + +- name: Set IP addresses on aggregate + arista.eos.eos_l3_interface: + aggregate: + - name: ethernet1 + ipv4: 192.168.2.10/24 + - name: ethernet1 + ipv4: 192.168.3.10/24 + ipv6: fd5d:12c9:2201:1::1/64 + +- name: Remove IP addresses on aggregate + arista.eos.eos_l3_interface: + aggregate: + - name: ethernet1 + ipv4: 192.168.2.10/24 + - name: ethernet1 + ipv4: 192.168.3.10/24 + ipv6: fd5d:12c9:2201:1::1/64 + state: absent +""" + +RETURN = """ +commands: + description: The list of configuration mode commands to send to the device + returned: always, except for the platforms that use Netconf transport to manage the device. + type: list + sample: + - interface ethernet1 + - ip address 192.168.0.1/24 + - ipv6 address fd5d:12c9:2201:1::1/64 +""" +import re +from copy import deepcopy + +from ansible.module_utils.basic import AnsibleModule +from ansible_collections.ansible.netcommon.plugins.module_utils.network.common.config import ( + NetworkConfig, +) +from ansible_collections.ansible.netcommon.plugins.module_utils.network.common.utils import ( + remove_default_spec, +) +from ansible_collections.arista.eos.plugins.module_utils.network.eos.eos import ( + get_config, + load_config, +) +from ansible_collections.arista.eos.plugins.module_utils.network.eos.eos import ( + eos_argument_spec, +) +from ansible_collections.ansible.netcommon.plugins.module_utils.network.common.utils import ( + is_masklen, +) + + +def validate_ipv4(value, module): + if value: + address = value.split("/") + if len(address) != 2: + module.fail_json( + msg="address format is <ipv4 address>/<mask>, got invalid format %s" + % value + ) + + if not is_masklen(address[1]): + module.fail_json( + msg="invalid value for mask: %s, mask should be in range 0-32" + % address[1] + ) + + +def validate_ipv6(value, module): + if value: + address = value.split("/") + if len(address) != 2: + module.fail_json( + msg="address format is <ipv6 address>/<mask>, got invalid format %s" + % value + ) + else: + if not 0 <= int(address[1]) <= 128: + module.fail_json( + msg="invalid value for mask: %s, mask should be in range 0-128" + % address[1] + ) + + +def validate_param_values(module, obj, param=None): + if param is None: + param = module.params + for key in obj: + # validate the param value (if validator func exists) + validator = globals().get("validate_%s" % key) + if callable(validator): + validator(param.get(key), module) + + +def parse_config_argument(configobj, name, arg=None): + cfg = configobj["interface %s" % name] + cfg = "\n".join(cfg.children) + match = re.search(r"%s (.+)$" % arg, cfg, re.M) + if match: + return match.group(1).strip() + + +def search_obj_in_list(name, lst): + for o in lst: + if o["name"] == name: + return o + + return None + + +def map_obj_to_commands(updates, module): + commands = list() + want, have = updates + + for w in want: + name = w["name"] + ipv4 = w["ipv4"] + ipv6 = w["ipv6"] + state = w["state"] + + interface = "interface " + name + commands.append(interface) + + obj_in_have = search_obj_in_list(name, have) + if state == "absent" and obj_in_have: + if obj_in_have["ipv4"]: + if ipv4: + commands.append("no ip address {0}".format(ipv4)) + else: + commands.append("no ip address") + if obj_in_have["ipv6"]: + if ipv6: + commands.append("no ipv6 address {0}".format(ipv6)) + else: + commands.append("no ipv6 address") + + elif state == "present": + if ipv4: + if ( + obj_in_have is None + or obj_in_have["ipv4"] is None + or ipv4 != obj_in_have["ipv4"] + ): + commands.append("ip address {0}".format(ipv4)) + + if ipv6: + if ( + obj_in_have is None + or obj_in_have["ipv6"] is None + or ipv6.lower() != obj_in_have["ipv6"].lower() + ): + commands.append("ipv6 address {0}".format(ipv6)) + + if commands[-1] == interface: + commands.pop(-1) + + return commands + + +def map_config_to_obj(module): + config = get_config(module, flags=["| section interface"]) + configobj = NetworkConfig(indent=3, contents=config) + + match = re.findall(r"^interface (\S+)", config, re.M) + if not match: + return list() + + instances = list() + + for item in set(match): + obj = { + "name": item.lower(), + "ipv4": parse_config_argument(configobj, item, "ip address"), + "ipv6": parse_config_argument(configobj, item, "ipv6 address"), + "state": "present", + } + instances.append(obj) + + return instances + + +def map_params_to_obj(module): + obj = [] + + aggregate = module.params.get("aggregate") + if aggregate: + for item in aggregate: + for key in item: + if item.get(key) is None: + item[key] = module.params[key] + + item["name"] = item["name"].lower() + validate_param_values(module, item, item) + obj.append(item.copy()) + else: + obj.append( + { + "name": module.params["name"].lower(), + "ipv4": module.params["ipv4"], + "ipv6": module.params["ipv6"], + "state": module.params["state"], + } + ) + + validate_param_values(module, obj) + + return obj + + +def main(): + """ main entry point for module execution + """ + element_spec = dict( + name=dict(), + ipv4=dict(), + ipv6=dict(), + state=dict(default="present", choices=["present", "absent"]), + ) + + aggregate_spec = deepcopy(element_spec) + aggregate_spec["name"] = dict(required=True) + + # remove default in aggregate spec, to handle common arguments + remove_default_spec(aggregate_spec) + + argument_spec = dict( + aggregate=dict(type="list", elements="dict", options=aggregate_spec) + ) + + argument_spec.update(element_spec) + argument_spec.update(eos_argument_spec) + + required_one_of = [["name", "aggregate"]] + mutually_exclusive = [["name", "aggregate"]] + module = AnsibleModule( + argument_spec=argument_spec, + required_one_of=required_one_of, + mutually_exclusive=mutually_exclusive, + supports_check_mode=True, + ) + + warnings = list() + result = {"changed": False} + if warnings: + result["warnings"] = warnings + + want = map_params_to_obj(module) + have = map_config_to_obj(module) + commands = map_obj_to_commands((want, have), module) + result["commands"] = commands + + if commands: + commit = not module.check_mode + response = load_config(module, commands, commit=commit) + if response.get("diff") and module._diff: + result["diff"] = {"prepared": response.get("diff")} + result["session_name"] = response.get("session") + result["changed"] = True + + module.exit_json(**result) + + +if __name__ == "__main__": + main() diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/modules/eos_l3_interfaces.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/modules/eos_l3_interfaces.py new file mode 100644 index 00000000..38ab683b --- /dev/null +++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/modules/eos_l3_interfaces.py @@ -0,0 +1,406 @@ +#!/usr/bin/python +# -*- coding: utf-8 -*- +# Copyright 2019 Red Hat +# GNU General Public License v3.0+ +# (see COPYING or https://www.gnu.org/licenses/gpl-3.0.txt) + +############################################# +# WARNING # +############################################# +# +# This file is auto generated by the resource +# module builder playbook. +# +# Do not edit this file manually. +# +# Changes to this file will be over written +# by the resource module builder. +# +# Changes should be made in the model used to +# generate this file or in the resource module +# builder template. +# +############################################# + +""" +The module file for eos_l3_interfaces +""" + +from __future__ import absolute_import, division, print_function + +__metaclass__ = type + + +DOCUMENTATION = """ +module: eos_l3_interfaces +short_description: L3 interfaces resource module +description: This module provides declarative management of Layer 3 interfaces on + Arista EOS devices. +version_added: 1.0.0 +author: Nathaniel Case (@qalthos) +notes: +- Tested against Arista EOS 4.20.10M +- This module works with connection C(network_cli). See the L(EOS Platform Options,../network/user_guide/platform_eos.html). + 'eos_l2_interfaces/eos_interfaces' should be used for preparing the interfaces , before applying L3 configurations using + this module (eos_l3_interfaces). +options: + config: + description: A dictionary of Layer 3 interface options + type: list + elements: dict + suboptions: + name: + description: + - Full name of the interface, i.e. Ethernet1. + type: str + required: true + ipv4: + description: + - List of IPv4 addresses to be set for the Layer 3 interface mentioned in + I(name) option. + type: list + elements: dict + suboptions: + address: + description: + - IPv4 address to be set in the format <ipv4 address>/<mask> eg. 192.0.2.1/24, + or C(dhcp) to query DHCP for an IP address. + type: str + secondary: + description: + - Whether or not this address is a secondary address. + type: bool + ipv6: + description: + - List of IPv6 addresses to be set for the Layer 3 interface mentioned in + I(name) option. + type: list + elements: dict + suboptions: + address: + description: + - IPv6 address to be set in the address format is <ipv6 address>/<mask> + eg. 2001:db8:2201:1::1/64 or C(auto-config) to use SLAAC to chose an + address. + type: str + running_config: + description: + - This option is used only with state I(parsed). + - The value of this option should be the output received from the EOS device by + executing the command B(show running-config | section ^interface). + - The state I(parsed) reads the configuration from C(running_config) option and + transforms it into Ansible structured data as per the resource module's argspec + and the value is then returned in the I(parsed) key within the result. + type: str + state: + description: + - The state of the configuration after module completion + type: str + choices: + - merged + - replaced + - overridden + - deleted + - parsed + - gathered + - rendered + default: merged + +""" + +EXAMPLES = """ + +# Using deleted + +# Before state: +# ------------- +# +# veos#show running-config | section interface +# interface Ethernet1 +# ip address 192.0.2.12/24 +# ! +# interface Ethernet2 +# ipv6 address 2001:db8::1/64 +# ! +# interface Management1 +# ip address dhcp +# ipv6 address auto-config + +- name: Delete L3 attributes of given interfaces. + arista.eos.eos_l3_interfaces: + config: + - name: Ethernet1 + - name: Ethernet2 + state: deleted + +# After state: +# ------------ +# +# veos#show running-config | section interface +# interface Ethernet1 +# ! +# interface Ethernet2 +# ! +# interface Management1 +# ip address dhcp +# ipv6 address auto-config + + +# Using merged + +# Before state: +# ------------- +# +# veos#show running-config | section interface +# interface Ethernet1 +# ip address 192.0.2.12/24 +# ! +# interface Ethernet2 +# ipv6 address 2001:db8::1/64 +# ! +# interface Management1 +# ip address dhcp +# ipv6 address auto-config + +- name: Merge provided configuration with device configuration. + arista.eos.eos_l3_interfaces: + config: + - name: Ethernet1 + ipv4: + - address: 198.51.100.14/24 + - name: Ethernet2 + ipv4: + - address: 203.0.113.27/24 + state: merged + +# After state: +# ------------ +# +# veos#show running-config | section interface +# interface Ethernet1 +# ip address 198.51.100.14/24 +# ! +# interface Ethernet2 +# ip address 203.0.113.27/24 +# ipv6 address 2001:db8::1/64 +# ! +# interface Management1 +# ip address dhcp +# ipv6 address auto-config + + +# Using overridden + +# Before state: +# ------------- +# +# veos#show running-config | section interface +# interface Ethernet1 +# ip address 192.0.2.12/24 +# ! +# interface Ethernet2 +# ipv6 address 2001:db8::1/64 +# ! +# interface Management1 +# ip address dhcp +# ipv6 address auto-config + +- name: Override device configuration of all L2 interfaces on device with provided + configuration. + arista.eos.eos_l3_interfaces: + config: + - name: Ethernet1 + ipv6: + - address: 2001:db8:feed::1/96 + - name: Management1 + ipv4: + - address: dhcp + ipv6: auto-config + state: overridden + +# After state: +# ------------ +# +# veos#show running-config | section interface +# interface Ethernet1 +# ipv6 address 2001:db8:feed::1/96 +# ! +# interface Ethernet2 +# ! +# interface Management1 +# ip address dhcp +# ipv6 address auto-config + + +# Using replaced + +# Before state: +# ------------- +# +# veos#show running-config | section interface +# interface Ethernet1 +# ip address 192.0.2.12/24 +# ! +# interface Ethernet2 +# ipv6 address 2001:db8::1/64 +# ! +# interface Management1 +# ip address dhcp +# ipv6 address auto-config + +- name: Replace device configuration of specified L2 interfaces with provided configuration. + arista.eos.eos_l3_interfaces: + config: + - name: Ethernet2 + ipv4: + - address: 203.0.113.27/24 + state: replaced + +# After state: +# ------------ +# +# veos#show running-config | section interface +# interface Ethernet1 +# ip address 192.0.2.12/24 +# ! +# interface Ethernet2 +# ip address 203.0.113.27/24 +# ! +# interface Management1 +# ip address dhcp +# ipv6 address auto-config + +# Using parsed: + +# parsed.cfg +# ------------ +# +# veos#show running-config | section interface +# interface Ethernet1 +# ip address 198.51.100.14/24 +# ! +# interface Ethernet2 +# ip address 203.0.113.27/24 +# ! + +- name: Use parsed to convert native configs to structured data + arista.eos.interfaces: + running_config: "{{ lookup('file', 'parsed.cfg') }}" + state: parsed + +# Output: + +# parsed: +# - name: Ethernet1 +# ipv4: +# - address: 198.51.100.14/24 +# - name: Ethernet2 +# ipv4: +# - address: 203.0.113.27/24 + +# Using rendered: + +- name: Use Rendered to convert the structured data to native config + arista.eos.eos_l3_interfaces: + config: + - name: Ethernet1 + ipv4: + - address: 198.51.100.14/24 + - name: Ethernet2 + ipv4: + - address: 203.0.113.27/24 + state: rendered + +# Output +# ------------ +#rendered: +# - "interface Ethernet1" +# - "ip address 198.51.100.14/24" +# - "interface Ethernet2" +# - "ip address 203.0.113.27/24" + +# using gathered: + +# Native COnfig: +# veos#show running-config | section interface +# interface Ethernet1 +# ip address 198.51.100.14/24 +# ! +# interface Ethernet2 +# ip address 203.0.113.27/24 +# ! + +- name: Gather l3 interfaces facts from the device + arista.eos.l3_interfaces: + state: gathered + +# gathered: +# - name: Ethernet1 +# ipv4: +# - address: 198.51.100.14/24 +# - name: Ethernet2 +# ipv4: +# - address: 203.0.113.27/24 + + +""" +RETURN = """ +before: + description: The configuration as structured data prior to module invocation. + returned: always + type: list + sample: > + The configuration returned will always be in the same format + of the parameters above. +after: + description: The configuration as structured data after module completion. + returned: when changed + type: list + sample: > + The configuration returned will always be in the same format + of the parameters above. +commands: + description: The set of commands pushed to the remote device. + returned: always + type: list + sample: ['interface Ethernet2', 'ip address 192.0.2.12/24'] +""" + + +from ansible.module_utils.basic import AnsibleModule +from ansible_collections.arista.eos.plugins.module_utils.network.eos.argspec.l3_interfaces.l3_interfaces import ( + L3_interfacesArgs, +) +from ansible_collections.arista.eos.plugins.module_utils.network.eos.config.l3_interfaces.l3_interfaces import ( + L3_interfaces, +) + + +def main(): + """ + Main entry point for module execution + + :returns: the result form module invocation + """ + + required_if = [ + ("state", "merged", ("config",)), + ("state", "replaced", ("config",)), + ("state", "overridden", ("config",)), + ("state", "rendered", ("config",)), + ("state", "parsed", ("running_config",)), + ] + mutually_exclusive = [("config", "running_config")] + module = AnsibleModule( + argument_spec=L3_interfacesArgs.argument_spec, + required_if=required_if, + supports_check_mode=True, + mutually_exclusive=mutually_exclusive, + ) + + result = L3_interfaces(module).execute_module() + module.exit_json(**result) + + +if __name__ == "__main__": + main() diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/modules/eos_lacp.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/modules/eos_lacp.py new file mode 100644 index 00000000..33102756 --- /dev/null +++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/modules/eos_lacp.py @@ -0,0 +1,245 @@ +#!/usr/bin/python +# -*- coding: utf-8 -*- +# Copyright 2019 Red Hat +# GNU General Public License v3.0+ +# (see COPYING or https://www.gnu.org/licenses/gpl-3.0.txt) + +############################################# +# WARNING # +############################################# +# +# This file is auto generated by the resource +# module builder playbook. +# +# Do not edit this file manually. +# +# Changes to this file will be over written +# by the resource module builder. +# +# Changes should be made in the model used to +# generate this file or in the resource module +# builder template. +# +############################################# + +""" +The module file for eos_lacp +""" + +from __future__ import absolute_import, division, print_function + +__metaclass__ = type + + +DOCUMENTATION = """ +module: eos_lacp +short_description: LACP resource module +description: +- This module manages Global Link Aggregation Control Protocol (LACP) on Arista EOS + devices. +version_added: 1.0.0 +author: Nathaniel Case (@Qalthos) +notes: +- Tested against Arista EOS 4.20.10M +- This module works with connection C(network_cli). See the L(EOS Platform Options,../network/user_guide/platform_eos.html). +options: + config: + description: LACP global options. + type: dict + suboptions: + system: + description: LACP system options. + type: dict + suboptions: + priority: + description: + - The system priority to use in LACP negotiations. + - Lower value is higher priority. + - Refer to vendor documentation for valid values. + type: int + running_config: + description: + - This option is used only with state I(parsed). + - The value of this option should be the output received from the EOS device by + executing the command B(show running-config | section ^lacp). + - The state I(parsed) reads the configuration from C(running_config) option and + transforms it into Ansible structured data as per the resource module's argspec + and the value is then returned in the I(parsed) key within the result. + type: str + state: + description: + - The state of the configuration after module completion. + type: str + choices: + - merged + - replaced + - deleted + - parsed + - rendered + - gathered + default: merged + +""" +EXAMPLES = """ +# Using merged + +# Before state: +# ------------- +# veos# show running-config | include lacp +# lacp system-priority 10 + +- name: Merge provided global LACP attributes with device attributes + arista.eos.eos_lacp: + config: + system: + priority: 20 + state: merged + +# After state: +# ------------ +# veos# show running-config | include lacp +# lacp system-priority 20 +# + + +# Using replaced + +# Before state: +# ------------- +# veos# show running-config | include lacp +# lacp system-priority 10 + +- name: Replace device global LACP attributes with provided attributes + arista.eos.eos_lacp: + config: + system: + priority: 20 + state: replaced + +# After state: +# ------------ +# veos# show running-config | include lacp +# lacp system-priority 20 +# + + +# Using deleted + +# Before state: +# ------------- +# veos# show running-config | include lacp +# lacp system-priority 10 + +- name: Delete global LACP attributes + arista.eos.eos_lacp: + state: deleted + +# After state: +# ------------ +# veos# show running-config | include lacp +# + +#Using rendered: + +- name: Use Rendered to convert the structured data to native config + arista.eos.eos_lacp: + config: + system: + priority: 20 + state: rendered + +# Output: +# ------------ +# rendered: +# - "lacp system-priority 20" +# + +# Using parsed: + +# parsed.cfg +# lacp system-priority 20 + +- name: Use parsed to convert native configs to structured data + arista.eos.eos_lacp: + running_config: "{{ lookup('file', 'parsed.cfg') }}" + state: parsed + +# Output: +# parsed: +# system: +# priority: 20 + +# Using gathered: +# nathive config: +# ------------- +# lacp system-priority 10 + +- name: Gather lacp facts from the device + arista.eos.eos_lacp: + state: gathered + +# Output: +# gathered: +# system: +# priority: 10 +# + +""" +RETURN = """ +before: + description: The configuration as structured data prior to module invocation. + returned: always + type: dict + sample: > + The configuration returned will always be in the same format + of the parameters above. +after: + description: The configuration as structured data after module completion. + returned: when changed + type: dict + sample: > + The configuration returned will always be in the same format + of the parameters above. +commands: + description: The set of commands pushed to the remote device. + returned: always + type: list + sample: ['lacp system-priority 10'] +""" + + +from ansible.module_utils.basic import AnsibleModule +from ansible_collections.arista.eos.plugins.module_utils.network.eos.argspec.lacp.lacp import ( + LacpArgs, +) +from ansible_collections.arista.eos.plugins.module_utils.network.eos.config.lacp.lacp import ( + Lacp, +) + + +def main(): + """ + Main entry point for module execution + + :returns: the result form module invocation + """ + required_if = [ + ("state", "merged", ("config",)), + ("state", "replaced", ("config",)), + ("state", "rendered", ("config",)), + ("state", "parsed", ("running_config",)), + ] + mutually_exclusive = [("config", "running_config")] + module = AnsibleModule( + argument_spec=LacpArgs.argument_spec, + required_if=required_if, + supports_check_mode=True, + mutually_exclusive=mutually_exclusive, + ) + + result = Lacp(module).execute_module() + module.exit_json(**result) + + +if __name__ == "__main__": + main() diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/modules/eos_lacp_interfaces.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/modules/eos_lacp_interfaces.py new file mode 100644 index 00000000..e841f2c9 --- /dev/null +++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/modules/eos_lacp_interfaces.py @@ -0,0 +1,336 @@ +#!/usr/bin/python +# -*- coding: utf-8 -*- +# Copyright 2019 Red Hat +# GNU General Public License v3.0+ +# (see COPYING or https://www.gnu.org/licenses/gpl-3.0.txt) + +############################################# +# WARNING # +############################################# +# +# This file is auto generated by the resource +# module builder playbook. +# +# Do not edit this file manually. +# +# Changes to this file will be over written +# by the resource module builder. +# +# Changes should be made in the model used to +# generate this file or in the resource module +# builder template. +# +############################################# + +""" +The module file for eos_lacp_interfaces +""" + +from __future__ import absolute_import, division, print_function + +__metaclass__ = type + + +DOCUMENTATION = """ +module: eos_lacp_interfaces +short_description: LACP interfaces resource module +description: +- This module manages Link Aggregation Control Protocol (LACP) attributes of interfaces + on Arista EOS devices. +version_added: 1.0.0 +author: Nathaniel Case (@Qalthos) +notes: +- Tested against Arista EOS 4.20.10M +- This module works with connection C(network_cli). See the L(EOS Platform Options,../network/user_guide/platform_eos.html). +options: + config: + description: A dictionary of LACP interfaces options. + type: list + elements: dict + suboptions: + name: + description: + - Full name of the interface (i.e. Ethernet1). + type: str + port_priority: + description: + - LACP port priority for the interface. Range 1-65535. + type: int + rate: + description: + - Rate at which PDUs are sent by LACP. At fast rate LACP is transmitted once + every 1 second. At normal rate LACP is transmitted every 30 seconds after + the link is bundled. + type: str + choices: + - fast + - normal + running_config: + description: + - This option is used only with state I(parsed). + - The value of this option should be the output received from the EOS device by + executing the command B(show running-config | section ^interfaces). + - The state I(parsed) reads the configuration from C(running_config) option and + transforms it into Ansible structured data as per the resource module's argspec + and the value is then returned in the I(parsed) key within the result. + type: str + state: + description: + - The state of the configuration after module completion. + type: str + choices: + - merged + - replaced + - overridden + - deleted + - parsed + - rendered + - gathered + default: merged + +""" +EXAMPLES = """ +# Using merged +# +# +# ------------ +# Before state +# ------------ +# +# +# veos#show run | section ^interface +# interface Ethernet1 +# lacp port-priority 30 +# interface Ethernet2 +# lacp rate fast + +- name: Merge provided configuration with device configuration + arista.eos.eos_lacp_interfaces: + config: + - name: Ethernet1 + rate: fast + - name: Ethernet2 + rate: normal + state: merged + +# +# ----------- +# After state +# ----------- +# +# veos#show run | section ^interface +# interface Ethernet1 +# lacp port-priority 30 +# lacp rate fast +# interface Ethernet2 + + +# Using replaced +# +# +# ------------ +# Before state +# ------------ +# +# +# veos#show run | section ^interface +# interface Ethernet1 +# lacp port-priority 30 +# interface Ethernet2 +# lacp rate fast + +- name: Replace existing LACP configuration of specified interfaces with provided + configuration + arista.eos.eos_lacp_interfaces: + config: + - name: Ethernet1 + rate: fast + state: replaced + +# +# ----------- +# After state +# ----------- +# +# veos#show run | section ^interface +# interface Ethernet1 +# lacp rate fast +# interface Ethernet2 +# lacp rate fast + + +# Using overridden +# +# +# ------------ +# Before state +# ------------ +# +# +# veos#show run | section ^interface +# interface Ethernet1 +# lacp port-priority 30 +# interface Ethernet2 +# lacp rate fast + +- name: Override the LACP configuration of all the interfaces with provided configuration + arista.eos.eos_lacp_interfaces: + config: + - name: Ethernet1 + rate: fast + state: overridden + +# +# ----------- +# After state +# +# +# veos#show run | section ^interface +# interface Ethernet1 +# lacp rate fast +# interface Ethernet2 + + +# Using deleted +# +# +# ------------ +# Before state +# ------------ +# +# +# veos#show run | section ^interface +# interface Ethernet1 +# lacp port-priority 30 +# interface Ethernet2 +# lacp rate fast + +- name: Delete LACP attributes of given interfaces (or all interfaces if none specified). + arista.eos.eos_lacp_interfaces: + state: deleted + +# +# ----------- +# After state +# ----------- +# +# veos#show run | section ^interface +# interface Ethernet1 +# interface Ethernet2 + +# using rendered: + +- name: Use Rendered to convert the structured data to native config + arista.eos.eos_lacp_interfaces: + config: + - name: Ethernet1 + rate: fast + - name: Ethernet2 + rate: normal + state: rendered + +# +# ----------- +# Output +# ----------- +# rendered: +# - "interface Ethernet1" +# - "lacp rate fast" + +# Using parsed: + +# parsed.cfg: +# "interface Ethernet1" +# "lacp rate fast" +# "interface Ethernet2" + +- name: Use parsed to convert native configs to structured data + arista.eos.eos_lacp_interfaces: + running_config: "{{ lookup('file', 'parsed.cfg') }}" + state: parsed + +# Output: +# parsed: +# - name: Ethernet1 +# rate: fast +# - name: Ethernet2 +# rate: normal + +# Using gathered: +# native config: +# veos#show run | section ^interface +# interface Ethernet1 +# lacp port-priority 30 +# interface Ethernet2 +# lacp rate fast + +- name: Gather LACP facts from the device + arista.eos.eos_lacp_interfaces: + state: gathered + +# Output: +# gathered: +# - name: Ethernet1 +# - name: Ethernet2 +# rate: fast + +""" +RETURN = """ +before: + description: The configuration as structured data prior to module invocation. + returned: always + type: list + sample: > + The configuration returned will always be in the same format + of the parameters above. +after: + description: The configuration as structured data after module completion. + returned: when changed + type: list + sample: > + The configuration returned will always be in the same format + of the parameters above. +commands: + description: The set of commands pushed to the remote device. + returned: always + type: list + sample: ['interface Ethernet1', 'lacp rate fast'] +""" + + +from ansible.module_utils.basic import AnsibleModule +from ansible_collections.arista.eos.plugins.module_utils.network.eos.argspec.lacp_interfaces.lacp_interfaces import ( + Lacp_interfacesArgs, +) +from ansible_collections.arista.eos.plugins.module_utils.network.eos.config.lacp_interfaces.lacp_interfaces import ( + Lacp_interfaces, +) + + +def main(): + """ + Main entry point for module execution + + :returns: the result form module invocation + """ + required_if = [ + ("state", "merged", ("config",)), + ("state", "replaced", ("config",)), + ("state", "overridden", ("config",)), + ("state", "rendered", ("config",)), + ("state", "parsed", ("running_config",)), + ] + mutually_exclusive = [("config", "running_config")] + module = AnsibleModule( + argument_spec=Lacp_interfacesArgs.argument_spec, + required_if=required_if, + supports_check_mode=True, + mutually_exclusive=mutually_exclusive, + ) + + result = Lacp_interfaces(module).execute_module() + module.exit_json(**result) + + +if __name__ == "__main__": + main() diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/modules/eos_lag_interfaces.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/modules/eos_lag_interfaces.py new file mode 100644 index 00000000..0d3bf983 --- /dev/null +++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/modules/eos_lag_interfaces.py @@ -0,0 +1,340 @@ +#!/usr/bin/python +# -*- coding: utf-8 -*- +# Copyright 2019 Red Hat +# GNU General Public License v3.0+ +# (see COPYING or https://www.gnu.org/licenses/gpl-3.0.txt) + +############################################# +# WARNING # +############################################# +# +# This file is auto generated by the resource +# module builder playbook. +# +# Do not edit this file manually. +# +# Changes to this file will be over written +# by the resource module builder. +# +# Changes should be made in the model used to +# generate this file or in the resource module +# builder template. +# +############################################# + +""" +The module file for eos_lag_interfaces +""" + +from __future__ import absolute_import, division, print_function + +__metaclass__ = type + + +DOCUMENTATION = """ +module: eos_lag_interfaces +short_description: LAG interfaces resource module +description: This module manages attributes of link aggregation groups on Arista EOS + devices. +version_added: 1.0.0 +author: Nathaniel Case (@Qalthos) +notes: +- Tested against Arista EOS 4.20.10M +- This module works with connection C(network_cli). See the L(EOS Platform Options,../network/user_guide/platform_eos.html). +options: + config: + description: A list of link aggregation group configurations. + type: list + elements: dict + suboptions: + name: + description: + - Name of the port-channel interface of the link aggregation group (LAG) e.g., + Port-Channel5. + type: str + required: true + members: + description: + - Ethernet interfaces that are part of the group. + type: list + elements: dict + suboptions: + member: + description: + - Name of ethernet interface that is a member of the LAG. + type: str + mode: + description: + - LAG mode for this interface. + type: str + choices: + - 'active' + - 'on' + - 'passive' + running_config: + description: + - This option is used only with state I(parsed). + - The value of this option should be the output received from the EOS device by + executing the command B(show running-config | section interfaces). + - The state I(parsed) reads the configuration from C(running_config) option and + transforms it into Ansible structured data as per the resource module's argspec + and the value is then returned in the I(parsed) key within the result. + type: str + state: + description: + - The state of the configuration after module completion. + type: str + choices: + - merged + - replaced + - overridden + - deleted + - rendered + - gathered + - parsed + default: merged + +""" + +EXAMPLES = """ + +# Using merged + +# Before state: +# ------------- +# +# veos#show running-config | section interface +# interface Ethernet1 +# channel-group 5 mode on +# interface Ethernet2 + +- name: Merge provided LAG attributes with existing device configuration + arista.eos.eos_lag_interfaces: + config: + - name: 5 + members: + - member: Ethernet2 + mode: on + state: merged + +# After state: +# ------------ +# +# veos#show running-config | section interface +# interface Ethernet1 +# channel-group 5 mode on +# interface Ethernet2 +# channel-group 5 mode on + + +# Using replaced + +# Before state: +# ------------- +# +# veos#show running-config | section interface +# interface Ethernet1 +# channel-group 5 mode on +# interface Ethernet2 + +- name: Replace all device configuration of specified LAGs with provided configuration + arista.eos.eos_lag_interfaces: + config: + - name: 5 + members: + - member: Ethernet2 + mode: on + state: replaced + +# After state: +# ------------ +# +# veos#show running-config | section interface +# interface Ethernet1 +# interface Ethernet2 +# channel-group 5 mode on + + +# Using overridden + +# Before state: +# ------------- +# +# veos#show running-config | section interface +# interface Ethernet1 +# channel-group 5 mode on +# interface Ethernet2 + +- name: Override all device configuration of all LAG attributes with provided configuration + arista.eos.eos_lag_interfaces: + config: + - name: 10 + members: + - member: Ethernet2 + mode: on + state: overridden + +# After state: +# ------------ +# +# veos#show running-config | section interface +# interface Ethernet1 +# interface Ethernet2 +# channel-group 10 mode on + + +# Using deleted + +# Before state: +# ------------- +# +# veos#show running-config | section interface +# interface Ethernet1 +# channel-group 5 mode on +# interface Ethernet2 +# channel-group 5 mode on + +- name: Delete LAG attributes of the given interfaces. + arista.eos.eos_lag_interfaces: + config: + - name: 5 + members: + - member: Ethernet1 + state: deleted + +# After state: +# ------------ +# +# veos#show running-config | section interface +# interface Ethernet1 +# interface Ethernet2 +# channel-group 5 mode on + +# Using parsed: + +# parsed.cfg +# interface Ethernet1 +# channel-group 5 mode on +# interface Ethernet2 +# channel-group 5 mode on + +- name: Use parsed to convert native configs to structured data + arista.eos.lag_interfaces: + running_config: "{{ lookup('file', 'parsed.cfg') }}" + state: parsed + +# Output: +# parsed: +# - name: 5 +# members: +# - member: Ethernet2 +# mode: on +# - member: Ethernet1 +# mode: on + +# using rendered: + +- name: Use Rendered to convert the structured data to native config + arista.eos.eos_lag_interfaces: + config: + - name: 5 + members: + - member: Ethernet2 + mode: on + - member: Ethernet1 + mode: on + state: rendered +# ----------- +# Output +# ----------- +# +# rendered: + +# interface Ethernet1 +# channel-group 5 mode on +# interface Ethernet2 +# channel-group 5 mode on + + +# Using gathered: + +# native config: +# interface Ethernet1 +# channel-group 5 mode on +# interface Ethernet2 +# channel-group 5 mode on + +- name: Gather lldp_global facts from the device + arista.eos.lldp_global: + state: gathered + +# Output: +# gathered: +# - name: 5 +# members: +# - member: Ethernet2 +# mode: on +# - member: Ethernet1 +# mode: on + +""" + +RETURN = """ +before: + description: The configuration as structured data prior to module invocation. + returned: always + type: list + sample: > + The configuration returned will always be in the same format + of the parameters above. +after: + description: The configuration as structured data after module completion. + returned: when changed + type: list + sample: > + The configuration returned will always be in the same format + of the parameters above. +commands: + description: The set of commands pushed to the remote device. + returned: always + type: list + sample: ['command 1', 'command 2', 'command 3'] +""" + + +from ansible.module_utils.basic import AnsibleModule +from ansible_collections.arista.eos.plugins.module_utils.network.eos.argspec.lag_interfaces.lag_interfaces import ( + Lag_interfacesArgs, +) +from ansible_collections.arista.eos.plugins.module_utils.network.eos.config.lag_interfaces.lag_interfaces import ( + Lag_interfaces, +) + + +def main(): + """ + Main entry point for module execution + + :returns: the result form module invocation + """ + required_if = [ + ("state", "merged", ("config",)), + ("state", "replaced", ("config",)), + ("state", "overridden", ("config",)), + ("state", "rendered", ("config",)), + ("state", "parsed", ("running_config",)), + ] + mutually_exclusive = [("config", "running_config")] + module = AnsibleModule( + argument_spec=Lag_interfacesArgs.argument_spec, + required_if=required_if, + supports_check_mode=True, + mutually_exclusive=mutually_exclusive, + ) + + result = Lag_interfaces(module).execute_module() + module.exit_json(**result) + + +if __name__ == "__main__": + main() diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/modules/eos_linkagg.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/modules/eos_linkagg.py new file mode 100644 index 00000000..32d289a1 --- /dev/null +++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/modules/eos_linkagg.py @@ -0,0 +1,432 @@ +#!/usr/bin/python +# -*- coding: utf-8 -*- + +# Copyright: (c) 2017, Ansible by Red Hat, inc +# 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 + + +DOCUMENTATION = """ +module: eos_linkagg +author: Trishna Guha (@trishnaguha) +short_description: (deprecated, removed after 2022-06-01) Manage link aggregation + groups on Arista EOS network devices +description: +- This module provides declarative management of link aggregation groups on Arista + EOS network devices. +version_added: 1.0.0 +deprecated: + alternative: eos_lag_interfaces + why: Updated modules released with more functionality + removed_at_date: '2022-06-01' +notes: +- Tested against EOS 4.15 +options: + group: + description: + - Channel-group number for the port-channel Link aggregation group. Range 1-2000. + type: int + mode: + description: + - Mode of the link aggregation group. + type: str + choices: + - active + - "on" + - passive + members: + description: + - List of members of the link aggregation group. + type: list + elements: str + min_links: + description: + - Minimum number of ports required up before bringing up the link aggregation + group. + type: int + aggregate: + description: List of link aggregation definitions. + type: list + elements: dict + suboptions: + group: + description: + - Channel-group number for the port-channel Link aggregation group. Range 1-2000. + type: int + required: True + mode: + description: + - Mode of the link aggregation group. + type: str + choices: + - active + - "on" + - passive + members: + description: + - List of members of the link aggregation group. + type: list + elements: str + min_links: + description: + - Minimum number of ports required up before bringing up the link aggregation + group. + type: int + state: + description: + - State of the link aggregation group. + type: str + choices: + - present + - absent + state: + description: + - State of the link aggregation group. + default: present + type: str + choices: + - present + - absent + purge: + description: + - Purge links not defined in the I(aggregate) parameter. + default: false + type: bool +extends_documentation_fragment: +- arista.eos.eos + +""" + +EXAMPLES = """ +- name: create link aggregation group + arista.eos.eos_linkagg: + group: 10 + state: present + +- name: delete link aggregation group + arista.eos.eos_linkagg: + group: 10 + state: absent + +- name: set link aggregation group to members + arista.eos.eos_linkagg: + group: 200 + min_links: 3 + mode: active + members: + - Ethernet0 + - Ethernet1 + +- name: remove link aggregation group from Ethernet0 + arista.eos.eos_linkagg: + group: 200 + min_links: 3 + mode: active + members: + - Ethernet1 + +- name: Create aggregate of linkagg definitions + arista.eos.eos_linkagg: + aggregate: + - {group: 3, mode: on, members: [Ethernet1]} + - {group: 100, mode: passive, min_links: 3, members: [Ethernet2]} + +- name: Remove aggregate of linkagg definitions + arista.eos.eos_linkagg: + aggregate: + - {group: 3, mode: on, members: [Ethernet1]} + - {group: 100, mode: passive, min_links: 3, members: [Ethernet2]} + state: absent +""" + +RETURN = """ +commands: + description: The list of configuration mode commands to send to the device + returned: always, except for the platforms that use Netconf transport to manage the device. + type: list + sample: + - interface port-channel 30 + - port-channel min-links 5 + - interface Ethernet3 + - channel-group 30 mode on + - no interface port-channel 30 +""" + +import re +from copy import deepcopy + +from ansible.module_utils.basic import AnsibleModule +from ansible_collections.ansible.netcommon.plugins.module_utils.network.common.utils import ( + remove_default_spec, +) +from ansible_collections.arista.eos.plugins.module_utils.network.eos.eos import ( + get_config, + load_config, +) +from ansible_collections.arista.eos.plugins.module_utils.network.eos.eos import ( + eos_argument_spec, +) + + +def search_obj_in_list(group, lst): + for o in lst: + if o["group"] == group: + return o + + +def map_obj_to_commands(updates, module): + commands = list() + want, have = updates + purge = module.params["purge"] + + for w in want: + group = w["group"] + mode = w["mode"] + min_links = w["min_links"] + members = w.get("members") or [] + state = w["state"] + del w["state"] + + obj_in_have = search_obj_in_list(group, have) + + if state == "absent": + if obj_in_have: + commands.append("no interface port-channel {0}".format(group)) + + elif state == "present": + cmd = ["interface port-channel {0}".format(group), "end"] + if not obj_in_have: + if not group: + module.fail_json(msg="group is a required option") + commands.extend(cmd) + + if min_links != "None": + commands.append( + "port-channel min-links {0}".format(min_links) + ) + + if members: + for m in members: + commands.append("interface {0}".format(m)) + commands.append( + "channel-group {0} mode {1}".format(group, mode) + ) + + else: + if members: + if "members" not in obj_in_have.keys(): + for m in members: + commands.extend(cmd) + commands.append("interface {0}".format(m)) + commands.append( + "channel-group {0} mode {1}".format( + group, mode + ) + ) + + elif set(members) != set(obj_in_have["members"]): + missing_members = list( + set(members) - set(obj_in_have["members"]) + ) + for m in missing_members: + commands.extend(cmd) + commands.append("interface {0}".format(m)) + commands.append( + "channel-group {0} mode {1}".format( + group, mode + ) + ) + + superfluous_members = list( + set(obj_in_have["members"]) - set(members) + ) + for m in superfluous_members: + commands.extend(cmd) + commands.append("interface {0}".format(m)) + commands.append( + "no channel-group {0}".format(group) + ) + + if purge: + for h in have: + obj_in_want = search_obj_in_list(h["group"], want) + if not obj_in_want: + commands.append( + "no interface port-channel {0}".format(h["group"]) + ) + + return commands + + +def map_params_to_obj(module, required_together=None): + obj = [] + + aggregate = module.params.get("aggregate") + if aggregate: + for item in aggregate: + for key in item: + if item.get(key) is None: + item[key] = module.params[key] + + module._check_required_together(required_together, item) + d = item.copy() + d["group"] = str(d["group"]) + d["min_links"] = str(d["min_links"]) + + obj.append(d) + else: + obj.append( + { + "group": str(module.params["group"]), + "mode": module.params["mode"], + "min_links": str(module.params["min_links"]), + "members": module.params["members"], + "state": module.params["state"], + } + ) + + return obj + + +def parse_mode(group, member, config): + mode = None + + for line in config.strip().split("!"): + match_int = re.findall(r"interface {0}\\b".format(member), line, re.M) + if match_int: + match = re.search( + r"channel-group {0} mode (\S+)".format(group), line, re.M + ) + if match: + mode = match.group(1) + + return mode + + +def parse_members(group, config): + members = [] + + for line in config.strip().split("!"): + match_group = re.findall( + r"channel-group {0} mode".format(group), line, re.M + ) + if match_group: + match = re.search(r"interface (\S+)", line, re.M) + if match: + members.append(match.group(1)) + + return members + + +def get_channel(group, module): + channel = {} + config = get_config(module, flags=["| section channel-group"]) + + for line in config.split("\n"): + stripped = line.strip() + match = re.search(r"interface (\S+)", stripped, re.M) + + if match: + member = match.group(1) + channel["mode"] = parse_mode(group, member, config) + channel["members"] = parse_members(group, config) + + return channel + + +def parse_min_links(group, config): + min_links = "" + + for line in config.strip().split("!"): + match_pc = re.findall( + r"interface Port-Channel{0}\\b".format(group), line, re.M + ) + if match_pc: + match = re.search(r"port-channel min-links (\S+)", line, re.M) + if match: + min_links = match.group(1) + + return min_links + + +def map_config_to_obj(module): + objs = list() + config = get_config(module, flags=["| section port-channel"]) + + for line in config.split("\n"): + stripped = line.strip() + match = re.search(r"interface Port-Channel(\S+)", stripped, re.M) + if match: + obj = {} + group = match.group(1) + obj["group"] = group + obj["min_links"] = parse_min_links(group, config) + obj.update(get_channel(group, module)) + objs.append(obj) + + return objs + + +def main(): + """ main entry point for module execution + """ + element_spec = dict( + group=dict(type="int"), + mode=dict(choices=["active", "on", "passive"]), + min_links=dict(type="int"), + members=dict(type="list", elements="str"), + state=dict(default="present", choices=["present", "absent"]), + ) + + aggregate_spec = deepcopy(element_spec) + aggregate_spec["group"] = dict(required=True, type="int") + + # remove default in aggregate spec, to handle common arguments + remove_default_spec(aggregate_spec) + + argument_spec = dict( + aggregate=dict(type="list", elements="dict", options=aggregate_spec), + purge=dict(default=False, type="bool"), + state=dict(default="present", choices=["present", "absent"]), + ) + + argument_spec.update(element_spec) + argument_spec.update(eos_argument_spec) + + required_one_of = [["group", "aggregate"]] + required_together = [["members", "mode"]] + mutually_exclusive = [["group", "aggregate"]] + module = AnsibleModule( + argument_spec=argument_spec, + required_one_of=required_one_of, + required_together=required_together, + mutually_exclusive=mutually_exclusive, + supports_check_mode=True, + ) + + warnings = list() + result = {"changed": False} + if warnings: + result["warnings"] = warnings + + want = map_params_to_obj(module) + have = map_config_to_obj(module) + commands = map_obj_to_commands((want, have), module) + result["commands"] = commands + + if commands: + commit = not module.check_mode + response = load_config(module, commands, commit=commit) + if response.get("diff") and module._diff: + result["diff"] = {"prepared": response.get("diff")} + result["session_name"] = response.get("session") + result["changed"] = True + + module.exit_json(**result) + + +if __name__ == "__main__": + main() diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/modules/eos_lldp.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/modules/eos_lldp.py new file mode 100644 index 00000000..d2e306cf --- /dev/null +++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/modules/eos_lldp.py @@ -0,0 +1,122 @@ +#!/usr/bin/python +# -*- coding: utf-8 -*- + +# (c) 2017, Ansible by Red Hat, inc +# 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 + + +DOCUMENTATION = """ +module: eos_lldp +author: Ganesh Nalawade (@ganeshrn) +short_description: Manage LLDP configuration on Arista EOS network devices +description: +- This module provides declarative management of LLDP service on Arista EOS network + devices. +version_added: 1.0.0 +notes: +- Tested against EOS 4.15 +options: + state: + description: + - State of the LLDP configuration. If value is I(present) lldp will be enabled + else if it is I(absent) it will be disabled. + default: present + type: str + choices: + - present + - absent + - enabled + - disabled +extends_documentation_fragment: +- arista.eos.eos +""" + +EXAMPLES = """ +- name: Enable LLDP service + arista.eos.eos_lldp: + state: present + +- name: Disable LLDP service + arista.eos.eos_lldp: + state: absent +""" + +RETURN = """ +commands: + description: The list of configuration mode commands to send to the device + returned: always, except for the platforms that use Netconf transport to manage the device. + type: list + sample: + - lldp run +""" +from ansible.module_utils.basic import AnsibleModule +from ansible_collections.arista.eos.plugins.module_utils.network.eos.eos import ( + get_config, + load_config, +) +from ansible_collections.arista.eos.plugins.module_utils.network.eos.eos import ( + eos_argument_spec, +) + + +def has_lldp(module): + config = get_config(module, flags=["| section lldp"]) + + is_lldp_enable = False + if "no lldp run" not in config: + is_lldp_enable = True + + return is_lldp_enable + + +def main(): + """ main entry point for module execution + """ + argument_spec = dict( + state=dict( + default="present", + choices=["present", "absent", "enabled", "disabled"], + ) + ) + + argument_spec.update(eos_argument_spec) + + module = AnsibleModule( + argument_spec=argument_spec, supports_check_mode=True + ) + + warnings = list() + + result = {"changed": False} + + if warnings: + result["warnings"] = warnings + + HAS_LLDP = has_lldp(module) + + commands = [] + + if module.params["state"] == "absent" and HAS_LLDP: + commands.append("no lldp run") + elif module.params["state"] == "present" and not HAS_LLDP: + commands.append("lldp run") + + result["commands"] = commands + + if commands: + commit = not module.check_mode + response = load_config(module, commands, commit=commit) + if response.get("diff") and module._diff: + result["diff"] = {"prepared": response.get("diff")} + result["session_name"] = response.get("session") + result["changed"] = True + + module.exit_json(**result) + + +if __name__ == "__main__": + main() diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/modules/eos_lldp_global.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/modules/eos_lldp_global.py new file mode 100644 index 00000000..4ad4bf4b --- /dev/null +++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/modules/eos_lldp_global.py @@ -0,0 +1,345 @@ +#!/usr/bin/python +# -*- coding: utf-8 -*- +# Copyright 2019 Red Hat +# GNU General Public License v3.0+ +# (see COPYING or https://www.gnu.org/licenses/gpl-3.0.txt) + +############################################# +# WARNING # +############################################# +# +# This file is auto generated by the resource +# module builder playbook. +# +# Do not edit this file manually. +# +# Changes to this file will be over written +# by the resource module builder. +# +# Changes should be made in the model used to +# generate this file or in the resource module +# builder template. +# +############################################# + +""" +The module file for eos_lldp_global +""" + +from __future__ import absolute_import, division, print_function + +__metaclass__ = type + + +DOCUMENTATION = """ +module: eos_lldp_global +short_description: LLDP resource module +description: +- This module manages Global Link Layer Discovery Protocol (LLDP) settings on Arista + EOS devices. +version_added: 1.0.0 +author: Nathaniel Case (@Qalthos) +notes: +- Tested against Arista EOS 4.20.10M +- This module works with connection C(network_cli). See the L(EOS Platform Options,../network/user_guide/platform_eos.html). +options: + config: + description: The provided global LLDP configuration. + type: dict + suboptions: + holdtime: + description: + - Specifies the holdtime (in sec) to be sent in packets. + type: int + reinit: + description: + - Specifies the delay (in sec) for LLDP initialization on any interface. + type: int + timer: + description: + - Specifies the rate at which LLDP packets are sent (in sec). + type: int + tlv_select: + description: + - Specifies the LLDP TLVs to enable or disable. + type: dict + suboptions: + link_aggregation: + description: + - Enable or disable link aggregation TLV. + type: bool + management_address: + description: + - Enable or disable management address TLV. + type: bool + max_frame_size: + description: + - Enable or disable maximum frame size TLV. + type: bool + port_description: + description: + - Enable or disable port description TLV. + type: bool + system_capabilities: + description: + - Enable or disable system capabilities TLV. + type: bool + system_description: + description: + - Enable or disable system description TLV. + type: bool + system_name: + description: + - Enable or disable system name TLV. + type: bool + running_config: + description: + - This option is used only with state I(parsed). + - The value of this option should be the output received from the EOS device by + executing the command B(show running-config | section lldp). + - The state I(parsed) reads the configuration from C(running_config) option and + transforms it into Ansible structured data as per the resource module's argspec + and the value is then returned in the I(parsed) key within the result. + type: str + state: + description: + - The state of the configuration after module completion. + type: str + choices: + - merged + - replaced + - deleted + - rendered + - gathered + - parsed + default: merged + +""" +EXAMPLES = """ +# Using merged +# +# ------------ +# Before State +# ------------ +# +# veos# show run | section lldp +# lldp timer 3000 +# lldp holdtime 100 +# lldp reinit 5 +# no lldp tlv-select management-address +# no lldp tlv-select system-description + +- name: Merge provided LLDP configuration with the existing configuration + arista.eos.eos_lldp_global: + config: + holdtime: 100 + tlv_select: + management_address: false + port_description: false + system_description: true + state: merged + +# ----------- +# After state +# ----------- +# +# veos# show run | section lldp +# lldp timer 3000 +# lldp holdtime 100 +# lldp reinit 5 +# no lldp tlv-select management-address +# no lldp tlv-select port-description + + +# Using replaced +# +# ------------ +# Before State +# ------------ +# +# veos# show run | section lldp +# lldp timer 3000 +# lldp holdtime 100 +# lldp reinit 5 +# no lldp tlv-select management-address +# no lldp tlv-select system-description + +- name: Replace existing LLDP device configuration with provided configuration + arista.eos.eos_lldp_global: + config: + holdtime: 100 + tlv_select: + management_address: false + port_description: false + system_description: true + state: replaced + +# ----------- +# After state +# ----------- +# +# veos# show run | section lldp +# lldp holdtime 100 +# no lldp tlv-select management-address +# no lldp tlv-select port-description + + +# Using deleted +# +# ------------ +# Before State +# ------------ +# +# veos# show run | section lldp +# lldp timer 3000 +# lldp holdtime 100 +# lldp reinit 5 +# no lldp tlv-select management-address +# no lldp tlv-select system-description + +- name: Delete existing LLDP configurations from the device + arista.eos.eos_lldp_global: + state: deleted + +# ----------- +# After state +# ----------- +# +# veos# show run | section ^lldp + +# Using rendered: + +- name: Use Rendered to convert the structured data to native config + arista.eos.eos_lldp_global: + config: + holdtime: 100 + tlv_select: + management_address: false + port_description: false + system_description: true + state: rendered + +# ----------- +# Output +# ----------- +# +# rendered: +# - "lldp holdtime 100" +# - "no lldp tlv-select management-address" +# - "no lldp tlv-select port-description" + +# Using parsed + +# parsed.cfg + +# lldp timer 3000 +# lldp holdtime 100 +# lldp reinit 5 +# no lldp tlv-select management-address +# no lldp tlv-select system-description + +- name: Use parsed to convert native configs to structured data + arista.eos.lldp_global: + running_config: "{{ lookup('file', 'parsed.cfg') }}" + state: parsed + +# ----------- +# Output +# ----------- + +# parsed: +# holdtime: 100 +# timer 3000 +# reinit 5 +# tlv_select: +# management_address: False +# port_description: False +# system_description: True + +# Using gathered: +# native config: +# lldp timer 3000 +# lldp holdtime 100 +# lldp reinit 5 +# no lldp tlv-select management-address +# no lldp tlv-select system-description + + +- name: Gather lldp_global facts from the device + arista.eos.lldp_global: + state: gathered + +# ----------- +# Output +# ----------- + +# gathered: +# holdtime: 100 +# timer 3000 +# reinit 5 +# tlv_select: +# management_address: False +# port_description: False +# system_description: True + +""" +RETURN = """ +before: + description: The configuration as structured data prior to module invocation. + returned: always + type: dict + sample: > + The configuration returned will always be in the same format + of the parameters above. +after: + description: The configuration as structured data after module completion. + returned: when changed + type: dict + sample: > + The configuration returned will always be in the same format + of the parameters above. +commands: + description: The set of commands pushed to the remote device. + returned: always + type: list + sample: ['lldp holdtime 100', 'no lldp timer', 'lldp tlv-select system-description'] +""" + + +from ansible.module_utils.basic import AnsibleModule +from ansible_collections.arista.eos.plugins.module_utils.network.eos.argspec.lldp_global.lldp_global import ( + Lldp_globalArgs, +) +from ansible_collections.arista.eos.plugins.module_utils.network.eos.config.lldp_global.lldp_global import ( + Lldp_global, +) + + +def main(): + """ + Main entry point for module execution + + :returns: the result form module invocation + """ + + required_if = [ + ("state", "merged", ("config",)), + ("state", "replaced", ("config",)), + ("state", "overridden", ("config",)), + ("state", "rendered", ("config",)), + ("state", "parsed", ("running_config",)), + ] + mutually_exclusive = [("config", "running_config")] + module = AnsibleModule( + argument_spec=Lldp_globalArgs.argument_spec, + required_if=required_if, + supports_check_mode=True, + mutually_exclusive=mutually_exclusive, + ) + + result = Lldp_global(module).execute_module() + module.exit_json(**result) + + +if __name__ == "__main__": + main() diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/modules/eos_lldp_interfaces.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/modules/eos_lldp_interfaces.py new file mode 100644 index 00000000..98a24913 --- /dev/null +++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/modules/eos_lldp_interfaces.py @@ -0,0 +1,344 @@ +#!/usr/bin/python +# -*- coding: utf-8 -*- +# Copyright 2019 Red Hat +# GNU General Public License v3.0+ +# (see COPYING or https://www.gnu.org/licenses/gpl-3.0.txt) + +############################################# +# WARNING # +############################################# +# +# This file is auto generated by the resource +# module builder playbook. +# +# Do not edit this file manually. +# +# Changes to this file will be over written +# by the resource module builder. +# +# Changes should be made in the model used to +# generate this file or in the resource module +# builder template. +# +############################################# + +""" +The module file for eos_lldp_interfaces +""" + +from __future__ import absolute_import, division, print_function + +__metaclass__ = type + + +DOCUMENTATION = """ +module: eos_lldp_interfaces +short_description: LLDP interfaces resource module +description: +- This module manages Link Layer Discovery Protocol (LLDP) attributes of interfaces + on Arista EOS devices. +version_added: 1.0.0 +author: Nathaniel Case (@Qalthos) +notes: +- Tested against Arista EOS 4.20.10M +- This module works with connection C(network_cli). See the L(EOS Platform Options,../network/user_guide/platform_eos.html). +options: + config: + description: A dictionary of LLDP interfaces options. + type: list + elements: dict + suboptions: + name: + description: + - Full name of the interface (i.e. Ethernet1). + type: str + receive: + description: + - Enable/disable LLDP RX on an interface. + type: bool + transmit: + description: + - Enable/disable LLDP TX on an interface. + type: bool + running_config: + description: + - This option is used only with state I(parsed). + - The value of this option should be the output received from the EOS device by + executing the command B(show running-config | section ^interface). + - The state I(parsed) reads the configuration from C(running_config) option and + transforms it into Ansible structured data as per the resource module's argspec + and the value is then returned in the I(parsed) key within the result. + type: str + state: + description: + - The state of the configuration after module completion. + type: str + choices: + - merged + - replaced + - overridden + - deleted + - parsed + - gathered + - rendered + default: merged + +""" +EXAMPLES = """ +# Using merged +# +# +# ------------ +# Before state +# ------------ +# +# +# veos#show run | section ^interface +# interface Ethernet1 +# no lldp receive +# interface Ethernet2 +# no lldp transmit + +- name: Merge provided configuration with running configuration + arista.eos.eos_lldp_interfaces: + config: + - name: Ethernet1 + transmit: false + - name: Ethernet2 + transmit: false + state: merged + +# +# ------------ +# After state +# ------------ +# +# veos#show run | section ^interface +# interface Ethernet1 +# no lldp transmit +# no lldp receive +# interface Ethernet2 +# no lldp transmit + + +# Using replaced +# +# +# ------------ +# Before state +# ------------ +# +# +# veos#show run | section ^interface +# interface Ethernet1 +# no lldp receive +# interface Ethernet2 +# no lldp transmit + +- name: Replace existing LLDP configuration of specified interfaces with provided + configuration + arista.eos.eos_lldp_interfaces: + config: + - name: Ethernet1 + transmit: false + state: replaced + +# +# ------------ +# After state +# ------------ +# +# veos#show run | section ^interface +# interface Ethernet1 +# no lldp transmit +# interface Ethernet2 +# no lldp transmit + + +# Using overridden +# +# +# ------------ +# Before state +# ------------ +# +# +# veos#show run | section ^interface +# interface Ethernet1 +# no lldp receive +# interface Ethernet2 +# no lldp transmit + +- name: Override the LLDP configuration of all the interfaces with provided configuration + arista.eos.eos_lldp_interfaces: + config: + - name: Ethernet1 + transmit: false + state: overridden + +# +# ------------ +# After state +# ------------ +# +# veos#show run | section ^interface +# interface Ethernet1 +# no lldp transmit +# interface Ethernet2 + + +# Using deleted +# +# +# ------------ +# Before state +# ------------ +# +# +# veos#show run | section ^interface +# interface Ethernet1 +# no lldp receive +# interface Ethernet2 +# no lldp transmit + +- name: Delete LLDP configuration of specified interfaces (or all interfaces if none + are specified) + arista.eos.eos_lldp_interfaces: + state: deleted + +# +# ------------ +# After state +# ------------ +# +# veos#show run | section ^interface +# interface Ethernet1 +# interface Ethernet2 + +# using rendered: + +- name: Use Rendered to convert the structured data to native config + arista.eos.eos_lldp_interfaces: + config: + - name: Ethernet1 + transmit: false + - name: Ethernet2 + transmit: false + state: rendered + +# +# ------------ +# Output +# ------------ +# +# interface Ethernet1 +# no lldp transmit +# interface Ethernet2 +# no lldp transmit + +# Using parsed +# parsed.cfg + +# interface Ethernet1 +# no lldp transmit +# interface Ethernet2 +# no lldp transmit + + +- name: Use parsed to convert native configs to structured data + arista.eos.lldp_interfaces: + running_config: "{{ lookup('file', 'parsed.cfg') }}" + state: parsed + +# ------------ +# Output +# ------------ + +# parsed: +# - name: Ethernet1 +# transmit: False +# - name: Ethernet2 +# transmit: False + +# Using gathered: + +# native config: +# interface Ethernet1 +# no lldp transmit +# interface Ethernet2 +# no lldp transmit + +- name: Gather lldp interfaces facts from the device + arista.eos.lldp_interfaces: + state: gathered + +# ------------ +# Output +# ------------ + +# gathered: +# - name: Ethernet1 +# transmit: False +# - name: Ethernet2 +# transmit: False + +""" +RETURN = """ +before: + description: The configuration as structured data prior to module invocation. + returned: always + type: list + sample: > + The configuration returned will always be in the same format + of the parameters above. +after: + description: The configuration as structured data after module completion. + returned: when changed + type: list + sample: > + The configuration returned will always be in the same format + of the parameters above. +commands: + description: The set of commands pushed to the remote device. + returned: always + type: list + sample: ['interface Ethernet1', 'no lldp transmit'] +""" + + +from ansible.module_utils.basic import AnsibleModule +from ansible_collections.arista.eos.plugins.module_utils.network.eos.argspec.lldp_interfaces.lldp_interfaces import ( + Lldp_interfacesArgs, +) +from ansible_collections.arista.eos.plugins.module_utils.network.eos.config.lldp_interfaces.lldp_interfaces import ( + Lldp_interfaces, +) + + +def main(): + """ + Main entry point for module execution + + :returns: the result form module invocation + """ + required_if = [ + ("state", "merged", ("config",)), + ("state", "replaced", ("config",)), + ("state", "overridden", ("config",)), + ("state", "rendered", ("config",)), + ("state", "parsed", ("running_config",)), + ] + mutually_exclusive = [("config", "running_config")] + module = AnsibleModule( + argument_spec=Lldp_interfacesArgs.argument_spec, + required_if=required_if, + supports_check_mode=True, + mutually_exclusive=mutually_exclusive, + ) + + result = Lldp_interfaces(module).execute_module() + module.exit_json(**result) + + +if __name__ == "__main__": + main() diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/modules/eos_logging.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/modules/eos_logging.py new file mode 100644 index 00000000..a03cf7ef --- /dev/null +++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/modules/eos_logging.py @@ -0,0 +1,505 @@ +#!/usr/bin/python +# -*- coding: utf-8 -*- +from __future__ import absolute_import, division, print_function + +__metaclass__ = type + +# Copyright: (c) 2017, Ansible by Red Hat, inc +# GNU General Public License v3.0+ (see COPYING or https://www.gnu.org/licenses/gpl-3.0.txt) + + +DOCUMENTATION = """ +module: eos_logging +author: Trishna Guha (@trishnaguha) +short_description: Manage logging on network devices +description: +- This module provides declarative management of logging on Arista Eos devices. +version_added: 1.0.0 +notes: +- Tested against EOS 4.15 +options: + dest: + description: + - Destination of the logs. + choices: + - "on" + - host + - console + - monitor + - buffered + type: str + name: + description: + - The hostname or IP address of the destination. + - Required when I(dest=host). + type: str + size: + description: + - Size of buffer. The acceptable value is in range from 10 to 2147483647 bytes. + type: int + facility: + description: + - Set logging facility. + type: str + level: + description: + - Set logging severity levels. + choices: + - emergencies + - alerts + - critical + - errors + - warnings + - notifications + - informational + - debugging + type: str + aggregate: + description: List of logging definitions. + type: list + elements: dict + suboptions: + dest: + description: + - Destination of the logs. + choices: + - "on" + - host + - console + - monitor + - buffered + type: str + name: + description: + - The hostname or IP address of the destination. + - Required when I(dest=host). + type: str + size: + description: + - Size of buffer. The acceptable value is in range from 10 to 2147483647 bytes. + type: int + facility: + description: + - Set logging facility. + type: str + level: + description: + - Set logging severity levels. + choices: + - emergencies + - alerts + - critical + - errors + - warnings + - notifications + - informational + - debugging + type: str + state: + description: + - State of the logging configuration. + default: present + type: str + choices: + - present + - absent + state: + description: + - State of the logging configuration. + default: present + type: str + choices: + - present + - absent +extends_documentation_fragment: +- arista.eos.eos +""" + +EXAMPLES = """ +- name: configure host logging + arista.eos.eos_logging: + dest: host + name: 172.16.0.1 + state: present + +- name: remove host logging configuration + arista.eos.eos_logging: + dest: host + name: 172.16.0.1 + state: absent + +- name: configure console logging level and facility + arista.eos.eos_logging: + dest: console + facility: local7 + level: debugging + state: present + +- name: enable logging to all + arista.eos.eos_logging: + dest: on + +- name: configure buffer size + arista.eos.eos_logging: + dest: buffered + size: 5000 + +- name: Configure logging using aggregate + arista.eos.eos_logging: + aggregate: + - {dest: console, level: warnings} + - {dest: buffered, size: 480000} + state: present +""" + +RETURN = """ +commands: + description: The list of configuration mode commands to send to the device + returned: always + type: list + sample: + - logging facility local7 + - logging host 172.16.0.1 +""" + +import re + + +from copy import deepcopy +from ansible.module_utils.basic import AnsibleModule +from ansible_collections.ansible.netcommon.plugins.module_utils.network.common.utils import ( + remove_default_spec, +) +from ansible_collections.arista.eos.plugins.module_utils.network.eos.eos import ( + get_config, + load_config, +) +from ansible_collections.arista.eos.plugins.module_utils.network.eos.eos import ( + eos_argument_spec, +) + + +DEST_GROUP = ["on", "host", "console", "monitor", "buffered"] +LEVEL_GROUP = [ + "emergencies", + "alerts", + "critical", + "errors", + "warnings", + "notifications", + "informational", + "debugging", +] + + +def validate_size(value, module): + if value: + if not int(10) <= value <= int(2147483647): + module.fail_json(msg="size must be between 10 and 2147483647") + else: + return value + + +def map_obj_to_commands(updates, module): + commands = list() + want, have = updates + + for w in want: + dest = w["dest"] + name = w["name"] + size = w["size"] + facility = w["facility"] + level = w["level"] + state = w["state"] + del w["state"] + + if state == "absent" and w in have: + if dest: + if dest == "host": + commands.append("no logging host {0}".format(name)) + + elif dest in DEST_GROUP: + commands.append("no logging {0}".format(dest)) + + else: + module.fail_json( + msg="dest must be among console, monitor, buffered, host, on" + ) + + if facility: + commands.append("no logging facility {0}".format(facility)) + + if state == "present" and w not in have: + if facility: + present = False + + # Iterate over every dictionary in the 'have' list to check if + # similar configuration for facility exists or not + + for entry in have: + if not entry["dest"] and entry["facility"] == facility: + present = True + + if not present: + commands.append("logging facility {0}".format(facility)) + + if dest == "host": + commands.append("logging host {0}".format(name)) + + elif dest == "on": + commands.append("logging on") + + elif dest == "buffered" and size: + + present = False + + # Deals with the following two cases: + # Case 1: logging buffered <size> <level> + # logging buffered <same-size> + # + # Case 2: Same buffered logging configuration + # already exists (i.e., both size & + # level are same) + + for entry in have: + if entry["dest"] == "buffered" and entry["size"] == size: + + if not level or entry["level"] == level: + present = True + + if not present: + if size and level: + commands.append( + "logging buffered {0} {1}".format(size, level) + ) + else: + commands.append("logging buffered {0}".format(size)) + + else: + if dest: + dest_cmd = "logging {0}".format(dest) + if level: + dest_cmd += " {0}".format(level) + + commands.append(dest_cmd) + return commands + + +def parse_facility(line): + facility = None + match = re.search(r"logging facility (\S+)", line, re.M) + if match: + facility = match.group(1) + + return facility + + +def parse_size(line, dest): + size = None + + if dest == "buffered": + match = re.search(r"logging buffered (\S+)", line, re.M) + if match: + try: + int_size = int(match.group(1)) + except ValueError: + int_size = None + + if int_size: + if isinstance(int_size, int): + size = str(match.group(1)) + else: + size = str(10) + + return size + + +def parse_name(line, dest): + name = None + if dest == "host": + match = re.search(r"logging host (\S+)", line, re.M) + if match: + name = match.group(1) + + return name + + +def parse_level(line, dest): + level = None + + if dest != "host": + + # Line for buffer logging entry in running-config is of the form: + # logging buffered <size> <level> + + if dest == "buffered": + match = re.search(r"logging buffered (?:\d+) (\S+)", line, re.M) + + else: + match = re.search(r"logging {0} (\S+)".format(dest), line, re.M) + + if match: + if match.group(1) in LEVEL_GROUP: + level = match.group(1) + + return level + + +def map_config_to_obj(module): + obj = [] + + data = get_config(module, flags=["section logging"]) + + for line in data.split("\n"): + + match = re.search(r"logging (\S+)", line, re.M) + + if match: + if match.group(1) in DEST_GROUP: + dest = match.group(1) + + else: + dest = None + + obj.append( + { + "dest": dest, + "name": parse_name(line, dest), + "size": parse_size(line, dest), + "facility": parse_facility(line), + "level": parse_level(line, dest), + } + ) + + return obj + + +def parse_obj(obj, module): + if module.params["size"] is None: + obj.append( + { + "dest": module.params["dest"], + "name": module.params["name"], + "size": module.params["size"], + "facility": module.params["facility"], + "level": module.params["level"], + "state": module.params["state"], + } + ) + + else: + obj.append( + { + "dest": module.params["dest"], + "name": module.params["name"], + "size": str(validate_size(module.params["size"], module)), + "facility": module.params["facility"], + "level": module.params["level"], + "state": module.params["state"], + } + ) + + return obj + + +def map_params_to_obj(module, required_if=None): + obj = [] + aggregate = module.params.get("aggregate") + if aggregate: + for item in aggregate: + for key in item: + if item.get(key) is None: + item[key] = module.params[key] + + module._check_required_if(required_if, item) + d = item.copy() + + if d["dest"] != "host": + d["name"] = None + + if d["dest"] == "buffered": + if "size" in d: + d["size"] = str(validate_size(d["size"], module)) + elif "size" not in d: + d["size"] = str(10) + else: + pass + + if d["dest"] != "buffered": + d["size"] = None + + obj.append(d) + + else: + if module.params["dest"] != "host": + module.params["name"] = None + + if module.params["dest"] == "buffered": + if not module.params["size"]: + module.params["size"] = str(10) + else: + module.params["size"] = None + + parse_obj(obj, module) + + return obj + + +def main(): + """ main entry point for module execution + """ + element_spec = dict( + dest=dict(choices=DEST_GROUP), + name=dict(), + size=dict(type="int"), + facility=dict(), + level=dict(choices=LEVEL_GROUP), + state=dict(default="present", choices=["present", "absent"]), + ) + + aggregate_spec = deepcopy(element_spec) + + # remove default in aggregate spec, to handle common arguments + remove_default_spec(aggregate_spec) + + aggregate_spec["state"].update(default="present") + argument_spec = dict( + aggregate=dict(type="list", elements="dict", options=aggregate_spec) + ) + + argument_spec.update(element_spec) + argument_spec.update(eos_argument_spec) + + required_if = [("dest", "host", ["name"])] + + module = AnsibleModule( + argument_spec=argument_spec, + required_if=required_if, + supports_check_mode=True, + ) + + warnings = list() + + result = {"changed": False} + if warnings: + result["warnings"] = warnings + + have = map_config_to_obj(module) + want = map_params_to_obj(module, required_if=required_if) + + commands = map_obj_to_commands((want, have), module) + result["commands"] = commands + + if commands: + commit = not module.check_mode + response = load_config(module, commands, commit=commit) + if response.get("diff") and module._diff: + result["diff"] = {"prepared": response.get("diff")} + result["session_name"] = response.get("session") + result["changed"] = True + + module.exit_json(**result) + + +if __name__ == "__main__": + main() diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/modules/eos_ospf_interfaces.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/modules/eos_ospf_interfaces.py new file mode 100644 index 00000000..64226223 --- /dev/null +++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/modules/eos_ospf_interfaces.py @@ -0,0 +1,1228 @@ +#!/usr/bin/python +# -*- coding: utf-8 -*- +# Copyright 2020 Red Hat +# GNU General Public License v3.0+ +# (see COPYING or https://www.gnu.org/licenses/gpl-3.0.txt) + +############################################# +# WARNING # +############################################# +# +# This file is auto generated by the resource +# module builder playbook. +# +# Do not edit this file manually. +# +# Changes to this file will be over written +# by the resource module builder. +# +# Changes should be made in the model used to +# generate this file or in the resource module +# builder template. +# +############################################# + +""" +The module file for eos_ospf_interfaces +""" + +from __future__ import absolute_import, division, print_function + +__metaclass__ = type + +DOCUMENTATION = """ +module: eos_ospf_interfaces +version_added: 1.1.0 +short_description: OSPF Interfaces Resource Module. +description: +- This module manages OSPF configuration of interfaces on devices running Arista EOS. +author: Gomathi Selvi Srinivasan (@GomathiselviS) +options: + config: + description: A list of OSPF configuration for interfaces. + type: list + elements: dict + suboptions: + name: + description: + - Name/Identifier of the interface. + type: str + address_family: + description: + - OSPF settings on the interfaces in address-family context. + type: list + elements: dict + suboptions: + afi: + description: + - Address Family Identifier (AFI) for OSPF settings on the interfaces. + type: str + choices: ['ipv4', 'ipv6'] + required: True + area: + description: + - Area associated with interface. + - Valid only when afi = ipv4. + type: dict + suboptions: + area_id: + description: + - Area ID as a decimal or IP address format. + type: str + required: True + authentication_v2: + description: + - Authentication settings on the interface. + - Valid only when afi = ipv4. + type: dict + suboptions: + message_digest: + description: + - Use message-digest authentication. + type: bool + set: + description: + - Enable authentication on the interface. + type: bool + authentication_v3: + description: + - Authentication settings on the interface. + - Valid only when afi = ipv6. + type: dict + suboptions: + spi: + description: IPsec Security Parameter Index. + type: int + algorithm: + description: Encryption alsgorithm. + type: str + choices: ["md5", "sha1"] + keytype: + description: + - Specifies if an unencrypted/hidden follows. + - 0 denotes unencrypted key. + - 7 denotes hidden key. + type: str + passphrase: + description: Passphrase String for deriving keys for authentication and encryption. + type: str + key: + description: 128 bit MD5 key or 140 bit SHA1 key. + type: str + authentication_key: + description: + - Configure the authentication key for the interface. + - Valid only when afi = ipv4. + type: dict + suboptions: + encryption: + description: + - 0 Specifies an UNENCRYPTED authentication key will follow. + - 7 Specifies a proprietry encryption type.` + type: str + key: + description: + - password (up to 8 chars). + type: str + bfd: + description: Enable BFD. + type: bool + cost: + description: + - metric associated with interface. + type: int + dead_interval: + description: + - Time interval to detect a dead router. + type: int + encryption_v3: + description: + - Authentication settings on the interface. + - Valid only when afi = ipv6. + type: dict + suboptions: + spi: + description: IPsec Security Parameter Index. + type: int + encryption: + description: encryption type. + choices: ["3des-cbc", "aes-128-cbc", "aes-192-cbc", "aes-256-cbc", "null"] + type: str + algorithm: + description: algorithm. + type: str + choices: ["md5", "sha1"] + keytype: + description: + - Specifies if an unencrypted/hidden follows. + - 0 denotes unencrypted key. + - 7 denotes hidden key. + type: str + passphrase: + description: Passphrase String for deriving keys for authentication and encryption. + type: str + key: + description: key + type: str + hello_interval: + description: + - Timer interval between transmission of hello packets. + type: int + ip_params: + description: + - Specify parameters for IPv4/IPv6. + - Valid only when afi = ipv6. + type: list + elements: dict + suboptions: + afi: + description: + - Address Family Identifier (AFI) for OSPF settings on the interfaces. + type: str + choices: ['ipv4', 'ipv6'] + required: True + area: + description: + - Area associated with interface. + - Valid only when afi = ipv4. + type: dict + suboptions: + area_id: + description: + - Area ID as a decimal or IP address format. + type: str + required: True + bfd: + description: Enable BFD. + type: bool + cost: + description: + - metric associated with interface. + type: int + dead_interval: + description: + - Time interval to detect a dead router. + type: int + hello_interval: + description: + - Timer interval between transmission of hello packets. + type: int + mtu_ignore: + description: + - if True, Disable MTU check for Database Description packets. + type: bool + network: + description: + - Interface type. + type: str + priority: + description: + - Interface priority. + type: int + retransmit_interval: + description: + - LSA retransmission interval. + type: int + passive_interface: + description: + - Suppress routing updates in an interface. + type: bool + transmit_delay: + description: + - LSA transmission delay. + type: int + message_digest_key: + description: + - Message digest authentication password (key) settings. + type: dict + suboptions: + key_id: + description: + - Key ID. + type: int + encryption: + description: + - 0 Specifies an UNENCRYPTED ospf password (key) will follow. + - 7 Specifies a proprietry encryption type. + type: str + key: + description: + - Authentication key (upto 16 chars). + type: str + mtu_ignore: + description: + - if True, Disable MTU check for Database Description packets. + type: bool + network: + description: + - Interface type. + type: str + passive_interface: + description: + - Suppress routing updates in an interface. + - Valid only when afi = ipv6. + type: bool + priority: + description: + - Interface priority. + type: int + retransmit_interval: + description: + - LSA retransmission interval. + type: int + shutdown: + description: + - Shutdown OSPF on this interface. + type: bool + transmit_delay: + description: + - LSA transmission delay. + type: int + running_config: + description: + - This option is used only with state I(parsed). + - The value of this option should be the output received from the EOS device by + executing the command B(show running-config | section interface). + - The state I(parsed) reads the configuration from C(running_config) option and + transforms it into Ansible structured data as per the resource module's argspec + and the value is then returned in the I(parsed) key within the result. + type: str + + state: + description: + - The state the configuration should be left in. + type: str + choices: + - merged + - replaced + - overridden + - deleted + - gathered + - parsed + - rendered + default: merged +""" + +EXAMPLES = """ + +# Using merged + +# Before state + +# veos(config)#show running-config | section interface | ospf +# veos(config)# + + - name: Merge provided configuration with device configuration + arista.eos.eos_ospf_interfaces: + config: + - name: "Vlan1" + address_family: + - afi: "ipv4" + area: + area_id: "0.0.0.50" + cost: 500 + mtu_ignore: True + - afi: "ipv6" + dead_interval: 44 + ip_params: + - afi: "ipv6" + mtu_ignore: True + network: "point-to-point" + state: merged + +# After State + +# veos(config)#show running-config | section interface | ospf +# interface Vlan1 +# ip ospf cost 500 +# ip ospf mtu-ignore +# ip ospf area 0.0.0.50 +# ospfv3 dead-interval 44 +# ospfv3 ipv6 network point-to-point +# ospfv3 ipv6 mtu-ignore +# veos(config)# +# +# +# Module Execution: +# +# "after": [ +# { +# "name": "Ethernet1" +# }, +# { +# "name": "Ethernet2" +# }, +# { +# "name": "Management1" +# }, +# { +# "address_family": [ +# { +# "afi": "ipv4", +# "area": { +# "area_id": "0.0.0.50" +# }, +# "cost": 500, +# "mtu_ignore": True +# }, +# { +# "afi": "ipv6", +# "dead_interval": 44, +# "ip_params": [ +# { +# "afi": "ipv6", +# "mtu_ignore": True, +# "network": "point-to-point" +# } +# ] +# } +# ], +# "name": "Vlan1" +# } +# ], +# "before": [ +# { +# "name": "Ethernet1" +# }, +# { +# "name": "Ethernet2" +# }, +# { +# "name": "Management1" +# } +# ], +# "changed": True, +# "commands": [ +# "interface Vlan1", +# "ip ospf area 0.0.0.50", +# "ip ospf cost 500", +# "ip ospf mtu-ignore", +# "ospfv3 dead-interval 44", +# "ospfv3 ipv6 mtu-ignore", +# "ospfv3 ipv6 network point-to-point" +# ], +# + +# Using replaced +#--------------- + +# Before State: + +# veos(config)#show running-config | section interface | ospf +# interface Vlan1 +# ip ospf cost 500 +# ip ospf dead-interval 29 +# ip ospf hello-interval 66 +# ip ospf mtu-ignore +# ip ospf area 0.0.0.50 +# ospfv3 cost 106 +# ospfv3 hello-interval 77 +# ospfv3 dead-interval 44 +# ospfv3 transmit-delay 100 +# ospfv3 ipv4 priority 45 +# ospfv3 ipv4 area 0.0.0.5 +# ospfv3 ipv6 passive-interface +# ospfv3 ipv6 retransmit-interval 115 +# ospfv3 ipv6 network point-to-point +# ospfv3 ipv6 mtu-ignore +# ! +# interface Vlan2 +# ospfv3 ipv4 hello-interval 45 +# ospfv3 ipv4 retransmit-interval 100 +# ospfv3 ipv4 area 0.0.0.6 +# veos(config)# + + + - name: Replace device configuration with provided configuration + arista.eos.eos_ospf_interfaces: + config: + - name: "Vlan1" + address_family: + - afi: "ipv6" + cost: 44 + bfd: True + ip_params: + - afi: "ipv6" + mtu_ignore: True + network: "point-to-point" + dead_interval: 56 + state: replaced + +# After State: + +# veos(config)#show running-config | section interface | ospf +# interface Vlan1 +# ospfv3 bfd +# ospfv3 cost 44 +# no ospfv3 ipv6 passive-interface +# ospfv3 ipv6 network point-to-point +# ospfv3 ipv6 mtu-ignore +# ! +# interface Vlan2 +# ospfv3 ipv4 hello-interval 45 +# ospfv3 ipv4 retransmit-interval 100 +# ospfv3 ipv4 area 0.0.0.6 +# veos(config)# +# +# Module Execution: +# +# "after": [ +# { +# "name": "Ethernet1" +# }, +# { +# "name": "Ethernet2" +# }, +# { +# "name": "Management1" +# }, +# { +# "address_family": [ +# { +# "afi": "ipv6", +# "bfd": True, +# "cost": 44, +# "ip_params": [ +# { +# "afi": "ipv6", +# "mtu_ignore": True, +# "network": "point-to-point" +# } +# ] +# } +# ], +# "name": "Vlan1" +# }, +# { +# "address_family": [ +# { +# "afi": "ipv6", +# "ip_params": [ +# { +# "afi": "ipv4", +# "area": { +# "area_id": "0.0.0.6" +# }, +# "hello_interval": 45, +# "retransmit_interval": 100 +# } +# ] +# } +# ], +# "name": "Vlan2" +# } +# ], +# "before": [ +# { +# "name": "Ethernet1" +# }, +# { +# "name": "Ethernet2" +# }, +# { +# "name": "Management1" +# }, +# { +# "address_family": [ +# { +# "afi": "ipv4", +# "area": { +# "area_id": "0.0.0.50" +# }, +# "cost": 500, +# "dead_interval": 29, +# "hello_interval": 66, +# "mtu_ignore": True +# }, +# { +# "afi": "ipv6", +# "cost": 106, +# "dead_interval": 44, +# "hello_interval": 77, +# "ip_params": [ +# { +# "afi": "ipv4", +# "area": { +# "area_id": "0.0.0.5" +# }, +# "priority": 45 +# }, +# { +# "afi": "ipv6", +# "mtu_ignore": True, +# "network": "point-to-point", +# "passive_interface": True, +# "retransmit_interval": 115 +# } +# ], +# "transmit_delay": 100 +# } +# ], +# "name": "Vlan1" +# }, +# { +# "address_family": [ +# { +# "afi": "ipv6", +# "ip_params": [ +# { +# "afi": "ipv4", +# "area": { +# "area_id": "0.0.0.6" +# }, +# "hello_interval": 45, +# "retransmit_interval": 100 +# } +# ] +# } +# ], +# "name": "Vlan2" +# } +# ], +# "changed": True, +# "commands": [ +# "interface Vlan1", +# "no ip ospf cost 500", +# "no ip ospf dead-interval 29", +# "no ip ospf hello-interval 66", +# "no ip ospf mtu-ignore", +# "no ip ospf area 0.0.0.50", +# "ospfv3 cost 44", +# "ospfv3 bfd", +# "ospfv3 authentication ipsec spi 30 md5 passphrase 7 7hl8FV3lZ6H1mAKpjL47hQ==", +# "no ospfv3 ipv4 priority 45", +# "no ospfv3 ipv4 area 0.0.0.5", +# "ospfv3 ipv6 dead-interval 56", +# "no ospfv3 ipv6 passive-interface", +# "no ospfv3 ipv6 retransmit-interval 115", +# "no ospfv3 hello-interval 77", +# "no ospfv3 dead-interval 44", +# "no ospfv3 transmit-delay 100" +# ], +# + +# Using overidden: +# ---------------- + +# Before State: +# veos(config)#show running-config | section interface | ospf +# interface Vlan1 +# ip ospf dead-interval 29 +# ip ospf hello-interval 66 +# ip ospf mtu-ignore +# ospfv3 bfd +# ospfv3 cost 106 +# ospfv3 hello-interval 77 +# ospfv3 transmit-delay 100 +# ospfv3 ipv4 priority 45 +# ospfv3 ipv4 area 0.0.0.5 +# ospfv3 ipv6 passive-interface +# ospfv3 ipv6 dead-interval 56 +# ospfv3 ipv6 retransmit-interval 115 +# ospfv3 ipv6 network point-to-point +# ospfv3 ipv6 mtu-ignore +# ! +# interface Vlan2 +# ospfv3 ipv4 hello-interval 45 +# ospfv3 ipv4 retransmit-interval 100 +# ospfv3 ipv4 area 0.0.0.6 +# veos(config)# + + - name: Override device configuration with provided configuration + arista.eos.eos_ospf_interfaces: + config: + - name: "Vlan1" + address_family: + - afi: "ipv6" + cost: 44 + bfd: True + ip_params: + - afi: "ipv6" + mtu_ignore: True + network: "point-to-point" + dead_interval: 56 + state: overridden + +# After State: + +# veos(config)#show running-config | section interface | ospf +# interface Vlan1 +# ospfv3 bfd +# ospfv3 cost 44 +# no ospfv3 ipv6 passive-interface +# ospfv3 ipv6 dead-interval 56 +# ospfv3 ipv6 network point-to-point +# ospfv3 ipv6 mtu-ignore +# veos(config)# +# +# +# Module Execution: +# +# "after": [ +# { +# "name": "Ethernet1" +# }, +# { +# "name": "Ethernet2" +# }, +# { +# "name": "Management1" +# }, +# { +# "address_family": [ +# { +# "afi": "ipv6", +# "bfd": True, +# "cost": 44, +# "ip_params": [ +# { +# "afi": "ipv6", +# "dead_interval": 56, +# "mtu_ignore": True, +# "network": "point-to-point" +# } +# ] +# } +# ], +# "name": "Vlan1" +# }, +# { +# "name": "Vlan2" +# } +# ], +# "before": [ +# { +# "name": "Ethernet1" +# }, +# { +# "name": "Ethernet2" +# }, +# { +# "name": "Management1" +# }, +# { +# "address_family": [ +# { +# "afi": "ipv4", +# "dead_interval": 29, +# "hello_interval": 66, +# "mtu_ignore": True +# }, +# { +# "afi": "ipv6", +# "bfd": True, +# "cost": 106, +# "hello_interval": 77, +# "ip_params": [ +# { +# "afi": "ipv4", +# "area": { +# "area_id": "0.0.0.5" +# }, +# "priority": 45 +# }, +# { +# "afi": "ipv6", +# "dead_interval": 56, +# "mtu_ignore": True, +# "network": "point-to-point", +# "passive_interface": True, +# "retransmit_interval": 115 +# } +# ], +# "transmit_delay": 100 +# } +# ], +# "name": "Vlan1" +# }, +# { +# "address_family": [ +# { +# "afi": "ipv6", +# "ip_params": [ +# { +# "afi": "ipv4", +# "area": { +# "area_id": "0.0.0.6" +# }, +# "hello_interval": 45, +# "retransmit_interval": 100 +# } +# ] +# } +# ], +# "name": "Vlan2" +# } +# ], +# "changed": True, +# "commands": [ +# "interface Vlan2", +# "no ospfv3 ipv4 hello-interval 45", +# "no ospfv3 ipv4 retransmit-interval 100", +# "no ospfv3 ipv4 area 0.0.0.6", +# "interface Vlan1", +# "no ip ospf dead-interval 29", +# "no ip ospf hello-interval 66", +# "no ip ospf mtu-ignore", +# "ospfv3 cost 44", +# "ospfv3 authentication ipsec spi 30 md5 passphrase 7 7hl8FV3lZ6H1mAKpjL47hQ==", +# "no ospfv3 ipv4 priority 45", +# "no ospfv3 ipv4 area 0.0.0.5", +# "no ospfv3 ipv6 passive-interface", +# "no ospfv3 ipv6 retransmit-interval 115", +# "no ospfv3 hello-interval 77", +# "no ospfv3 transmit-delay 100" +# ], +# + +# Using deleted: +#-------------- + +# before State: + +# veos(config)#show running-config | section interface | ospf +# interface Vlan1 +# ip ospf dead-interval 29 +# ip ospf hello-interval 66 +# ip ospf mtu-ignore +# ospfv3 bfd +# ospfv3 cost 106 +# ospfv3 hello-interval 77 +# ospfv3 transmit-delay 100 +# ospfv3 ipv4 priority 45 +# ospfv3 ipv4 area 0.0.0.5 +# ospfv3 ipv6 passive-interface +# ospfv3 ipv6 dead-interval 56 +# ospfv3 ipv6 retransmit-interval 115 +# ospfv3 ipv6 network point-to-point +# ospfv3 ipv6 mtu-ignore +# ! +# interface Vlan2 +# ospfv3 ipv4 hello-interval 45 +# ospfv3 ipv4 retransmit-interval 100 +# ospfv3 ipv4 area 0.0.0.6 +# veos(config)# + + - name: Delete device configuration + arista.eos.eos_ospf_interfaces: + config: + - name: "Vlan1" + state: deleted + +# After State: + +# veos#show running-config | section interface | ospf +# interface Vlan2 +# ospfv3 ipv4 hello-interval 45 +# ospfv3 ipv4 retransmit-interval 100 +# ospfv3 ipv4 area 0.0.0.6 +# +# Module Execution: +# +# "after": [ +# { +# "name": "Ethernet1" +# }, +# { +# "name": "Ethernet2" +# }, +# { +# "name": "Management1" +# }, +# { +# "name": "Vlan1" +# }, +# { +# "address_family": [ +# { +# "afi": "ipv6", +# "ip_params": [ +# { +# "afi": "ipv4", +# "area": { +# "area_id": "0.0.0.6" +# }, +# "hello_interval": 45, +# "retransmit_interval": 100 +# } +# ] +# } +# ], +# "name": "Vlan2" +# } +# ], +# "before": [ +# { +# "name": "Ethernet1" +# }, +# { +# "name": "Ethernet2" +# }, +# { +# "name": "Management1" +# }, +# { +# "address_family": [ +# { +# "afi": "ipv4", +# "dead_interval": 29, +# "hello_interval": 66, +# "mtu_ignore": True +# }, +# { +# "afi": "ipv6", +# "bfd": True, +# "cost": 106, +# "hello_interval": 77, +# "ip_params": [ +# { +# "afi": "ipv4", +# "area": { +# "area_id": "0.0.0.5" +# }, +# "priority": 45 +# }, +# { +# "afi": "ipv6", +# "dead_interval": 56, +# "mtu_ignore": True, +# "network": "point-to-point", +# "passive_interface": True, +# "retransmit_interval": 115 +# } +# ], +# "transmit_delay": 100 +# } +# ], +# "name": "Vlan1" +# }, +# { +# "address_family": [ +# { +# "afi": "ipv6", +# "ip_params": [ +# { +# "afi": "ipv4", +# "area": { +# "area_id": "0.0.0.6" +# }, +# "hello_interval": 45, +# "retransmit_interval": 100 +# } +# ] +# } +# ], +# "name": "Vlan2" +# } +# ], +# "changed": True, +# "commands": [ +# "interface Vlan1", +# "no ip ospf dead-interval 29", +# "no ip ospf hello-interval 66", +# "no ip ospf mtu-ignore", +# "no ospfv3 bfd", +# "no ospfv3 cost 106", +# "no ospfv3 hello-interval 77", +# "no ospfv3 transmit-delay 100", +# "no ospfv3 ipv4 priority 45", +# "no ospfv3 ipv4 area 0.0.0.5", +# "no ospfv3 ipv6 passive-interface", +# "no ospfv3 ipv6 dead-interval 56", +# "no ospfv3 ipv6 retransmit-interval 115", +# "no ospfv3 ipv6 network point-to-point", +# "no ospfv3 ipv6 mtu-ignore" +# ], +# + +# Using parsed: +# ------------ + +# parsed.cfg: +# ---------- + +# interface Vlan1 +# ip ospf dead-interval 29 +# ip ospf hello-interval 66 +# ip ospf mtu-ignore +# ip ospf cost 500 +# ospfv3 bfd +# ospfv3 cost 106 +# ospfv3 hello-interval 77 +# ospfv3 transmit-delay 100 +# ospfv3 ipv4 priority 45 +# ospfv3 ipv4 area 0.0.0.5 +# ospfv3 ipv6 passive-interface +# ospfv3 ipv6 dead-interval 56 +# ospfv3 ipv6 retransmit-interval 115 +# ospfv3 ipv6 network point-to-point +# ospfv3 ipv6 mtu-ignore +# ! +# interface Vlan2 +# ospfv3 ipv4 hello-interval 45 +# ospfv3 ipv4 retransmit-interval 100 +# ospfv3 ipv4 area 0.0.0.6 +# + + - name: parse configs + arista.eos.eos_ospf_interfaces: + running_config: "{{ lookup('file', './parsed.cfg') }}" + state: parsed + +# Module Execution: +# "parsed": [ +# { +# "address_family": [ +# { +# "afi": "ipv4", +# "cost": 500, +# "dead_interval": 29, +# "hello_interval": 66, +# "mtu_ignore": True +# }, +# { +# "afi": "ipv6", +# "bfd": True, +# "cost": 106, +# "hello_interval": 77, +# "ip_params": [ +# { +# "afi": "ipv4", +# "area": { +# "area_id": "0.0.0.5" +# }, +# "priority": 45 +# }, +# { +# "afi": "ipv6", +# "dead_interval": 56, +# "mtu_ignore": True, +# "network": "point-to-point", +# "passive_interface": True, +# "retransmit_interval": 115 +# } +# ], +# "transmit_delay": 100 +# } +# ], +# "name": "Vlan1" +# }, +# { +# "address_family": [ +# { +# "afi": "ipv6", +# "ip_params": [ +# { +# "afi": "ipv4", +# "area": { +# "area_id": "0.0.0.6" +# }, +# "hello_interval": 45, +# "retransmit_interval": 100 +# } +# ] +# } +# ], +# "name": "Vlan2" +# } +# ] + +# Using gathered: + +# Device COnfig: + +# veos#show running-config | section interface | ospf +# interface Vlan1 +# ip ospf cost 500 +# ip ospf dead-interval 29 +# ip ospf hello-interval 66 +# ip ospf mtu-ignore +# ip ospf area 0.0.0.50 +# ospfv3 cost 106 +# ospfv3 hello-interval 77 +# ospfv3 transmit-delay 100 +# ospfv3 ipv4 priority 45 +# ospfv3 ipv4 area 0.0.0.5 +# ospfv3 ipv6 passive-interface +# ospfv3 ipv6 dead-interval 56 +# ospfv3 ipv6 retransmit-interval 115 +# ospfv3 ipv6 network point-to-point +# ospfv3 ipv6 mtu-ignore +# ! +# interface Vlan2 +# ospfv3 ipv4 hello-interval 45 +# ospfv3 ipv4 retransmit-interval 100 +# ospfv3 ipv4 area 0.0.0.6 +# veos# + + - name: gather configs + arista.eos.eos_ospf_interfaces: + state: gathered + +# Module Execution: +# +# "gathered": [ +# { +# "name": "Ethernet1" +# }, +# { +# "name": "Ethernet2" +# }, +# { +# "name": "Management1" +# }, +# { +# "address_family": [ +# { +# "afi": "ipv4", +# "area": { +# "area_id": "0.0.0.50" +# }, +# "cost": 500, +# "dead_interval": 29, +# "hello_interval": 66, +# "mtu_ignore": True +# }, +# { +# "afi": "ipv6", +# "cost": 106, +# "hello_interval": 77, +# "ip_params": [ +# { +# "afi": "ipv4", +# "area": { +# "area_id": "0.0.0.5" +# }, +# "priority": 45 +# }, +# { +# "afi": "ipv6", +# "dead_interval": 56, +# "mtu_ignore": True, +# "network": "point-to-point", +# "passive_interface": True, +# "retransmit_interval": 115 +# } +# ], +# "transmit_delay": 100 +# } +# ], +# "name": "Vlan1" +# }, +# { +# "address_family": [ +# { +# "afi": "ipv6", +# "ip_params": [ +# { +# "afi": "ipv4", +# "area": { +# "area_id": "0.0.0.6" +# }, +# "hello_interval": 45, +# "retransmit_interval": 100 +# } +# ] +# } +# ], +# "name": "Vlan2" +# } +# ], +# + + +# Using rendered: +# -------------- + + - name: Render provided configuration + arista.eos.eos_ospf_interfaces: + config: + - name: "Vlan1" + address_family: + - afi: "ipv4" + dead_interval: 29 + mtu_ignore: True + hello_interval: 66 + - afi: "ipv6" + hello_interval: 77 + cost : 106 + transmit_delay: 100 + ip_params: + - afi: "ipv6" + retransmit_interval: 115 + dead_interval: 56 + passive_interface: True + - afi: "ipv4" + area: + area_id: "0.0.0.5" + priority: 45 + - name: "Vlan2" + address_family: + - afi: "ipv6" + ip_params: + - afi: "ipv4" + area: + area_id: "0.0.0.6" + hello_interval: 45 + retransmit_interval: 100 + - afi: "ipv4" + message_digest_key: + key_id: 200 + encryption: 7 + key: "hkdfhtu==" + + state: rendered + +# Module Execution: +# +# "rendered": [ +# "interface Vlan1", +# "ip ospf dead-interval 29", +# "ip ospf mtu-ignore", +# "ip ospf hello-interval 66", +# "ospfv3 hello-interval 77", +# "ospfv3 cost 106", +# "ospfv3 transmit-delay 100", +# "ospfv3 ipv4 area 0.0.0.5", +# "ospfv3 ipv4 priority 45", +# "ospfv3 ipv6 retransmit-interval 115", +# "ospfv3 ipv6 dead-interval 56", +# "ospfv3 ipv6 passive-interface", +# "interface Vlan2", +# "ip ospf message-digest-key 200 md5 7 hkdfhtu==", +# "ospfv3 ipv4 area 0.0.0.6", +# "ospfv3 ipv4 hello-interval 45", +# "ospfv3 ipv4 retransmit-interval 100" +# ] +# + +""" + +from ansible.module_utils.basic import AnsibleModule +from ansible_collections.arista.eos.plugins.module_utils.network.eos.argspec.ospf_interfaces.ospf_interfaces import ( + Ospf_interfacesArgs, +) +from ansible_collections.arista.eos.plugins.module_utils.network.eos.config.ospf_interfaces.ospf_interfaces import ( + Ospf_interfaces, +) + + +def main(): + """ + Main entry point for module execution + + :returns: the result form module invocation + """ + module = AnsibleModule( + argument_spec=Ospf_interfacesArgs.argument_spec, + mutually_exclusive=[], + required_if=[], + supports_check_mode=False, + ) + + result = Ospf_interfaces(module).execute_module() + module.exit_json(**result) + + +if __name__ == "__main__": + main() diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/modules/eos_ospfv2.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/modules/eos_ospfv2.py new file mode 100644 index 00000000..ff178d9c --- /dev/null +++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/modules/eos_ospfv2.py @@ -0,0 +1,1545 @@ +#!/usr/bin/python +# -*- coding: utf-8 -*- +# Copyright 2019 Red Hat +# GNU General Public License v3.0+ +# (see COPYING or https://www.gnu.org/licenses/gpl-3.0.txt) + +############################################# +# WARNING # +############################################# +# +# This file is auto generated by the resource +# module builder playbook. +# +# Do not edit this file manually. +# +# Changes to this file will be over written +# by the resource module builder. +# +# Changes should be made in the model used to +# generate this file or in the resource module +# builder template. +# +############################################# + +""" +The module file for eos_ospfv2 +""" + +from __future__ import absolute_import, division, print_function + +__metaclass__ = type + + +DOCUMENTATION = """ +module: eos_ospfv2 +short_description: OSPFv2 resource module +description: This module configures and manages the attributes of ospfv2 on Arista + EOS platforms. +version_added: 1.0.0 +author: Gomathi Selvi Srinivasan (@GomathiselviS) +notes: +- Tested against Arista EOS 4.23.0F +- This module works with connection C(network_cli). See the L(EOS Platform Options,../network/user_guide/platform_eos.html). +options: + config: + description: A list of configurations for ospfv2. + type: dict + suboptions: + processes: + description: A list of dictionary specifying the ospfv2 processes. + type: list + elements: dict + suboptions: + process_id: + description: ID of OSPFV2 process. + type: int + vrf: + description: VRF name . + type: str + traffic_engineering: + description: Enter traffic engineering config mode + type: bool + adjacency: + description: Configure adjacency options for OSPF instance. + type: dict + suboptions: + exchange_start: + description: Configure exchange-start options for OSPF instance. + type: dict + suboptions: + threshold: + description: Number of peers to bring up simultaneously. + type: int + router_id: + description: 32-bit number assigned to a router running OSPFv2. + type: str + max_lsa: + description: Specifies the switch behavior on reaching max lsa count. + type: dict + suboptions: + count: + description: maximum count of lsas. + type: int + threshold: + description: percentage of <count> , when a warning should be raised. + type: int + ignore_time: + description: time in minutes, for which the switch shoud be shutdown + on max-lsa warning + type: int + ignore_count: + description: No. of times the switch can shut down temporarily on + warning + type: int + reset_time: + description: Time in minutes, after which the shutdown counter resets. + type: int + warning: + description: Only give warning message when limit is exceeded + type: bool + max_metric: + description: Set maximum metric. + type: dict + suboptions: + router_lsa: + description: Maximum metric in self-originated router-LSAs. + type: dict + suboptions: + set: + description: + - Set router-lsa attribute. + type: bool + external_lsa: + description: Override external-lsa metric with max-metric value. + type: dict + suboptions: + set: + description: + - Set external-lsa attribute. + type: bool + max_metric_value: + description: + - Set max metric value for external LSAs. + type: int + include_stub: + description: Set maximum metric for stub links in router-LSAs. + type: bool + on_startup: + description: Set maximum metric temporarily after reboot. + type: dict + suboptions: + wait_period: + description: + - Wait period in seconds after startup. + type: int + summary_lsa: + description: Override summary-lsa metric with max-metric value. + type: dict + suboptions: + set: + description: + - Set external-lsa attribute. + type: bool + max_metric_value: + description: + - Set max metric value for external LSAs. + type: int + log_adjacency_changes: + description: To configure link-state changes and transitions of OSPFv2 + neighbors. + type: dict + suboptions: + detail: + description: If true , configures the switch to log all link-state + changes. + type: bool + maximum_paths: + description: Maximum number of next-hops in an ECMP route. + type: int + mpls_ldp: + description: mpls ldp sync configuration. + type: bool + networks: + description: Configure routing for a network. + type: list + elements: dict + suboptions: + network_address: + description: Network Address. + type: str + prefix: + description: Prefix. + type: str + mask: + description: Network Wildcard Mask. + type: str + area: + description: Configure OSPF area. + type: str + passive_interface: + description: Include interface but without actively running OSPF. + type: dict + suboptions: + interface_list: + description: Interface range. + type: str + default: + description: If True, Set all interfaces to passive by default + type: bool + point_to_point: + description: Configure Point-to-point specific features. + type: bool + rfc1583compatibility: + description: Specifies different methods for calculating summary route + metrics. + type: bool + distance: + description: Specifies the administrative distance for routes. + type: dict + suboptions: + external: + description: Routes external to the area + type: int + inter_area: + description: Routes from other areas + type: int + intra_area: + description: Routes with in an area + type: int + redistribute: + description: Specifies the routes to be redistributed + type: list + elements: dict + suboptions: + routes: + description: Route types (BGP,isis,connected etc) + type: str + route_map: + description: Specify which route map to use. + type: str + isis_level: + description: ISIS levels. + type: str + retransmission_threshold: + description: Configure threshold for retransmission. + type: int + distribute_list: + description: Specifies the list of routes to be filtered. + type: dict + suboptions: + route_map: + description: route map to be filtered + type: str + prefix_list: + description: prefix list to be filtered + type: str + areas: + description: Specifies the configuration for OSPF areas + type: list + elements: dict + suboptions: + area_id: + description: Specifies a 32 bit number expressed in decimal or dotted-decimal + notation. + type: str + default_cost: + description: Specify the cost for default summary route in stub/NSSA + area. + type: int + filter: + description: Specify the filter for incoming summary LSAs. + type: dict + suboptions: + address: + description: IP address. + type: str + subnet_address: + description: IP address with mask length + type: str + subnet_mask: + description: IP subnet mask + type: str + prefix_list: + description: Specify list to filter for incoming LSAs. + type: str + nssa: + description: Configures NSSA parameters. + type: dict + suboptions: + default_information_originate: + description: Originate default Type 7 LSA. + type: dict + suboptions: + metric: + description: Metric for default route. + type: int + metric_type: + description: Metric type for default route. + type: int + nssa_only: + description: Limit default advertisement to this NSSA area. + type: bool + no_summary: + description: Filter all type-3 LSAs in the nssa area. + type: bool + nssa_only: + description: Disable Type-7 LSA p-bit setting + type: bool + set: + description: Set config up to nssa + type: bool + not_so_stubby: + description: Configures NSSA parameters. + type: dict + suboptions: + default_information_originate: + description: Originate default Type 7 LSA. + type: dict + suboptions: + metric: + description: Metric for default route. + type: int + metric_type: + description: Metric type for default route. + type: int + nssa_only: + description: Limit default advertisement to this NSSA area. + type: bool + lsa: + description: lsa parameters + type: bool + no_summary: + description: Filter all type-3 LSAs in the nssa area. + type: bool + nssa_only: + description: Disable Type-7 LSA p-bit setting + type: bool + set: + description: Set config up to not-so-stubby + type: bool + range: + description: Configure route summarization. + type: dict + suboptions: + address: + description: IP address. + type: str + subnet_address: + description: IP address with mask length + type: str + subnet_mask: + description: IP subnet mask + type: str + advertise: + description: Enable Advertisement of the range. + type: bool + cost: + description: Configures the metric. + type: int + stub: + description: Stub area. + type: dict + suboptions: + no_summary: + description: If False , Filter all type-3 LSAs in the stub area. + type: bool + set: + description: When true sets the stub config alone. + type: bool + auto_cost: + description: Set auto-cost. + type: dict + suboptions: + reference_bandwidth: + description: reference bandwidth in megabits per sec. + type: int + bfd: + description: Enable BFD. + type: dict + suboptions: + all_interfaces: + description: Enable BFD on all interfaces. + type: bool + default_information: + description: Control distribution of default information. + type: dict + suboptions: + originate: + description: Distribute a default route. + type: bool + always: + description: Always advertise default route. + type: bool + metric: + description: Metric for default route. + type: int + metric_type: + description: Metric type for default route. + type: int + route_map: + description: Specify which route-map to use. + type: str + default_metric: + description: Configure the default metric for redistributed routes + type: int + dn_bit_ignore: + description: If True, Disable dn-bit check for Type-3 LSAs in non-default + VRFs. + type: bool + graceful_restart: + description: Enable graceful restart mode. + type: dict + suboptions: + grace_period: + description: Specify maximum time to wait for graceful-restart to + complete. + type: int + set: + description: When true sets the grace_fulrestart config alone. + type: bool + graceful_restart_helper: + description: If True, Enable graceful restart helper. + type: bool + shutdown: + description: Disable the OSPF instance. + type: bool + summary_address: + description: Summary route configuration. + type: dict + suboptions: + address: + description: IP summary address. + type: str + prefix: + description: Prefix. + type: str + mask: + description: Summary Mask. + type: str + attribute_map: + description: Set attributes of summary route. + type: str + not_advertise: + description: Do not advertise summary route. + type: bool + tag: + description: Set tag. + type: int + timers: + description: Configure OSPF timers. + type: list + elements: dict + suboptions: + lsa: + description: Configure OSPF LSA timers. + type: dict + suboptions: + rx: + description: Configure OSPF LSA receiving timers + type: dict + suboptions: + min_interval: + description: Configure OSPF LSA arrival timer. + type: int + tx: + description: Configure OSPF LSA transmission timers. + type: dict + suboptions: + delay: + description: Configure OSPF LSA transmission delay. + type: dict + suboptions: + initial: + description: Delay to generate first occurrence of LSA + in msecs. + type: int + min: + description: Min delay between originating the same LSA + in msecs. + type: int + max: + description: Maximum delay between originating the same + LSA in msecs. + type: int + out_delay: + description: Configure out-delay timer. + type: int + pacing: + description: Configure OSPF packet pacing. + type: int + spf: + description: Configure SPF timers + type: dict + suboptions: + seconds: + description: Seconds. + type: int + initial: + description: Initial SPF schedule delay in msecs. + type: int + min: + description: Min Hold time between two SPFs in msecs + type: int + max: + description: Max wait time between two SPFs in msecs. + type: int + fips_restrictions: + description: Use FIPS compliant algorithms + type: str + running_config: + description: + - This option is used only with state I(parsed). + - The value of this option should be the output received from the EOS device by + executing the command B(show running-config | section ospf). + - The state I(parsed) reads the configuration from C(running_config) option and + transforms it into Ansible structured data as per the resource module's argspec + and the value is then returned in the I(parsed) key within the result. + type: str + state: + description: + - The state the configuration should be left in. + type: str + choices: [deleted, merged, overridden, replaced, gathered, rendered, parsed] + default: merged + +""" +EXAMPLES = """ +# Using merged + +# Before state: +# ------------ +# localhost#show running-config | section ospf +# localhost# + + - name: replace Ospf configs + arista.eos.eos_ospfv2: + config: + - processes: + - process_id: 1 + adjacency: + exchange_start: + threshold: 20045623 + areas: + - filter: + address: "10.1.1.0/24" + id: "0.0.0.2" + - id: "0.0.0.50" + range: + address: "172.20.0.0/16" + cost: 34 + default_information: + metric: 100 + metric_type: 1 + originate: True + distance: + intra_area: 85 + max_lsa: + count: 8000 + ignore_count: 3 + ignore_time: 6 + reset_time: 20 + threshold: 40 + networks: + - area: "0.0.0.0" + prefix: 10.10.2.0/24 + - area: "0.0.0.0" + prefix: "10.10.3.0/24" + redistribute: + - routes: "static" + router_id: "170.21.0.4" + - process_id: 2 + vrf: "vrf01" + areas: + - id: "0.0.0.9" + default_cost: 20 + max_lsa: + count: 8000 + ignore_count: 3 + ignore_time: 6 + reset_time: 20 + threshold: 40 + networks: + - area: "0.0.0.0" + prefix: 10.10.2.0/24 + - area: "0.0.0.0" + prefix: "10.10.3.0/24" + redistribute: + - routes: "static" + router_id: "170.21.0.4" + - process_id: 2 + vrf: "vrf01" + areas: + - id: "0.0.0.9" + default_cost: 20 + max_lsa: + count: 8000 + ignore_count: 3 + ignore_time: 6 + reset_time: 20 + threshold: 40 + - process_id: 3 + vrf: "vrf02" + redistribute: + - routes: "connected" + +# After state: +# localhost#show running-config | section ospf +# router ospf 1 +# router-id 170.21.0.4 +# distance ospf intra-area 85 +# redistribute static +# area 0.0.0.2 filter 10.1.1.0/24 +# area 0.0.0.50 range 172.20.0.0/16 cost 34 +# network 10.10.2.0/24 area 0.0.0.0 +# network 10.10.3.0/24 area 0.0.0.0 +# max-lsa 8000 40 ignore-time 6 ignore-count 3 reset-time 20 +# adjacency exchange-start threshold 20045623 +# default-information originate metric 100 metric-type 1 +# +# router ospf 2 vrf vrf01 +# area 0.0.0.9 default-cost 20 +# max-lsa 8000 40 ignore-time 6 ignore-count 3 reset-time 20 +# ! +# router ospf 3 vrf vrf02 +# redistribute connected +# max-lsa 12000 +# localhost# +# +# "processes": [ +# { +# "adjacency": { +# "exchange_start": { +# "threshold": 20045623 +# } +# }, +# "areas": [ +# { +# "filter": { +# "address": "10.1.1.0/24" +# }, +# "id": "0.0.0.2" +# }, +# { +# "id": "0.0.0.50", +# "range": { +# "address": "172.20.0.0/16", +# "cost": 34 +# } +# } +# ], +# "default_information": { +# "metric": 100, +# "metric_type": 1, +# "originate": true +# }, +# "distance": { +# "intra_area": 85 +# }, +# "max_lsa": { +# "count": 8000, +# "ignore_count": 3, +# "ignore_time": 6, +# "reset_time": 20, +# "threshold": 40 +# }, +# "networks": [ +# { +# "area": "0.0.0.0", +# "prefix": "10.10.2.0/24" +# }, +# { +# "area": "0.0.0.0", +# "prefix": "10.10.3.0/24" +# } +# ], +# "process_id": 1, +# "redistribute": [ +# { +# "routes": "static" +# } +# ], +# "router_id": "170.21.0.4" +# }, +# { +# "areas": [ +# { +# "default_cost": 20, +# "id": "0.0.0.9" +# } +# ], +# "max_lsa": { +# "count": 8000, +# "ignore_count": 3, +# "ignore_time": 6, +# "reset_time": 20, +# "threshold": 40 +# }, +# "process_id": 2, +# "vrf": "vrf01" +# }, +# { +# "max_lsa": { +# "count": 12000 +# }, +# "process_id": 3, +# "redistribute": [ +# { +# "routes": "connected" +# } +# ], +# "vrf": "vrf02" +# } +# ] +# } +# ] +# + + +# Using replaced: +# -------------- + +# Before State: + +# localhost#show running-config | section ospf +# router ospf 1 +# router-id 170.21.0.4 +# distance ospf intra-area 85 +# redistribute static +# area 0.0.0.2 filter 10.1.1.0/24 +# area 0.0.0.50 range 172.20.0.0/16 cost 34 +# network 10.10.2.0/24 area 0.0.0.0 +# network 10.10.3.0/24 area 0.0.0.0 +# max-lsa 8000 40 ignore-time 6 ignore-count 3 reset-time 20 +# adjacency exchange-start threshold 20045623 +# default-information originate metric 100 metric-type 1 +# ! +# router ospf 2 vrf vrf01 +# area 0.0.0.9 default-cost 20 +# max-lsa 8000 40 ignore-time 6 ignore-count 3 reset-time 20 +# ! +# router ospf 3 vrf vrf02 +# redistribute connected +# max-lsa 12000 +# localhost# +# +# "before": [ +# { +# "processes": [ +# { +# "adjacency": { +# "exchange_start": { +# "threshold": 20045623 +# } +# }, +# "areas": [ +# { +# "filter": { +# "address": "10.1.1.0/24" +# }, +# "id": "0.0.0.2" +# }, +# { +# "id": "0.0.0.50", +# "range": { +# "address": "172.20.0.0/16", +# "cost": 34 +# } +# } +# ], +# "default_information": { +# "metric": 100, +# "metric_type": 1, +# "originate": true +# }, +# "distance": { +# "intra_area": 85 +# }, +# "max_lsa": { +# "count": 8000, +# "ignore_count": 3, +# "ignore_time": 6, +# "reset_time": 20, +# "threshold": 40 +# }, +# "networks": [ +# { +# "area": "0.0.0.0", +# "prefix": "10.10.2.0/24" +# }, +# { +# "area": "0.0.0.0", +# "prefix": "10.10.3.0/24" +# } +# ], +# "process_id": 1, +# "redistribute": [ +# { +# "routes": "static" +# } +# ], +# "router_id": "170.21.0.4" +# }, +# { +# "areas": [ +# { +# "default_cost": 20, +# "id": "0.0.0.9" +# } +# ], +# "max_lsa": { +# "count": 8000, +# "ignore_count": 3, +# "ignore_time": 6, +# "reset_time": 20, +# "threshold": 40 +# }, +# "process_id": 2, +# "vrf": "vrf01" +# }, +# { +# "max_lsa": { +# "count": 12000 +# }, +# "process_id": 3, +# "redistribute": [ +# { +# "routes": "connected" +# } +# ], +# "vrf": "vrf02" +# } +# ] +# } +# ] +# + - name: replace Ospf configs + arista.eos.eos_ospfv2: + config: + - processes: + - process_id: 2 + vrf: "vrf01" + point_to_point: True + redistribute: + - routes: "isis" + isis_level: "level-1" + + state: replaced + +# After State: +# ----------- +# "router ospf 2 vrf vrf01", +# "no area 0.0.0.9 default-cost 20", +# "no max-lsa 8000 40 ignore-time 6 ignore-count 3 reset-time 20", +# "point-to-point routes", +# "redistribute isis level-1" +# +# "after": [ +# { +# "processes": [ +# { +# "adjacency": { +# "exchange_start": { +# "threshold": 20045623 +# } +# }, +# "areas": [ +# { +# "filter": { +# "address": "10.1.1.0/24" +# }, +# "id": "0.0.0.2" +# }, +# { +# "id": "0.0.0.50", +# "range": { +# "address": "172.20.0.0/16", +# "cost": 34 +# } +# } +# ], +# "default_information": { +# "metric": 100, +# "metric_type": 1, +# "originate": true +# }, +# "distance": { +# "intra_area": 85 +# }, +# "max_lsa": { +# "count": 8000, +# "ignore_count": 3, +# "ignore_time": 6, +# "reset_time": 20, +# "threshold": 40 +# }, +# "networks": [ +# { +# "area": "0.0.0.0", +# "prefix": "10.10.2.0/24" +# }, +# { +# "area": "0.0.0.0", +# "prefix": "10.10.3.0/24" +# } +# ], +# "process_id": 1, +# "redistribute": [ +# { +# "routes": "static" +# } +# ], +# "router_id": "170.21.0.4" +# }, +# { +# "max_lsa": { +# "count": 12000 +# }, +# "process_id": 2, +# "redistribute": [ +# { +# "isis_level": "level-1", +# "routes": "isis" +# } +# ], +# "vrf": "vrf01" +# }, +# { +# "max_lsa": { +# "count": 12000 +# }, +# "process_id": 3, +# "redistribute": [ +# { +# "routes": "connected" +# } +# ], +# "vrf": "vrf02" +# } +# ] +# } +# ] +# + +# Using overridden: +# ---------------- + +# Before State: +# localhost#show running-config | section ospf +# router ospf 1 +# router-id 170.21.0.4 +# distance ospf intra-area 85 +# redistribute static +# area 0.0.0.2 filter 10.1.1.0/24 +# area 0.0.0.50 range 172.20.0.0/16 cost 34 +# network 10.10.2.0/24 area 0.0.0.0 +# network 10.10.3.0/24 area 0.0.0.0 +# max-lsa 8000 40 ignore-time 6 ignore-count 3 reset-time 20 +# adjacency exchange-start threshold 20045623 +# default-information originate metric 100 metric-type 1 +# ! +# router ospf 2 vrf vrf01 +# redistribute isis level-1 +# max-lsa 12000 +# ! +# router ospf 3 vrf vrf02 +# redistribute connected +# max-lsa 12000 +# localhost# +# +# "before": [ +# { +# "processes": [ +# { +# "adjacency": { +# "exchange_start": { +# "threshold": 20045623 +# } +# }, +# "areas": [ +# { +# "filter": { +# "address": "10.1.1.0/24" +# }, +# "id": "0.0.0.2" +# }, +# { +# "id": "0.0.0.50", +# "range": { +# "address": "172.20.0.0/16", +# "cost": 34 +# } +# } +# ], +# "default_information": { +# "metric": 100, +# "metric_type": 1, +# "originate": true +# }, +# "distance": { +# "intra_area": 85 +# }, +# "max_lsa": { +# "count": 8000, +# "ignore_count": 3, +# "ignore_time": 6, +# "reset_time": 20, +# "threshold": 40 +# }, +# "networks": [ +# { +# "area": "0.0.0.0", +# "prefix": "10.10.2.0/24" +# }, +# { +# "area": "0.0.0.0", +# "prefix": "10.10.3.0/24" +# } +# ], +# "process_id": 1, +# "redistribute": [ +# { +# "routes": "static" +# } +# ], +# "router_id": "170.21.0.4" +# }, +# { +# "max_lsa": { +# "count": 12000 +# }, +# "process_id": 2, +# "redistribute": [ +# { +# "isis_level": "level-1", +# "routes": "isis" +# } +# ], +# "vrf": "vrf01" +# }, +# { +# "max_lsa": { +# "count": 12000 +# }, +# "process_id": 3, +# "redistribute": [ +# { +# "routes": "connected" +# } +# ], +# "vrf": "vrf02" +# } +# ] +# } +# ] + + - name: override Ospf configs + arista.eos.eos_ospfv2: + config: + - processes: + - process_id: 2 + vrf: "vrf01" + redistribute: + - routes: "connected" + + state: override + +# After State: + +# "no router ospf 1", +# "no router ospf 3", +# "router ospf 2 vrf vrf01", +# "no max-lsa 12000", +# "no redistribute isis level-1", +# "redistribute connected" +# +# "after": [ +# { +# "processes": [ +# { +# "max_lsa": { +# "count": 12000 +# }, +# "process_id": 2, +# "redistribute": [ +# { +# "routes": "connected" +# } +# ], +# "vrf": "vrf01" +# } +# ] +# } +# ] + +# Using Deleted: + +# localhost#show running-config | section ospf +# router ospf 1 +# router-id 170.21.0.4 +# distance ospf intra-area 85 +# redistribute static +# area 0.0.0.2 filter 10.1.1.0/24 +# area 0.0.0.50 range 172.20.0.0/16 cost 34 +# network 10.10.2.0/24 area 0.0.0.0 +# network 10.10.3.0/24 area 0.0.0.0 +# max-lsa 8000 40 ignore-time 6 ignore-count 3 reset-time 20 +# adjacency exchange-start threshold 20045623 +# default-information originate metric 100 metric-type 1 +# ! +# router ospf 2 vrf vrf01 +# redistribute connected +# area 0.0.0.9 default-cost 20 +# max-lsa 8000 40 ignore-time 6 ignore-count 3 reset-time 20 +# ! +# router ospf 3 vrf vrf02 +# redistribute connected +# max-lsa 12000 +# localhost# +# +# "before": [ +# { +# "processes": [ +# { +# "adjacency": { +# "exchange_start": { +# "threshold": 20045623 +# } +# }, +# "areas": [ +# { +# "filter": { +# "address": "10.1.1.0/24" +# }, +# "id": "0.0.0.2" +# }, +# { +# "id": "0.0.0.50", +# "range": { +# "address": "172.20.0.0/16", +# "cost": 34 +# } +# } +# ], +# "default_information": { +# "metric": 100, +# "metric_type": 1, +# "originate": true +# }, +# "distance": { +# "intra_area": 85 +# }, +# "max_lsa": { +# "count": 8000, +# "ignore_count": 3, +# "ignore_time": 6, +# "reset_time": 20, +# "threshold": 40 +# }, +# "networks": [ +# { +# "area": "0.0.0.0", +# "prefix": "10.10.2.0/24" +# }, +# { +# "area": "0.0.0.0", +# "prefix": "10.10.3.0/24" +# } +# ], +# "process_id": 1, +# "redistribute": [ +# { +# "routes": "static" +# } +# ], +# "router_id": "170.21.0.4" +# }, +# { +# "areas": [ +# { +# "default_cost": 20, +# "id": "0.0.0.9" +# } +# ], +# "max_lsa": { +# "count": 8000, +# "ignore_count": 3, +# "ignore_time": 6, +# "reset_time": 20, +# "threshold": 40 +# }, +# "process_id": 2, +# "redistribute": [ +# { +# "routes": "connected" +# } +# ], +# "vrf": "vrf01" +# }, +# { +# "max_lsa": { +# "count": 12000 +# }, +# "process_id": 3, +# "redistribute": [ +# { +# "routes": "connected" +# } +# ], +# "vrf": "vrf02" +# } +# ] +# } +# ] + + - name: Delete Ospf configs + arista.eos.eos_ospfv2: + config: + - processes: + - process_id: 1 + + state: deleted + +# After State: +# Commands: +# "no router ospf 1" + +# "after": [ +# { +# "processes": [ +# { +# "areas": [ +# { +# "default_cost": 20, +# "id": "0.0.0.9" +# } +# ], +# "max_lsa": { +# "count": 8000, +# "ignore_count": 3, +# "ignore_time": 6, +# "reset_time": 20, +# "threshold": 40 +# }, +# "process_id": 2, +# "redistribute": [ +# { +# "routes": "connected" +# } +# ], +# "vrf": "vrf01" +# }, +# { +# "max_lsa": { +# "count": 12000 +# }, +# "process_id": 3, +# "redistribute": [ +# { +# "routes": "connected" +# } +# ], +# "vrf": "vrf02" +# } +# ] +# } +# ] + +# Using gathered: +# localhost#show running-config | section ospf +# router ospf 2 vrf vrf01 +# redistribute connected +# area 0.0.0.9 default-cost 20 +# max-lsa 8000 40 ignore-time 6 ignore-count 3 reset-time 20 +# ! +# router ospf 3 vrf vrf02 +# redistribute connected +# max-lsa 12000 +# localhost# + + - name: replace Ospf configs + arista.eos.eos_ospfv2: + state: gathered + +# "gathered": [ +# { +# "processes": [ +# { +# "areas": [ +# { +# "default_cost": 20, +# "id": "0.0.0.9" +# } +# ], +# "max_lsa": { +# "count": 8000, +# "ignore_count": 3, +# "ignore_time": 6, +# "reset_time": 20, +# "threshold": 40 +# }, +# "process_id": 2, +# "redistribute": [ +# { +# "routes": "connected" +# } +# ], +# "vrf": "vrf01" +# }, +# { +# "max_lsa": { +# "count": 12000 +# }, +# "process_id": 3, +# "redistribute": [ +# { +# "routes": "connected" +# } +# ], +# "vrf": "vrf02" +# } +# ] +# } +# ] + +# Using parsed: +# ------------ + +# parsed.cfg +# router ospf 1 +# adjacency exchange-start threshold 20045623 +# area 0.0.0.2 filter 10.1.1.0/24 +# area 0.0.0.50 range 172.20.0.0/16 cost 34 +# default-information originate metric 100 metric-type 1 +# distance ospf intra-area 85 +# max-lsa 80000 40 ignore-count 3 ignore-time 6 reset-time 20 +# network 10.10.2.0/24 area 0.0.0.0 +# network 10.10.3.0/24 area 0.0.0.0 +# redistribute static +# router-id 170.21.0.4 +# router ospf 2 vrf vrf01, +# area 0.0.0.9 default-cost 20 +# max-lsa 80000 40 ignore-count 3 ignore-time 6 reset-time 20 +# router ospf 3 vrf vrf02 +# redistribute static + + - name: Parse Ospf configs + arista.eos.eos_ospfv2: + running_config: "{{ lookup('file', './parsed.cfg') }}" + state: parsed + +# "parsed": [ +# { +# "processes": [ +# { +# "adjacency": { +# "exchange_start": { +# "threshold": 20045623 +# } +# }, +# "areas": [ +# { +# "filter": { +# "address": "10.1.1.0/24" +# }, +# "id": "0.0.0.2" +# }, +# { +# "id": "0.0.0.50", +# "range": { +# "address": "172.20.0.0/16", +# "cost": 34 +# } +# } +# ], +# "default_information": { +# "metric": 100, +# "metric_type": 1, +# "originate": true +# }, +# "distance": { +# "intra_area": 85 +# }, +# "max_lsa": { +# "count": 80000, +# "ignore_count": 3, +# "ignore_time": 6, +# "reset_time": 20, +# "threshold": 40 +# }, +# "networks": [ +# { +# "area": "0.0.0.0", +# "prefix": "10.10.2.0/24" +# }, +# { +# "area": "0.0.0.0", +# "prefix": "10.10.3.0/24" +# } +# ], +# "process_id": 1, +# "redistribute": [ +# { +# "routes": "static" +# } +# ], +# "router_id": "170.21.0.4" +# }, +# { +# "areas": [ +# { +# "default_cost": 20, +# "id": "0.0.0.9" +# } +# ], +# "max_lsa": { +# "count": 80000, +# "ignore_count": 3, +# "ignore_time": 6, +# "reset_time": 20, +# "threshold": 40 +# }, +# "process_id": 2, +# "vrf": "vrf01," +# }, +# { +# "process_id": 3, +# "redistribute": [ +# { +# "routes": "static" +# } +# ], +# "vrf": "vrf02" +# } +# ] +# } +# ] + +# Using rendered: +# -------------- + + - name: replace Ospf configs + arista.eos.eos_ospfv2: + config: + - processes: + - process_id: 1 + adjacency: + exchange_start: + threshold: 20045623 + areas: + - filter: + address: 10.1.1.0/24 + id: 0.0.0.2 + - id: 0.0.0.50 + range: + address: 172.20.0.0/16 + cost: 34 + default_information: + metric: 100 + metric_type: 1 + originate: true + distance: + intra_area: 85 + max_lsa: + count: 8000 + ignore_count: 3 + ignore_time: 6 + reset_time: 20 + threshold: 40 + networks: + - area: 0.0.0.0 + prefix: 10.10.2.0/24 + - area: 0.0.0.0 + prefix: 10.10.3.0/24 + redistribute: + - routes: static + router_id: 170.21.0.4 + state: rendered + +# "rendered": [ +# "router ospf 1", +# "adjacency exchange-start threshold 20045623", +# "area 0.0.0.2 filter 10.1.1.0/24", +# "area 0.0.0.50 range 172.20.0.0/16 cost 34", +# "default-information originate metric 100 metric-type 1", +# "distance ospf intra-area 85", +# "max-lsa 8000 40 ignore-count 3 ignore-time 6 reset-time 20", +# "network 10.10.2.0/24 area 0.0.0.0", +# "network 10.10.3.0/24 area 0.0.0.0", +# "redistribute static", +# "router-id 170.21.0.4" +# ] +# + +""" +RETURN = """ +before: + description: The configuration prior to the model invocation. + returned: always + sample: > + The configuration returned will always be in the same format + of the parameters above. + type: list +after: + description: The resulting configuration model invocation. + returned: when changed + sample: > + The configuration returned will always be in the same format + of the parameters above. + type: list +commands: + description: The set of commands pushed to the remote device. + returned: always + type: list + sample: ["router ospf 1", + "adjacency exchange-start threshold 20045623", + "area 0.0.0.2 filter 10.1.1.0/24", + "area 0.0.0.50 range 172.20.0.0/16 cost 34", + "default-information originate metric 100 metric-type 1", + "distance ospf intra-area 85", + "max-lsa 8000 40 ignore-count 3 ignore-time 6 reset-time 20", + "network 10.10.2.0/24 area 0.0.0.0", + "network 10.10.3.0/24 area 0.0.0.0", + "redistribute static", + "router-id 170.21.0.4"] +""" + + +from ansible.module_utils.basic import AnsibleModule +from ansible_collections.arista.eos.plugins.module_utils.network.eos.argspec.ospfv2.ospfv2 import ( + Ospfv2Args, +) +from ansible_collections.arista.eos.plugins.module_utils.network.eos.config.ospfv2.ospfv2 import ( + Ospfv2, +) + + +def main(): + """ + Main entry point for module execution + + :returns: the result form module invocation + """ + + required_if = [ + ("state", "merged", ("config",)), + ("state", "replaced", ("config",)), + ("state", "overridden", ("config",)), + ("state", "rendered", ("config",)), + ("state", "parsed", ("running_config",)), + ] + mutually_exclusive = [("config", "running_config")] + + module = AnsibleModule( + argument_spec=Ospfv2Args.argument_spec, + required_if=required_if, + supports_check_mode=True, + mutually_exclusive=mutually_exclusive, + ) + + result = Ospfv2(module).execute_module() + module.exit_json(**result) + + +if __name__ == "__main__": + main() diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/modules/eos_ospfv3.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/modules/eos_ospfv3.py new file mode 100644 index 00000000..777e1ece --- /dev/null +++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/modules/eos_ospfv3.py @@ -0,0 +1,1528 @@ +#!/usr/bin/python +# -*- coding: utf-8 -*- +# Copyright 2020 Red Hat +# GNU General Public License v3.0+ +# (see COPYING or https://www.gnu.org/licenses/gpl-3.0.txt) + +############################################# +# WARNING # +############################################# +# +# This file is auto generated by the resource +# module builder playbook. +# +# Do not edit this file manually. +# +# Changes to this file will be over written +# by the resource module builder. +# +# Changes should be made in the model used to +# generate this file or in the resource module +# builder template. +# +############################################# + +""" +The module file for eos_ospfv3 +""" + +from __future__ import absolute_import, division, print_function + +__metaclass__ = type + +DOCUMENTATION = """ +module: eos_ospfv3 +short_description: OSPFv3 resource module +description: This module configures and manages the attributes of ospfv3 on Arista + EOS platforms. +version_added: 1.1.0 +author: Gomathi Selvi Srinivasan (@GomathiselviS) +notes: +- Tested against Arista EOS 4.23.0F +- This module works with connection C(network_cli). See the L(EOS Platform Options,../network/user_guide/platform_eos.html). +options: + config: + description: A list of configurations for ospfv3. + type: dict + suboptions: + processes: + description: A list of dictionary specifying the ospfv3 processes. + type: list + elements: dict + suboptions: + vrf: + description: VRF name . + type: str + adjacency: + description: Configure adjacency options for OSPF instance. + type: dict + suboptions: + exchange_start: + description: Configure exchange-start options for OSPF instance. + type: dict + suboptions: + threshold: + description: Number of peers to bring up simultaneously. + type: int + auto_cost: + description: Set auto-cost. + type: dict + suboptions: + reference_bandwidth: + description: reference bandwidth in megabits per sec. + type: int + areas: + description: Specifies the configuration for OSPF areas + type: list + elements: dict + suboptions: + area_id: + description: Specifies a 32 bit number expressed in decimal or dotted-decimal + notation. + type: str + default_cost: + description: Specify the cost for default summary route in stub/NSSA + area. + type: int + authentication: + description: Configure authentication for the area incase of ospfv3. + type: dict + suboptions: + spi: + description: Specify the SPI value + type: int + algorithm: + description: Name of algorithm to be used. + type: str + choices: ['md5', 'sha1'] + encrypt_key: + description: If False, key string is not encrypted + type: bool + hidden_key: + description: If True, Specifies that a HIDDEN key will follow. + type: bool + key: + description: 128 bit MD5 key or 140 bit SHA1 key. + type: str + passphrase: + description: Passphrase String for deriving keys for authentication and encryption. + type: str + encryption: + description: Configure encryption for the area + type: dict + suboptions: + spi: + description: Specify the SPI value + type: int + encryption: + description: name of encryption to be used. + type: str + choices: ['3des-cbc', 'aes-128-cbc', 'aes-192-cbc', 'aes-256-cbc', 'null'] + algorithm: + description: name of the algorithm to be used. + type: str + choices: ['sha1', 'md5'] + encrypt_key: + description: If False, key string is not encrypted + type: bool + hidden_key: + description: If True, Specifies that a HIDDEN key will follow. + type: bool + key: + description: 128 bit MD5 key or 140 bit SHA1 key. + type: str + passphrase: + description: Passphrase String for deriving keys for authentication and encryption. + type: str + nssa: + description: Configures NSSA parameters. + type: dict + suboptions: + default_information_originate: + description: Originate default Type 7 LSA. + type: dict + suboptions: + metric: + description: Metric for default route. + type: int + metric_type: + description: Metric type for default route. + type: int + nssa_only: + description: Limit default advertisement to this NSSA area. + type: bool + set: + description: True if only default information orignate is set + type: bool + no_summary: + description: Filter all type-3 LSAs in the nssa area. + type: bool + nssa_only: + description: Disable Type-7 LSA p-bit setting + type: bool + translate: + description: Enable LSA translation. + type: bool + set: + description: True if only nssa is set + type: bool + stub: + description: Stub area. + type: dict + suboptions: + set: + description: True if only stub is set. + type: bool + summary_lsa: + description: If False , Filter all type-3 LSAs in the stub area. + type: bool + + bfd: + description: Enable BFD. + type: dict + suboptions: + all_interfaces: + description: Enable BFD on all interfaces. + type: bool + fips_restrictions: + description: Use FIPS compliant algorithms + type: bool + graceful_restart: + description: Enable graceful restart mode. + type: dict + suboptions: + grace_period: + description: Specify maximum time to wait for graceful-restart to + complete. + type: int + set: + description: When true sets the grace_fulrestart config alone. + type: bool + graceful_restart_helper: + description: If True, Enable graceful restart helper. + type: bool + log_adjacency_changes: + description: To configure link-state changes and transitions of OSPFv3 + neighbors. + type: dict + suboptions: + detail: + description: If true , configures the switch to log all link-state + changes. + type: bool + set: + description: When true sets the log_adjacency_changes config alone. + type: bool + max_metric: + description: Set maximum metric. + type: dict + suboptions: + router_lsa: + description: Maximum metric in self-originated router-LSAs. + type: dict + suboptions: + set: + description: + - Set router-lsa attribute. + type: bool + external_lsa: + description: Override external-lsa metric with max-metric value. + type: dict + suboptions: + set: + description: + - Set external-lsa attribute. + type: bool + max_metric_value: + description: + - Set max metric value for external LSAs. + type: int + include_stub: + description: Set maximum metric for stub links in router-LSAs. + type: bool + on_startup: + description: Set maximum metric temporarily after reboot. + type: dict + suboptions: + wait_period: + description: + - Wait period in seconds after startup. + type: int + wait_for_bgp: + description: + - Let BGP decide when to originate router-LSA with normal metric + type: bool + summary_lsa: + description: Override summary-lsa metric with max-metric value. + type: dict + suboptions: + set: + description: + - Set external-lsa attribute. + type: bool + max_metric_value: + description: + - Set max metric value for external LSAs. + type: int + passive_interface: + description: Include interface but without actively running OSPF. + type: bool + router_id: + description: 32-bit number assigned to a router running OSPFv3. + type: str + shutdown: + description: Disable the OSPF instance. + type: bool + timers: + description: Configure OSPF timers. + type: dict + suboptions: + lsa: + description: Configure OSPF LSA timers. + type: int + out_delay: + description: Configure out-delay timer. + type: int + pacing: + description: Configure OSPF packet pacing. + type: int + throttle: + description: Configure SPF timers + type: dict + suboptions: + lsa: + description: Configure threshold for retransmission of lsa + type: bool + spf: + description: Configure time between SPF calculations + type: bool + initial: + description: Initial SPF schedule delay in msecs. + type: int + min: + description: Min Hold time between two SPFs in msecs + type: int + max: + description: Max wait time between two SPFs in msecs. + type: int + + address_family: + description: Enable address family and enter its config mode + type: list + elements: dict + suboptions: + afi: + description: address family . + type: str + choices: + - ipv4 + - ipv6 + adjacency: + description: Configure adjacency options for OSPF instance. + type: dict + suboptions: + exchange_start: + description: Configure exchange-start options for OSPF instance. + type: dict + suboptions: + threshold: + description: Number of peers to bring up simultaneously. + type: int + auto_cost: + description: Set auto-cost. + type: dict + suboptions: + reference_bandwidth: + description: reference bandwidth in megabits per sec. + type: int + areas: + description: Specifies the configuration for OSPF areas + type: list + elements: dict + suboptions: + area_id: + description: Specifies a 32 bit number expressed in decimal or dotted-decimal + notation. + type: str + default_cost: + description: Specify the cost for default summary route in stub/NSSA + area. + type: int + authentication: + description: Configure authentication for the area incase of ospfv3. + type: dict + suboptions: + spi: + description: Specify the SPI value + type: int + algorithm: + description: Name of algorithm to be used. + type: str + choices: ['md5', 'sha1'] + encrypt_key: + description: If False, key string is not encrypted + type: bool + hidden_key: + description: If True, Specifies that a HIDDEN key will follow. + type: bool + key: + description: 128 bit MD5 key or 140 bit SHA1 key. + type: str + passphrase: + description: Passphrase String for deriving keys for authentication and encryption. + type: str + encryption: + description: Configure encryption for the area + type: dict + suboptions: + spi: + description: Specify the SPI value + type: int + encryption: + description: name of encryption to be used. + type: str + choices: ['3des-cbc', 'aes-128-cbc', 'aes-192-cbc', 'aes-256-cbc', 'null'] + algorithm: + description: name of the algorithm to be used. + type: str + choices: ['sha1', 'md5'] + encrypt_key: + description: If False, key string is not encrypted + type: bool + hidden_key: + description: If True, Specifies that a HIDDEN key will follow. + type: bool + key: + description: 128 bit MD5 key or 140 bit SHA1 key. + type: str + passphrase: + description: Passphrase String for deriving keys for authentication and encryption. + type: str + nssa: + description: Configures NSSA parameters. + type: dict + suboptions: + default_information_originate: + description: Originate default Type 7 LSA. + type: dict + suboptions: + metric: + description: Metric for default route. + type: int + metric_type: + description: Metric type for default route. + type: int + nssa_only: + description: Limit default advertisement to this NSSA area. + type: bool + set: + description: True if only default information orignate is set + type: bool + no_summary: + description: Filter all type-3 LSAs in the nssa area. + type: bool + nssa_only: + description: Disable Type-7 LSA p-bit setting + type: bool + translate: + description: Enable LSA translation. + type: bool + set: + description: True if only nssa is set + type: bool + ranges: + description: Configure route summarization. + type: list + elements: dict + suboptions: + address: + description: IP address. + type: str + subnet_address: + description: IP address with mask length + type: str + subnet_mask: + description: IP subnet mask + type: str + advertise: + description: Enable Advertisement of the range. + type: bool + cost: + description: Configures the metric. + type: int + stub: + description: Stub area. + type: dict + suboptions: + set: + description: True if only stub is set + type: bool + summary_lsa: + description: If False , Filter all type-3 LSAs in the stub area. + type: bool + + bfd: + description: Enable BFD. + type: dict + suboptions: + all_interfaces: + description: Enable BFD on all interfaces. + type: bool + default_information: + description: Control distribution of default information. + type: dict + suboptions: + originate: + description: Distribute a default route. + type: bool + always: + description: Always advertise default route. + type: bool + metric: + description: Metric for default route. + type: int + metric_type: + description: Metric type for default route. + type: int + route_map: + description: Specify which route-map to use. + type: str + default_metric: + description: Configure the default metric for redistributed routes. + type: int + distance: + description: Specifies the administrative distance for routes. + type: int + fips_restrictions: + description: Use FIPS compliant algorithms + type: bool + graceful_restart: + description: Enable graceful restart mode. + type: dict + suboptions: + grace_period: + description: Specify maximum time to wait for graceful-restart to complete. + type: int + set: + description: When true sets the grace_fulrestart config alone. + type: bool + graceful_restart_helper: + description: If True, Enable graceful restart helper. + type: bool + log_adjacency_changes: + description: To configure link-state changes and transitions of OSPFv3 + neighbors. + type: dict + suboptions: + detail: + description: If true , configures the switch to log all link-state + changes. + type: bool + set: + description: When true sets the log_adjacency_changes config alone. + type: bool + max_metric: + description: Set maximum metric. + type: dict + suboptions: + router_lsa: + description: Maximum metric in self-originated router-LSAs. + type: dict + suboptions: + set: + description: + - Set router-lsa attribute. + type: bool + external_lsa: + description: Override external-lsa metric with max-metric value. + type: dict + suboptions: + set: + description: + - Set external-lsa attribute. + type: bool + max_metric_value: + description: + - Set max metric value for external LSAs. + type: int + include_stub: + description: Set maximum metric for stub links in router-LSAs. + type: bool + on_startup: + description: Set maximum metric temporarily after reboot. + type: dict + suboptions: + wait_period: + description: + - Wait period in seconds after startup. + type: int + wait_for_bgp: + description: + - Let BGP decide when to originate router-LSA with normal metric + type: bool + summary_lsa: + description: Override summary-lsa metric with max-metric value. + type: dict + suboptions: + set: + description: + - Set external-lsa attribute. + type: bool + max_metric_value: + description: + - Set max metric value for external LSAs. + type: int + maximum_paths: + description: Maximum number of next-hops in an ECMP route. + type: int + passive_interface: + description: Include interface but without actively running OSPF. + type: bool + redistribute: + description: Specifies the routes to be redistributed. + type: list + elements: dict + suboptions: + routes: + description: Route types (BGP,static,connected) + type: str + choices: ['bgp', 'connected', 'static'] + route_map: + description: Specify which route map to use. + type: str + router_id: + description: 32-bit number assigned to a router running OSPFv3. + type: str + shutdown: + description: Disable the OSPF instance. + type: bool + timers: + description: Configure OSPF timers. + type: dict + suboptions: + lsa: + description: Configure OSPF LSA timers. + type: int + out_delay: + description: Configure out-delay timer. + type: int + pacing: + description: Configure OSPF packet pacing. + type: int + throttle: + description: Configure SPF timers + type: dict + suboptions: + lsa: + description: Configure threshold for retransmission of lsa + type: bool + spf: + description: Configure time between SPF calculations + type: bool + initial: + description: Initial SPF schedule delay in msecs. + type: int + min: + description: Min Hold time between two SPFs in msecs + type: int + max: + description: Max wait time between two SPFs in msecs. + type: int + running_config: + description: + - This option is used only with state I(parsed). + - The value of this option should be the output received from the EOS device by + executing the command B(show running-config | section ospfv3). + - The state I(parsed) reads the configuration from C(running_config) option and + transforms it into Ansible structured data as per the resource module's argspec + and the value is then returned in the I(parsed) key within the result. + type: str + state: + description: + - The state the configuration should be left in. + type: str + choices: [deleted, merged, overridden, replaced, gathered, rendered, parsed] + default: merged +""" + +EXAMPLES = """ + +# Using merged + +# Before state + +# veos#show running-config | section ospfv3 +# veos# + + + - arista.eos.eos_ospfv3: + config: + processes: + - address_family: + - timers: + lsa: 22 + graceful_restart: + grace_period: 35 + afi: "ipv6" + timers: + pacing: 55 + fips_restrictions: True + router_id: "2.2.2.2" + vrf: "vrfmerge" + + +# After state + +# veos#show running-config | section ospfv3 +# router ospfv3 vrf vrfmerge +# router-id 2.2.2.2 +# fips restrictions +# timers pacing flood 55 +# ! +# address-family ipv6 +# fips restrictions +# timers lsa arrival 22 +# graceful-restart grace-period 35 +# veos# + +# Module Execution +# "after": { +# "processes": [ +# { +# "address_family": [ +# { +# "afi": "ipv6", +# "fips_restrictions": true, +# "graceful_restart": { +# "grace_period": 35 +# }, +# "timers": { +# "lsa": 22 +# } +# } +# ], +# "fips_restrictions": true, +# "router_id": "2.2.2.2", +# "timers": { +# "pacing": 55 +# }, +# "vrf": "vrfmerge" +# } +# ] +# }, +# "before": {}, +# "changed": true, +# "commands": [ +# "router ospfv3 vrf vrfmerge", +# "address-family ipv6", +# "graceful-restart grace-period 35", +# "timers lsa arrival 22", +# "exit", +# "timers pacing flood 55", +# "fips restrictions", +# "router-id 2.2.2.2", +# "exit" +# ], + + +# using replaced + +# before state + +# veos#show running-config | section ospfv3 +# router ospfv3 +# fips restrictions +# area 0.0.0.0 encryption ipsec spi 43 esp null md5 passphrase 7 h8pZp9eprTYjjoY/NKFFe0Ei7x03Y7dyLotRhI0a5t4= +# ! +# router ospfv3 vrf vrfmerge +# router-id 2.2.2.2 +# fips restrictions +# timers pacing flood 55 +# ! +# address-family ipv6 +# fips restrictions +# timers lsa arrival 22 +# graceful-restart grace-period 35 +# veos# + + + - arista.eos.eos_ospfv3: + config: + processes: + - areas: + - area_id: "0.0.0.0" + encryption: + spi: 43 + encryption: "null" + algorithm: "md5" + encrypt_key: False + passphrase: "7hl8FV3lZ6H1mAKpjL47hQ==" + vrf: "default" + address_family: + - afi: "ipv4" + router_id: "7.1.1.1" + state: replaced + +# After state +# veos#show running-config | section ospfv3 +# router ospfv3 +# area 0.0.0.0 encryption ipsec spi 43 esp null md5 passphrase 7 h8pZp9eprTYjjoY/NKFFe0Ei7x03Y7dyLotRhI0a5t4= +# ! +# router ospfv3 vrf vrfmerge +# passive-interface default +# ! +# address-family ipv6 +# area 0.0.0.3 range 10.1.2.0/24 +# area 0.0.0.3 range 60.1.0.0/16 cost 30 +# veos# + +# Module execution + +# "after": { +# "processes": [ +# { +# "areas": [ +# { +# "area_id": "0.0.0.0", +# "encryption": { +# "algorithm": "md5", +# "encryption": "null", +# "hidden_key": true, +# "passphrase": "h8pZp9eprTYjjoY/NKFFe0Ei7x03Y7dyLotRhI0a5t4=" +# } +# } +# ], +# "vrf": "default" +# }, +# { +# "address_family": [ +# { +# "afi": "ipv6", +# "areas": [ +# { +# "area_id": "0.0.0.3", +# "ranges": [ +# { +# "address": "10.1.2.0/24" +# }, +# { +# "address": "60.1.0.0/16", +# "cost": 30 +# } +# ] +# } +# ] +# } +# ], +# "passive_interface": true, +# "vrf": "vrfmerge" +# } +# ] +# }, +# "before": { +# "processes": [ +# { +# "areas": [ +# { +# "area_id": "0.0.0.0", +# "encryption": { +# "algorithm": "md5", +# "encryption": "null", +# "hidden_key": true, +# "passphrase": "h8pZp9eprTYjjoY/NKFFe0Ei7x03Y7dyLotRhI0a5t4=" +# } +# } +# ], +# "fips_restrictions": true, +# "vrf": "default" +# }, +# { +# "address_family": [ +# { +# "afi": "ipv6", +# "fips_restrictions": true, +# "graceful_restart": { +# "grace_period": 35 +# }, +# "timers": { +# "lsa": 22 +# } +# } +# ], +# "fips_restrictions": true, +# "router_id": "2.2.2.2", +# "timers": { +# "pacing": 55 +# }, +# "vrf": "vrfmerge" +# } +# ] +# }, +# "changed": true, +# "commands": [ +# "router ospfv3 vrf vrfmerge", +# "address-family ipv6", +# "no fips restrictions", +# "no graceful-restart", +# "no timers lsa arrival 22", +# "area 0.0.0.3 range 10.1.2.2/24 advertise", +# "area 0.0.0.3 range 60.1.1.1 255.255.0.0 cost 30", +# "exit", +# "passive-interface default", +# "no router-id", +# "no fips restrictions", +# "no timers pacing flood 55", +# "exit" +# ], + + +# using overridden + +# before state + +# veos#show running-config | section ospfv3 +# router ospfv3 +# area 0.0.0.0 encryption ipsec spi 43 esp null md5 passphrase 7 h8pZp9eprTYjjoY/NKFFe0Ei7x03Y7dyLotRhI0a5t4= +# ! +# router ospfv3 vrf vrfmerge +# passive-interface default +# ! +# address-family ipv6 +# area 0.0.0.3 range 10.1.2.0/24 +# area 0.0.0.3 range 60.1.0.0/16 cost 30 +# veos# + + + - arista.eos.eos_ospfv3: + config: + processes: + - address_family: + - areas: + - area_id: "0.0.0.3" + ranges: + - address: 10.1.2.2/24 + advertise: True + - address: 60.1.1.1 + subnet_mask: 255.255.0.0 + cost: 30 + afi: "ipv6" + passive_interface: True + vrf: "vrfmerge" + state: overridden + +# After state + +# veos#show running-config | section ospfv3 +# router ospfv3 vrf vrfmerge +# passive-interface default +# ! +# address-family ipv6 +# area 0.0.0.3 range 10.1.2.0/24 +# area 0.0.0.3 range 60.1.0.0/16 cost 30 +# veos# + + + +# Module execution + +# "after": { +# "processes": [ +# { +# "address_family": [ +# { +# "afi": "ipv6", +# "areas": [ +# { +# "area_id": "0.0.0.3", +# "ranges": [ +# { +# "address": "10.1.2.0/24" +# }, +# { +# "address": "60.1.0.0/16", +# "cost": 30 +# } +# ] +# } +# ] +# } +# ], +# "passive_interface": true, +# "vrf": "vrfmerge" +# } +# ] +# }, +# "before": { +# "processes": [ +# { +# "areas": [ +# { +# "area_id": "0.0.0.0", +# "encryption": { +# "algorithm": "md5", +# "encryption": "null", +# "hidden_key": true, +# "passphrase": "h8pZp9eprTYjjoY/NKFFe0Ei7x03Y7dyLotRhI0a5t4=" +# } +# } +# ], +# "vrf": "default" +# }, +# { +# "address_family": [ +# { +# "afi": "ipv6", +# "areas": [ +# { +# "area_id": "0.0.0.3", +# "ranges": [ +# { +# "address": "10.1.2.0/24" +# }, +# { +# "address": "60.1.0.0/16", +# "cost": 30 +# } +# ] +# } +# ] +# } +# ], +# "passive_interface": true, +# "vrf": "vrfmerge" +# } +# ] +# }, +# "changed": true, +# "commands": [ +# "no router ospfv3", +# "router ospfv3 vrf vrfmerge", +# "address-family ipv6", +# "no area 0.0.0.3 range 10.1.2.0/24", +# "no area 0.0.0.3 range 60.1.0.0/16 cost 30", +# "area 0.0.0.3 range 10.1.2.2/24 advertise", +# "area 0.0.0.3 range 60.1.1.1 255.255.0.0 cost 30", +# "exit", +# "exit" +# ], + +# using deleted + +# Before state + +# veos#show running-config | section ospfv3 +# router ospfv3 +# area 0.0.0.0 encryption ipsec spi 43 esp null md5 passphrase 7 h8pZp9eprTYjjoY/NKFFe0Ei7x03Y7dyLotRhI0a5t4= +# ! +# router ospfv3 vrf vrfmerge +# passive-interface default +# ! +# address-family ipv4 +# redistribute connected +# redistribute static route-map MAP01 +# area 0.0.0.3 range 10.1.2.0/24 +# area 0.0.0.3 range 60.1.0.0/16 cost 30 +# ! +# address-family ipv6 +# area 0.0.0.3 range 10.1.2.0/24 +# area 0.0.0.3 range 60.1.0.0/16 cost 30 +# veos# + + + - arista.eos.eos_ospfv3: + config: + processes: + - vrf: "default" + state: deleted + +# After state + +# veos#show running-config | section ospfv3 +# router ospfv3 vrf vrfmerge +# passive-interface default +# ! +# address-family ipv4 +# redistribute connected +# redistribute static route-map MAP01 +# area 0.0.0.3 range 10.1.2.0/24 +# area 0.0.0.3 range 60.1.0.0/16 cost 30 +# ! +# address-family ipv6 +# area 0.0.0.3 range 10.1.2.0/24 +# area 0.0.0.3 range 60.1.0.0/16 cost 30 +# veos# + + +# Module execution +# "after": { +# "processes": [ +# { +# "address_family": [ +# { +# "afi": "ipv4", +# "areas": [ +# { +# "area_id": "0.0.0.3", +# "ranges": [ +# { +# "address": "10.1.2.0/24" +# }, +# { +# "address": "60.1.0.0/16", +# "cost": 30 +# } +# ] +# } +# ], +# "redistribute": [ +# { +# "routes": "connected" +# }, +# { +# "route_map": "MAP01", +# "routes": "static" +# } +# ] +# }, +# { +# "afi": "ipv6", +# "areas": [ +# { +# "area_id": "0.0.0.3", +# "ranges": [ +# { +# "address": "10.1.2.0/24" +# }, +# { +# "address": "60.1.0.0/16", +# "cost": 30 +# } +# ] +# } +# ] +# } +# ], +# "passive_interface": true, +# "vrf": "vrfmerge" +# } +# ] +# }, +# "before": { +# "processes": [ +# { +# "areas": [ +# { +# "area_id": "0.0.0.0", +# "encryption": { +# "algorithm": "md5", +# "encryption": "null", +# "hidden_key": true, +# "passphrase": "h8pZp9eprTYjjoY/NKFFe0Ei7x03Y7dyLotRhI0a5t4=" +# } +# } +# ], +# "vrf": "default" +# }, +# { +# "address_family": [ +# { +# "afi": "ipv4", +# "areas": [ +# { +# "area_id": "0.0.0.3", +# "ranges": [ +# { +# "address": "10.1.2.0/24" +# }, +# { +# "address": "60.1.0.0/16", +# "cost": 30 +# } +# ] +# } +# ], +# "redistribute": [ +# { +# "routes": "connected" +# }, +# { +# "route_map": "MAP01", +# "routes": "static" +# } +# ] +# }, +# { +# "afi": "ipv6", +# "areas": [ +# { +# "area_id": "0.0.0.3", +# "ranges": [ +# { +# "address": "10.1.2.0/24" +# }, +# { +# "address": "60.1.0.0/16", +# "cost": 30 +# } +# ] +# } +# ] +# } +# ], +# "passive_interface": true, +# "vrf": "vrfmerge" +# } +# ] +# }, +# "changed": true, +# "commands": [ +# "no router ospfv3" +# ], + +# using parsed + +# parsed_ospfv3.cfg + +# router ospfv3 +# fips restrictions +# area 0.0.0.20 stub +# area 0.0.0.20 authentication ipsec spi 33 sha1 passphrase 7 4O8T3zo4xBdRWXBnsnK934o9SEb+jEhHUN6+xzZgCo2j9EnQBUvtwNxxLEmYmm6w +# area 0.0.0.40 default-cost 45 +# area 0.0.0.40 stub +# timers pacing flood 7 +# adjacency exchange-start threshold 11 +# ! +# address-family ipv4 +# fips restrictions +# redistribute connected +# ! +# address-family ipv6 +# router-id 10.1.1.1 +# fips restrictions +# ! +# router ospfv3 vrf vrf01 +# bfd all-interfaces +# fips restrictions +# area 0.0.0.0 encryption ipsec spi 256 esp null sha1 passphrase 7 7hl8FV3lZ6H1mAKpjL47hQ== +# log-adjacency-changes detail +# ! +# address-family ipv4 +# passive-interface default +# fips restrictions +# redistribute connected route-map MAP01 +# maximum-paths 100 +# ! +# address-family ipv6 +# fips restrictions +# area 0.0.0.10 nssa no-summary +# default-information originate route-map DefaultRouteFilter +# max-metric router-lsa external-lsa 25 summary-lsa +# ! +# router ospfv3 vrf vrf02 +# fips restrictions +# ! +# address-family ipv6 +# router-id 10.17.0.3 +# distance ospf intra-area 200 +# fips restrictions +# area 0.0.0.1 stub +# timers throttle spf 56 56 56 +# timers out-delay 10 + + + - arista.eos.eos_ospfv3: + running_config: "{{ lookup('file', './parsed_ospfv3.cfg') }}" + state: parsed + +# Module execution + +# "parsed": { +# "processes": [ +# { +# "address_family": [ +# { +# "afi": "ipv4", +# "fips_restrictions": true, +# "redistribute": [ +# { +# "routes": "connected" +# } +# ] +# }, +# { +# "afi": "ipv6", +# "fips_restrictions": true, +# "router_id": "10.1.1.1" +# } +# ], +# "adjacency": { +# "exchange_start": { +# "threshold": 11 +# } +# }, +# "areas": [ +# { +# "area_id": "0.0.0.20", +# "authentication": { +# "algorithm": "sha1", +# "hidden_key": true, +# "passphrase": "4O8T3zo4xBdRWXBnsnK934o9SEb+jEhHUN6+xzZgCo2j9EnQBUvtwNxxLEmYmm6w", +# "spi": 33 +# }, +# "stub": { +# "set": true +# } +# }, +# { +# "area_id": "0.0.0.40", +# "default_cost": 45, +# "stub": { +# "set": true +# } +# } +# ], +# "fips_restrictions": true, +# "timers": { +# "pacing": 7 +# }, +# "vrf": "default" +# }, +# { +# "address_family": [ +# { +# "afi": "ipv4", +# "fips_restrictions": true, +# "maximum_paths": 100, +# "passive_interface": true, +# "redistribute": [ +# { +# "route_map": "MAP01", +# "routes": "connected" +# } +# ] +# }, +# { +# "afi": "ipv6", +# "areas": [ +# { +# "area_id": "0.0.0.10", +# "nssa": { +# "no_summary": true +# } +# } +# ], +# "default_information": { +# "originate": true, +# "route_map": "DefaultRouteFilter" +# }, +# "fips_restrictions": true, +# "max_metric": { +# "router_lsa": { +# "external_lsa": { +# "max_metric_value": 25 +# }, +# "summary_lsa": { +# "set": true +# } +# } +# } +# } +# ], +# "areas": [ +# { +# "area_id": "0.0.0.0", +# "encryption": { +# "algorithm": "sha1", +# "encryption": "null", +# "hidden_key": true, +# "passphrase": "7hl8FV3lZ6H1mAKpjL47hQ==" +# } +# } +# ], +# "bfd": { +# "all_interfaces": true +# }, +# "fips_restrictions": true, +# "log_adjacency_changes": { +# "detail": true +# }, +# "vrf": "vrf01" +# }, +# { +# "address_family": [ +# { +# "afi": "ipv6", +# "areas": [ +# { +# "area_id": "0.0.0.1", +# "stub": { +# "set": true +# } +# } +# ], +# "distance": 200, +# "fips_restrictions": true, +# "router_id": "10.17.0.3", +# "timers": { +# "out_delay": 10, +# "throttle": { +# "initial": 56, +# "max": 56, +# "min": 56, +# "spf": true +# } +# } +# } +# ], +# "fips_restrictions": true, +# "vrf": "vrf02" +# } +# ] + +# using gathered + +# native config + +# veos#show running-config | section ospfv3 +# router ospfv3 vrf vrfmerge +# passive-interface default +# ! +# address-family ipv4 +# redistribute connected +# redistribute static route-map MAP01 +# area 0.0.0.3 range 10.1.2.0/24 +# area 0.0.0.3 range 60.1.0.0/16 cost 30 +# ! +# address-family ipv6 +# area 0.0.0.3 range 10.1.2.0/24 +# area 0.0.0.3 range 60.1.0.0/16 cost 30 +# veos# + + + - arista.eos.eos_ospfv3: + state: gathered + +# module execution + +# "gathered": { +# "processes": [ +# { +# "address_family": [ +# { +# "afi": "ipv4", +# "areas": [ +# { +# "area_id": "0.0.0.3", +# "ranges": [ +# { +# "address": "10.1.2.0/24" +# }, +# { +# "address": "60.1.0.0/16", +# "cost": 30 +# } +# ] +# } +# ], +# "redistribute": [ +# { +# "routes": "connected" +# }, +# { +# "route_map": "MAP01", +# "routes": "static" +# } +# ] +# }, +# { +# "afi": "ipv6", +# "areas": [ +# { +# "area_id": "0.0.0.3", +# "ranges": [ +# { +# "address": "10.1.2.0/24" +# }, +# { +# "address": "60.1.0.0/16", +# "cost": 30 +# } +# ] +# } +# ] +# } +# ], +# "passive_interface": true, +# "vrf": "vrfmerge" +# } +# ] + +# using rendered + + - arista.eos.eos_ospfv3: + config: + processes: + - address_family: + - timers: + lsa: 22 + graceful_restart: + grace_period: 35 + afi: "ipv6" + timers: + pacing: 55 + fips_restrictions: True + router_id: "2.2.2.2" + vrf: "vrfmerge" + state: rendered + +# module execution + +# "rendered": [ +# "router ospfv3 vrf vrfmerge", +# "address-family ipv6", +# "graceful-restart grace-period 35", +# "timers lsa arrival 22", +# "exit", +# "timers pacing flood 55", +# "fips restrictions", +# "router-id 2.2.2.2", +# "exit" +# ] + + +""" + +from ansible.module_utils.basic import AnsibleModule +from ansible_collections.arista.eos.plugins.module_utils.network.eos.argspec.ospfv3.ospfv3 import ( + Ospfv3Args, +) +from ansible_collections.arista.eos.plugins.module_utils.network.eos.config.ospfv3.ospfv3 import ( + Ospfv3, +) + + +def main(): + """ + Main entry point for module execution + + :returns: the result form module invocation + """ + module = AnsibleModule( + argument_spec=Ospfv3Args.argument_spec, + mutually_exclusive=[["config", "running_config"]], + required_if=[ + ["state", "merged", ["config"]], + ["state", "replaced", ["config"]], + ["state", "overridden", ["config"]], + ["state", "rendered", ["config"]], + ["state", "parsed", ["running_config"]], + ], + supports_check_mode=False, + ) + + result = Ospfv3(module).execute_module() + module.exit_json(**result) + + +if __name__ == "__main__": + main() diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/modules/eos_static_route.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/modules/eos_static_route.py new file mode 100644 index 00000000..da6753a5 --- /dev/null +++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/modules/eos_static_route.py @@ -0,0 +1,372 @@ +#!/usr/bin/python +# -*- coding: utf-8 -*- + +# Copyright: (c) 2017, Ansible by Red Hat, inc +# 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 + + +DOCUMENTATION = """ +module: eos_static_route +author: Trishna Guha (@trishnaguha) +short_description: (deprecated, removed after 2022-06-01) Manage static IP + routes on Arista EOS network devices +description: +- This module provides declarative management of static IP routes on Arista EOS network + devices. +version_added: 1.0.0 +deprecated: + alternative: eos_static_routes + why: Updated modules with more functionality + removed_at_date: '2022-06-01' +notes: +- Tested against EOS 4.15 +options: + address: + description: + - Network address with prefix of the static route. + aliases: + - prefix + type: str + next_hop: + description: + - Next hop IP of the static route. + type: str + vrf: + description: + - VRF for static route. + default: default + type: str + admin_distance: + description: + - Admin distance of the static route. + default: 1 + type: int + aggregate: + description: List of static route definitions + type: list + elements: dict + suboptions: + address: + description: + - Network address with prefix of the static route. + aliases: + - prefix + type: str + required: True + next_hop: + description: + - Next hop IP of the static route. + type: str + vrf: + description: + - VRF for static route. + default: default + type: str + admin_distance: + description: + - Admin distance of the static route. + type: int + state: + description: + - State of the static route configuration. + choices: + - present + - absent + type: str + state: + description: + - State of the static route configuration. + default: present + choices: + - present + - absent + type: str +extends_documentation_fragment: +- arista.eos.eos + +""" + +EXAMPLES = """ +- name: configure static route + arista.eos.eos_static_route: + address: 10.0.2.0/24 + next_hop: 10.8.38.1 + admin_distance: 2 +- name: delete static route + arista.eos.eos_static_route: + address: 10.0.2.0/24 + next_hop: 10.8.38.1 + state: absent +- name: configure static routes using aggregate + arista.eos.eos_static_route: + aggregate: + - {address: 10.0.1.0/24, next_hop: 10.8.38.1} + - {address: 10.0.3.0/24, next_hop: 10.8.38.1} +- name: Delete static route using aggregate + arista.eos.eos_static_route: + aggregate: + - {address: 10.0.1.0/24, next_hop: 10.8.38.1} + - {address: 10.0.3.0/24, next_hop: 10.8.38.1} + state: absent +""" + +RETURN = """ +commands: + description: The list of configuration mode commands to send to the device + returned: always + type: list + sample: + - ip route 10.0.2.0/24 10.8.38.1 3 + - no ip route 10.0.2.0/24 10.8.38.1 +""" + +import re + +from copy import deepcopy + +from ansible.module_utils.basic import AnsibleModule +from ansible_collections.ansible.netcommon.plugins.module_utils.network.common.utils import ( + is_masklen, + validate_ip_address, +) +from ansible_collections.ansible.netcommon.plugins.module_utils.network.common.utils import ( + remove_default_spec, + validate_prefix, +) +from ansible_collections.arista.eos.plugins.module_utils.network.eos.eos import ( + get_config, + load_config, +) +from ansible_collections.arista.eos.plugins.module_utils.network.eos.eos import ( + eos_argument_spec, +) + + +def is_address(value): + if value: + address = value.split("/") + if is_masklen(address[1]) and validate_ip_address(address[0]): + return True + return False + + +def is_hop(value): + if value: + if validate_ip_address(value): + return True + return False + + +def map_obj_to_commands(updates, module): + commands = list() + want, have = updates + + for w in want: + address = w["address"] + next_hop = w["next_hop"] + admin_distance = w["admin_distance"] + vrf = w["vrf"] + state = w["state"] + del w["state"] + + if state == "absent" and w in have: + if vrf == "default": + commands.append("no ip route %s %s" % (address, next_hop)) + else: + commands.append( + "no ip route vrf %s %s %s" % (vrf, address, next_hop) + ) + elif state == "present" and w not in have: + if vrf == "default": + commands.append( + "ip route %s %s %d" % (address, next_hop, admin_distance) + ) + else: + commands.append( + "ip route vrf %s %s %s %d" + % (vrf, address, next_hop, admin_distance) + ) + + return commands + + +def map_params_to_obj(module, required_together=None): + obj = [] + + aggregate = module.params.get("aggregate") + if aggregate: + for item in aggregate: + for key in item: + if item.get(key) is None: + item[key] = module.params[key] + + module._check_required_together(required_together, item) + d = item.copy() + + obj.append(d) + else: + obj.append( + { + "address": module.params["address"].strip(), + "next_hop": module.params["next_hop"].strip(), + "admin_distance": module.params["admin_distance"], + "state": module.params["state"], + "vrf": module.params["vrf"], + } + ) + + return obj + + +def map_config_to_obj(module): + objs = [] + + try: + out = get_config(module, flags=["| include ip.route"]) + except IndexError: + out = "" + if out: + lines = out.splitlines() + for line in lines: + obj = {} + add_match = re.search(r"ip route ([\d\./]+)", line, re.M) + if add_match: + obj["vrf"] = "default" + address = add_match.group(1) + if is_address(address): + obj["address"] = address + hop_match = re.search( + r"ip route {0} ([\d\./]+)".format(address), line, re.M + ) + if hop_match: + hop = hop_match.group(1) + if is_hop(hop): + obj["next_hop"] = hop + dist_match = re.search( + r"ip route {0} {1} (\d+)".format(address, hop), + line, + re.M, + ) + if dist_match: + distance = dist_match.group(1) + obj["admin_distance"] = int(distance) + else: + obj["admin_distance"] = 1 + + vrf_match = re.search( + r"ip route vrf ([\w]+) ([\d\./]+)", line, re.M + ) + if vrf_match: + vrf = vrf_match.group(1) + obj["vrf"] = vrf + address = vrf_match.group(2) + if is_address(address): + obj["address"] = address + hop_vrf_match = re.search( + r"ip route vrf {0} {1} ([\d\./]+)".format(vrf, address), + line, + re.M, + ) + if hop_vrf_match: + hop = hop_vrf_match.group(1) + if is_hop(hop): + obj["next_hop"] = hop + dist_vrf_match = re.search( + r"ip route vrf {0} {1} {2} (\d+)".format( + vrf, address, hop + ), + line, + re.M, + ) + if dist_vrf_match: + distance = dist_vrf_match.group(1) + obj["admin_distance"] = int(distance) + else: + obj["admin_distance"] = 1 + + objs.append(obj) + + return objs + + +def main(): + """ main entry point for module execution + """ + element_spec = dict( + address=dict(type="str", aliases=["prefix"]), + next_hop=dict(type="str"), + vrf=dict(type="str", default="default"), + admin_distance=dict(default=1, type="int"), + state=dict(default="present", choices=["present", "absent"]), + ) + + aggregate_spec = deepcopy(element_spec) + aggregate_spec["address"] = dict(required=True, aliases=["prefix"]) + + # remove default in aggregate spec, to handle common arguments + remove_default_spec(aggregate_spec) + + aggregate_spec["vrf"].update(default="default") + argument_spec = dict( + aggregate=dict(type="list", elements="dict", options=aggregate_spec) + ) + + argument_spec.update(element_spec) + argument_spec.update(eos_argument_spec) + + required_one_of = [["aggregate", "address"]] + required_together = [["address", "next_hop"]] + mutually_exclusive = [["aggregate", "address"]] + + module = AnsibleModule( + argument_spec=argument_spec, + required_one_of=required_one_of, + required_together=required_together, + mutually_exclusive=mutually_exclusive, + supports_check_mode=True, + ) + + address = module.params["address"] + if address is not None: + prefix = address.split("/")[-1] + + if address and prefix: + if "/" not in address or not validate_ip_address( + address.split("/")[0] + ): + module.fail_json( + msg="{0} is not a valid IP address".format(address) + ) + + if not validate_prefix(prefix): + module.fail_json( + msg="Length of prefix should be between 0 and 32 bits" + ) + + warnings = list() + result = {"changed": False} + if warnings: + result["warnings"] = warnings + + want = map_params_to_obj(module) + have = map_config_to_obj(module) + commands = map_obj_to_commands((want, have), module) + result["commands"] = commands + + if commands: + commit = not module.check_mode + response = load_config(module, commands, commit=commit) + if response.get("diff") and module._diff: + result["diff"] = {"prepared": response.get("diff")} + result["session_name"] = response.get("session") + result["changed"] = True + + module.exit_json(**result) + + +if __name__ == "__main__": + main() diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/modules/eos_static_routes.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/modules/eos_static_routes.py new file mode 100644 index 00000000..516847a3 --- /dev/null +++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/modules/eos_static_routes.py @@ -0,0 +1,967 @@ +#!/usr/bin/python +# -*- coding: utf-8 -*- +# Copyright 2019 Red Hat +# GNU General Public License v3.0+ +# (see COPYING or https://www.gnu.org/licenses/gpl-3.0.txt) + +############################################# +# WARNING # +############################################# +# +# This file is auto generated by the resource +# module builder playbook. +# +# Do not edit this file manually. +# +# Changes to this file will be over written +# by the resource module builder. +# +# Changes should be made in the model used to +# generate this file or in the resource module +# builder template. +# +############################################# + +""" +The module file for eos_static_routes +""" + +from __future__ import absolute_import, division, print_function + +__metaclass__ = type + + +DOCUMENTATION = """ +module: eos_static_routes +short_description: Static routes resource module +description: This module configures and manages the attributes of static routes on + Arista EOS platforms. +version_added: 1.0.0 +author: Gomathi Selvi Srinivasan (@GomathiselviS) +notes: +- Tested against Arista EOS 4.20.10M +- This module works with connection C(network_cli). See the L(EOS Platform Options,../network/user_guide/platform_eos.html). +options: + config: + description: + - A list of configurations for static routes. + type: list + elements: dict + suboptions: + vrf: + description: + - The VRF to which the static route(s) belong. + type: str + address_families: + description: A dictionary specifying the address family to which the static + route(s) belong. + type: list + elements: dict + suboptions: + afi: + description: + - Specifies the top level address family indicator. + type: str + choices: + - ipv4 + - ipv6 + required: true + routes: + description: A dictionary that specifies the static route configurations. + elements: dict + type: list + suboptions: + dest: + description: + - Destination IPv4 subnet (CIDR or address-mask notation). + - The address format is <v4/v6 address>/<mask> or <v4/v6 address> + <mask>. + - The mask is number in range 0-32 for IPv4 and in range 0-128 for + IPv6. + type: str + required: true + next_hops: + description: + - Details of route to be taken. + type: list + elements: dict + suboptions: + forward_router_address: + description: + - Forwarding router's address on destination interface. + type: str + interface: + description: + - Outgoing interface to take. For anything except 'null0', then + next hop IP address should also be configured. + - IP address of the next hop router or + - null0 Null0 interface or + - ethernet e_num Ethernet interface or + - loopback l_num Loopback interface or + - management m_num Management interface or + - port-channel p_num + - vlan v_num + - vxlan vx_num + - Nexthop-Group Specify nexthop group name + - Tunnel Tunnel interface + - vtep Configure VXLAN Tunnel End Points + type: str + nexthop_grp: + description: + - Nexthop group + type: str + admin_distance: + description: + - Preference or administrative distance of route (range 1-255). + type: int + description: + description: + - Name of the static route. + type: str + tag: + description: + - Route tag value (ranges from 0 to 4294967295). + type: int + track: + description: + - Track value (range 1 - 512). Track must already be configured + on the device before adding the route. + type: str + mpls_label: + description: + - MPLS label + type: int + vrf: + description: + - VRF of the destination. + type: str + running_config: + description: + - This option is used only with state I(parsed). + - The value of this option should be the output received from the EOS device by + executing the command B(show running-config | grep routes). + - The state I(parsed) reads the configuration from C(running_config) option and + transforms it into Ansible structured data as per the resource module's argspec + and the value is then returned in the I(parsed) key within the result. + type: str + state: + description: + - The state the configuration should be left in. + type: str + choices: + - deleted + - merged + - overridden + - replaced + - gathered + - rendered + - parsed + default: merged + +""" +EXAMPLES = """ +# Using deleted + +# Before State: +# ------------ + +# veos(config)#show running-config | grep route +# ip route vrf testvrf 22.65.1.0/24 Null0 90 name testroute +# ipv6 route 5222:5::/64 Management1 4312:100::1 +# ipv6 route vrf testvrf 2222:6::/64 Management1 4312:100::1 +# ipv6 route vrf testvrf 2222:6::/64 Ethernet1 55 +# ipv6 route vrf testvrf 2222:6::/64 Null0 90 name testroute1 +# veos(config)# + +- name: Delete afi + arista.eos.eos_static_routes: + config: + - vrf: testvrf + address_families: + - afi: ipv4 + state: deleted + +# "after": [ +# { +# "address_families": [ +# { +# "afi": "ipv6", +# "routes": [ +# { +# "dest": "5222:5::/64", +# "next_hops": [ +# { +# "forward_router_address": "4312:100::1", +# "interface": "Management1" +# } +# ] +# } +# ] +# } +# ] +# }, +# { +# "address_families": [ +# { +# "afi": "ipv6", +# "routes": [ +# { +# "dest": "2222:6::/64", +# "next_hops": [ +# { +# "forward_router_address": "4312:100::1", +# "interface": "Management1" +# }, +# { +# "admin_distance": 55, +# "interface": "Ethernet1" +# }, +# { +# "admin_distance": 90, +# "description": "testroute1", +# "interface": "Null0" +# } +# ] +# } +# ] +# } +# ], +# "vrf": "testvrf" +# } +# ], +# "before": [ +# { +# "address_families": [ +# { +# "afi": "ipv6", +# "routes": [ +# { +# "dest": "5222:5::/64", +# "next_hops": [ +# { +# "forward_router_address": "4312:100::1", +# "interface": "Management1" +# } +# ] +# } +# ] +# } +# ] +# }, +# { +# "address_families": [ +# { +# "afi": "ipv4", +# "routes": [ +# { +# "dest": "22.65.1.0/24", +# "next_hops": [ +# { +# "admin_distance": 90, +# "description": "testroute", +# "interface": "Null0" +# } +# ] +# } +# ] +# }, +# { +# "afi": "ipv6", +# "routes": [ +# { +# "dest": "2222:6::/64", +# "next_hops": [ +# { +# "forward_router_address": "4312:100::1", +# "interface": "Management1" +# }, +# { +# "admin_distance": 55, +# "interface": "Ethernet1" +# }, +# { +# "admin_distance": 90, +# "description": "testroute1", +# "interface": "Null0" +# } +# ] +# } +# ] +# } +# ], +# "vrf": "testvrf" +# } +# ], +# "changed": true, +# "commands": [ +# "no ip route vrf testvrf 22.65.1.0/24 Null0 90 name testroute" +# ], + +# After State +# ___________ + +# veos(config)#show running-config | grep route +# ipv6 route 5222:5::/64 Management1 4312:100::1 +# ipv6 route vrf testvrf 2222:6::/64 Management1 4312:100::1 +# ipv6 route vrf testvrf 2222:6::/64 Ethernet1 55 +# ipv6 route vrf testvrf 2222:6::/64 Null0 90 name testroute1 + +# +# Using merged + +# Before : [ +# { +# "address_families": [ +# { +# "afi": "ipv4", +# "routes": [ +# { +# "dest": "165.10.1.0/24", +# "next_hops": [ +# { +# "admin_distance": 100, +# "interface": "Ethernet1" +# } +# ] +# }, +# { +# "dest": "172.17.252.0/24", +# "next_hops": [ +# { +# "nexthop_grp": "testgroup" +# } +# ] +# } +# ] +# }, +# { +# "afi": "ipv6", +# "routes": [ +# { +# "dest": "5001::/64", +# "next_hops": [ +# { +# "admin_distance": 50, +# "interface": "Ethernet1" +# } +# ] +# } +# ] +# } +# ] +# }, +# { +# "address_families": [ +# { +# "afi": "ipv4", +# "routes": [ +# { +# "dest": "130.1.122.0/24", +# "next_hops": [ +# { +# "interface": "Ethernet1", +# "tag": 50 +# } +# ] +# } +# ] +# } +# ], +# "vrf": "testvrf" +# } +# ] +# +# Before State +# ------------- +# veos(config)#show running-config | grep "route" +# ip route 165.10.1.0/24 Ethernet1 100 +# ip route 172.17.252.0/24 Nexthop-Group testgroup +# ip route vrf testvrf 130.1.122.0/24 Ethernet1 tag 50 +# ipv6 route 5001::/64 Ethernet1 50 +# veos(config)# + +- name: Merge new static route configuration + arista.eos.eos_static_routes: + config: + - vrf: testvrf + address_families: + - afi: ipv6 + routes: + - dest: 2211::0/64 + next_hop: + - forward_router_address: 100:1::2 + interface: Ethernet1 + state: merged + +# After State +# ----------- + +#After [ +# { +# "address_families": [ +# { +# "afi": "ipv4", +# "routes": [ +# { +# "dest": "165.10.1.0/24", +# "next_hops": [ +# { +# "admin_distance": 100, +# "interface": "Ethernet1" +# } +# ] +# }, +# { +# "dest": "172.17.252.0/24", +# "next_hops": [ +# { +# "nexthop_grp": "testgroup" +# } +# ] +# } +# ] +# }, +# { +# "afi": "ipv6", +# "routes": [ +# { +# "dest": "5001::/64", +# "next_hops": [ +# { +# "admin_distance": 50, +# "interface": "Ethernet1" +# } +# ] +# } +# ] +# } +# ] +# }, +# { +# "address_families": [ +# { +# "afi": "ipv4", +# "routes": [ +# { +# "dest": "130.1.122.0/24", +# "next_hops": [ +# { +# "interface": "Ethernet1", +# "tag": 50 +# } +# ] +# } +# ] +# }, +# { +# "afi": "ipv6", +# "routes": [ +# { +# "dest": "2211::0/64", +# "next_hops": [ +# { +# "aforward_router_address": 100:1::2 +# "interface": "Ethernet1" +# } +# ] +# } +# ] +# } + +# ], +# "vrf": "testvrf" +# } +# ] +# +# veos(config)#show running-config | grep "route" +# ip route 165.10.1.0/24 Ethernet1 100 +# ip route 172.17.252.0/24 Nexthop-Group testgroup +# ip route vrf testvrf 130.1.122.0/24 Ethernet1 tag 50 +# ipv6 route 2211::/64 Ethernet1 100:1::2 +# ipv6 route 5001::/64 Ethernet1 50 +# veos(config)# + + +# Using overridden + + +# Before State +# ------------- + +# "before": [ +# { +# "address_families": [ +# { +# "afi": "ipv4", +# "routes": [ +# { +# "dest": "165.10.1.0/24", +# "next_hops": [ +# { +# "admin_distance": 100, +# "interface": "Ethernet1" +# } +# ] +# }, +# { +# "dest": "172.17.252.0/24", +# "next_hops": [ +# { +# "nexthop_grp": "testgroup" +# } +# ] +# } +# ] +# }, +# { +# "afi": "ipv6", +# "routes": [ +# { +# "dest": "5001::/64", +# "next_hops": [ +# { +# "admin_distance": 50, +# "interface": "Ethernet1" +# } +# ] +# } +# ] +# } +# ] +# }, +# { +# "address_families": [ +# { +# "afi": "ipv4", +# "routes": [ +# { +# "dest": "130.1.122.0/24", +# "next_hops": [ +# { +# "interface": "Ethernet1", +# "tag": 50 +# } +# ] +# } +# ] +# } +# ], +# "vrf": "testvrf" +# } +# ] +# veos(config)#show running-config | grep "route" +# ip route 165.10.1.0/24 Ethernet1 100 +# ip route 172.17.252.0/24 Nexthop-Group testgroup +# ip route vrf testvrf 130.1.122.0/24 Ethernet1 tag 50 +# ipv6 route 5001::/64 Ethernet1 50 +# veos(config)# + +- name: Overridden static route configuration + arista.eos.eos_static_routes: + config: + - address_families: + - afi: ipv4 + routes: + - dest: 10.2.2.0/24 + next_hop: + - interface: Ethernet1 + state: replaced + +# After State +# ----------- + +# "after": [ +# { +# "address_families": [ +# { +# "afi": "ipv4", +# "routes": [ +# { +# "dest": "10.2.2.0/24", +# "next_hops": [ +# { +# "interface": "Ethernet1" +# } +# ] +# } +# ] +# } +# ] +# } +# ] +# veos(config)#show running-config | grep "route" +# ip route 10.2.2.0/24 Ethernet1 +# veos(config)# + + +# Using replaced + +# Before State +# ------------- + +# ip route 10.2.2.0/24 Ethernet1 +# ip route 10.2.2.0/24 64.1.1.1 label 17 33 +# ip route 33.33.33.0/24 Nexthop-Group testgrp +# ip route vrf testvrf 22.65.1.0/24 Null0 90 name testroute +# ipv6 route 5222:5::/64 Management1 4312:100::1 +# ipv6 route vrf testvrf 2222:6::/64 Management1 4312:100::1 +# ipv6 route vrf testvrf 2222:6::/64 Ethernet1 55 +# ipv6 route vrf testvrf 2222:6::/64 Null0 90 name testroute1 + +# [ +# { +# "address_families": [ +# { +# "afi": "ipv4", +# "routes": [ +# { +# "dest": "10.2.2.0/24", +# "next_hops": [ +# { +# "interface": "Ethernet1" +# }, +# { +# "admin_distance": 33, +# "interface": "64.1.1.1", +# "mpls_label": 17 +# } +# ] +# }, +# { +# "dest": "33.33.33.0/24", +# "next_hops": [ +# { +# "nexthop_grp": "testgrp" +# } +# ] +# } +# ] +# }, +# { +# "afi": "ipv6", +# "routes": [ +# { +# "dest": "5222:5::/64", +# "next_hops": [ +# { +# "forward_router_address": "4312:100::1", +# "interface": "Management1" +# } +# ] +# } +# ] +# } +# ] +# }, +# { +# "address_families": [ +# { +# "afi": "ipv4", +# "routes": [ +# { +# "dest": "22.65.1.0/24", +# "next_hops": [ +# { +# "admin_distance": 90, +# "description": "testroute", +# "interface": "Null0" +# } +# ] +# } +# ] +# }, +# { +# "afi": "ipv6", +# "routes": [ +# { +# "dest": "2222:6::/64", +# "next_hops": [ +# { +# "forward_router_address": "4312:100::1", +# "interface": "Management1" +# }, +# { +# "admin_distance": 90, +# "description": "testroute1", +# "interface": "Null0" +# } +# ] +# } +# ] +# } +# ], +# "vrf": "testvrf" +# } +# ] + +- name: Replace nexthop + arista.eos.eos_static_routes: + config: + - vrf: testvrf + address_families: + - afi: ipv6 + routes: + - dest: 2222:6::/64 + next_hops: + - admin_distance: 55 + interface: Ethernet1 + state: replaced + +# After State +# ----------- + +# veos(config)#show running-config | grep route +# ip route 10.2.2.0/24 Ethernet1 +# ip route 10.2.2.0/24 64.1.1.1 label 17 33 +# ip route 33.33.33.0/24 Nexthop-Group testgrp +# ip route vrf testvrf 22.65.1.0/24 Null0 90 name testroute +# ipv6 route 5222:5::/64 Management1 4312:100::1 +# ipv6 route vrf testvrf 2222:6::/64 Ethernet1 55 + +# "after": [ +# { +# "address_families": [ +# { +# "afi": "ipv4", +# "routes": [ +# { +# "dest": "10.2.2.0/24", +# "next_hops": [ +# { +# "interface": "Ethernet1" +# }, +# { +# "admin_distance": 33, +# "interface": "64.1.1.1", +# "mpls_label": 17 +# } +# ] +# }, +# { +# "dest": "33.33.33.0/24", +# "next_hops": [ +# { +# "nexthop_grp": "testgrp" +# } +# ] +# } +# ] +# }, +# { +# "afi": "ipv6", +# "routes": [ +# { +# "dest": "5222:5::/64", +# "next_hops": [ +# { +# "forward_router_address": "4312:100::1", +# "interface": "Management1" +# } +# ] +# } +# ] +# } +# ] +# }, +# { +# "address_families": [ +# { +# "afi": "ipv4", +# "routes": [ +# { +# "dest": "22.65.1.0/24", +# "next_hops": [ +# { +# "admin_distance": 90, +# "description": "testroute", +# "interface": "Null0" +# } +# ] +# } +# ] +# }, +# { +# "afi": "ipv6", +# "routes": [ +# { +# "dest": "2222:6::/64", +# "next_hops": [ +# { +# "admin_distance": 55, +# "interface": "Ethernet1" +# } +# ] +# } +# ] +# } +# ], +# "vrf": "testvrf" +# } +# ] + +# Before State +# ------------- +# veos(config)#show running-config | grep "route" +# ip route 165.10.1.0/24 Ethernet1 10.1.1.2 100 +# ipv6 route 5001::/64 Ethernet1 +# veos(config)# + + +- name: Gather the exisitng condiguration + arista.eos.eos_static_routes: + state: gathered + +# returns : +# arista.eos.eos_static_routes: +# config: +# - address_families: +# - afi: ipv4 +# routes: +# - dest: 165.10.1.0/24 +# next_hop: +# - forward_router_address: 10.1.1.2 +# interface: "Ethernet1" +# admin_distance: 100 +# - afi: ipv6 +# routes: +# - dest: 5001::/64 +# next_hop: +# - interface: "Ethernet1" + + +# Using rendered + +# arista.eos.eos_static_routes: +# config: +# - address_families: +# - afi: ipv4 +# routes: +# - dest: 165.10.1.0/24 +# next_hop: +# - forward_router_address: 10.1.1.2 +# interface: "Ethernet1" +# admin_distance: 100 +# - afi: ipv6 +# routes: +# - dest: 5001::/64 +# next_hop: +# - interface: "Ethernet1" + +# returns: + +# ip route 165.10.1.0/24 Ethernet1 10.1.1.2 100 +# ipv6 route 5001::/64 Ethernet1 + + +""" +RETURN = """ +before: + description: The configuration prior to the model invocation. + returned: always + type: list + sample: > + The configuration returned will always be in the same format + of the parameters above. +after: + description: The resulting configuration model invocation. + returned: when changed + type: list + sample: > + The configuration returned will always be in the same format + of the parameters above. +commands: + description: The set of commands pushed to the remote device. + returned: always + type: list + sample: + - ip route vrf vrf1 192.2.2.0/24 125.2.3.1 93 +rendered: + description: The set of CLI commands generated from the value in C(config) option + returned: When C(state) is I(rendered) + type: list + sample: > + "address_families": [ + { + "afi": "ipv4", + "routes": [ + { + "dest": "192.2.2.0/24", + "next_hops": [ + { + "admin_distance": 93, + "description": null, + "forward_router_address": null, + "interface": "125.2.3.1", + "mpls_label": null, + "nexthop_grp": null, + "tag": null, + "track": null, + "vrf": null + } + ] + } + ] + } + ], + "vrf": "vrf1" + } + ], + "running_config": null, + "state": "rendered" + } +gathered: + description: The configuration as structured data transformed for the running configuration + fetched from remote host + returned: When C(state) is I(gathered) + type: list + sample: > + The configuration returned will always be in the same format + of the parameters above. +parsed: + description: The configuration as structured data transformed for the value of + C(running_config) option + returned: When C(state) is I(parsed) + type: list + sample: > + The configuration returned will always be in the same format + of the parameters above. +""" + + +from ansible.module_utils.basic import AnsibleModule +from ansible_collections.arista.eos.plugins.module_utils.network.eos.argspec.static_routes.static_routes import ( + Static_routesArgs, +) +from ansible_collections.arista.eos.plugins.module_utils.network.eos.config.static_routes.static_routes import ( + Static_routes, +) + + +def main(): + """ + Main entry point for module execution + + :returns: the result form module invocation + """ + + required_if = [ + ("state", "merged", ("config",)), + ("state", "replaced", ("config",)), + ("state", "overridden", ("config",)), + ("state", "parsed", ("running_config",)), + ] + mutually_exclusive = [("config", "running_config")] + + module = AnsibleModule( + argument_spec=Static_routesArgs.argument_spec, + required_if=required_if, + supports_check_mode=True, + mutually_exclusive=mutually_exclusive, + ) + + result = Static_routes(module).execute_module() + module.exit_json(**result) + + +if __name__ == "__main__": + main() diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/modules/eos_system.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/modules/eos_system.py new file mode 100644 index 00000000..fd010037 --- /dev/null +++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/modules/eos_system.py @@ -0,0 +1,376 @@ +#!/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/>. +# +from __future__ import absolute_import, division, print_function + +__metaclass__ = type + + +DOCUMENTATION = """ +module: eos_system +author: Peter Sprygada (@privateip) +short_description: Manage the system attributes on Arista EOS devices +description: +- This module provides declarative management of node system attributes on Arista + EOS devices. It provides an option to configure host system parameters or remove + those parameters from the device active configuration. +version_added: 1.0.0 +extends_documentation_fragment: +- arista.eos.eos +notes: +- Tested against EOS 4.15 +options: + hostname: + description: + - Configure the device hostname parameter. This option takes an ASCII string value. + type: str + domain_name: + description: + - Configure the IP domain name on the remote device to the provided value. Value + should be in the dotted name form and will be appended to the C(hostname) to + create a fully-qualified domain name. + type: str + domain_list: + description: + - Provides the list of domain suffixes to append to the hostname for the purpose + of doing name resolution. This argument accepts a list of names and will be + reconciled with the current active configuration on the running node. + aliases: + - domain_search + type: list + elements: str + lookup_source: + description: + - Provides one or more source interfaces to use for performing DNS lookups. The + interface provided in C(lookup_source) can only exist in a single VRF. This + argument accepts either a list of interface names or a list of hashes that configure + the interface name and VRF name. See examples. + elements: raw + type: list + name_servers: + description: + - List of DNS name servers by IP address to use to perform name resolution lookups. This + argument accepts either a list of DNS servers or a list of hashes that configure + the name server and VRF name. See examples. + type: list + elements: str + state: + description: + - State of the configuration values in the device's current active configuration. When + set to I(present), the values should be configured in the device active configuration + and when set to I(absent) the values should not be in the device active configuration + default: present + type: str + choices: + - present + - absent +""" + +EXAMPLES = """ +- name: configure hostname and domain-name + arista.eos.eos_system: + hostname: eos01 + domain_name: test.example.com + +- name: remove configuration + arista.eos.eos_system: + state: absent + +- name: configure DNS lookup sources + arista.eos.eos_system: + lookup_source: Management1 + +- name: configure DNS lookup sources with VRF support + arista.eos.eos_system: + lookup_source: + - interface: Management1 + vrf: mgmt + - interface: Ethernet1 + vrf: myvrf + +- name: configure name servers + arista.eos.eos_system: + name_servers: + - 8.8.8.8 + - 8.8.4.4 + +- name: configure name servers with VRF support + arista.eos.eos_system: + name_servers: + - {server: 8.8.8.8, vrf: mgmt} + - {server: 8.8.4.4, vrf: mgmt} +""" + +RETURN = """ +commands: + description: The list of configuration mode commands to send to the device + returned: always + type: list + sample: + - hostname eos01 + - ip domain-name test.example.com +session_name: + description: The EOS config session name used to load the configuration + returned: changed + type: str + sample: ansible_1479315771 +""" +import re + +from ansible.module_utils.basic import AnsibleModule +from ansible_collections.ansible.netcommon.plugins.module_utils.network.common.utils import ( + ComplexList, +) +from ansible_collections.arista.eos.plugins.module_utils.network.eos.eos import ( + load_config, + get_config, +) +from ansible_collections.arista.eos.plugins.module_utils.network.eos.eos import ( + eos_argument_spec, +) + +_CONFIGURED_VRFS = None + + +def has_vrf(module, vrf): + global _CONFIGURED_VRFS + if _CONFIGURED_VRFS is not None: + return vrf in _CONFIGURED_VRFS + config = get_config(module) + _CONFIGURED_VRFS = re.findall(r"vrf definition (\S+)", config) + _CONFIGURED_VRFS.append("default") + return vrf in _CONFIGURED_VRFS + + +def map_obj_to_commands(want, have, module): + commands = list() + state = module.params["state"] + + def needs_update(x): + return want.get(x) and (want.get(x) != have.get(x)) + + if state == "absent": + if have["domain_name"]: + commands.append("no ip domain-name") + + if have["hostname"] != "localhost": + commands.append("no hostname") + + if state == "present": + if needs_update("hostname"): + commands.append("hostname %s" % want["hostname"]) + + if needs_update("domain_name"): + commands.append("ip domain-name %s" % want["domain_name"]) + + if want["domain_list"]: + # handle domain_list items to be removed + for item in set(have["domain_list"]).difference( + want["domain_list"] + ): + commands.append("no ip domain-list %s" % item) + + # handle domain_list items to be added + for item in set(want["domain_list"]).difference( + have["domain_list"] + ): + commands.append("ip domain-list %s" % item) + + if want["lookup_source"]: + # handle lookup_source items to be removed + for item in have["lookup_source"]: + if item not in want["lookup_source"]: + if item["vrf"]: + if not has_vrf(module, item["vrf"]): + module.fail_json( + msg="vrf %s is not configured" % item["vrf"] + ) + values = (item["vrf"], item["interface"]) + commands.append( + "no ip domain lookup vrf %s source-interface %s" + % values + ) + else: + commands.append( + "no ip domain lookup source-interface %s" + % item["interface"] + ) + + # handle lookup_source items to be added + for item in want["lookup_source"]: + if item not in have["lookup_source"]: + if item["vrf"]: + if not has_vrf(module, item["vrf"]): + module.fail_json( + msg="vrf %s is not configured" % item["vrf"] + ) + values = (item["vrf"], item["interface"]) + commands.append( + "ip domain lookup vrf %s source-interface %s" + % values + ) + else: + commands.append( + "ip domain lookup source-interface %s" + % item["interface"] + ) + + if want["name_servers"]: + # handle name_servers items to be removed. Order does matter here + # since name servers can only be in one vrf at a time + for item in have["name_servers"]: + if item not in want["name_servers"]: + if not has_vrf(module, item["vrf"]): + module.fail_json( + msg="vrf %s is not configured" % item["vrf"] + ) + if item["vrf"] not in ("default", None): + values = (item["vrf"], item["server"]) + commands.append("no ip name-server vrf %s %s" % values) + else: + commands.append( + "no ip name-server %s" % item["server"] + ) + + # handle name_servers items to be added + for item in want["name_servers"]: + if item not in have["name_servers"]: + if not has_vrf(module, item["vrf"]): + module.fail_json( + msg="vrf %s is not configured" % item["vrf"] + ) + if item["vrf"] not in ("default", None): + values = (item["vrf"], item["server"]) + commands.append("ip name-server vrf %s %s" % values) + else: + commands.append("ip name-server %s" % item["server"]) + + return commands + + +def parse_hostname(config): + match = re.search(r"^hostname (\S+)", config, re.M) + if match: + return match.group(1) + + +def parse_domain_name(config): + match = re.search(r"^ip domain-name (\S+)", config, re.M) + if match: + return match.group(1) + + +def parse_lookup_source(config): + objects = list() + regex = r"ip domain lookup (?:vrf (\S+) )*source-interface (\S+)" + for vrf, intf in re.findall(regex, config, re.M): + if len(vrf) == 0: + vrf = None + objects.append({"interface": intf, "vrf": vrf}) + return objects + + +def parse_name_servers(config): + objects = list() + for vrf, addr in re.findall( + r"ip name-server vrf (\S+) (\S+)", config, re.M + ): + objects.append({"server": addr, "vrf": vrf}) + return objects + + +def map_config_to_obj(module): + config = get_config(module) + return { + "hostname": parse_hostname(config), + "domain_name": parse_domain_name(config), + "domain_list": re.findall(r"^ip domain-list (\S+)", config, re.M), + "lookup_source": parse_lookup_source(config), + "name_servers": parse_name_servers(config), + } + + +def map_params_to_obj(module): + obj = { + "hostname": module.params["hostname"], + "domain_name": module.params["domain_name"], + "domain_list": module.params["domain_list"], + } + + lookup_source = ComplexList( + dict(interface=dict(key=True), vrf=dict()), module + ) + + name_servers = ComplexList( + dict(server=dict(key=True), vrf=dict(default="default")), module + ) + + for arg, cast in [ + ("lookup_source", lookup_source), + ("name_servers", name_servers), + ]: + if module.params[arg] is not None: + obj[arg] = cast(module.params[arg]) + else: + obj[arg] = None + + return obj + + +def main(): + """ main entry point for module execution + """ + argument_spec = dict( + hostname=dict(), + domain_name=dict(), + domain_list=dict( + type="list", aliases=["domain_search"], elements="str" + ), + # { interface: <str>, vrf: <str> } + lookup_source=dict(type="list", elements="raw"), + # { server: <str>; vrf: <str> } + name_servers=dict(type="list", elements="str"), + state=dict(default="present", choices=["present", "absent"]), + ) + + argument_spec.update(eos_argument_spec) + + module = AnsibleModule( + argument_spec=argument_spec, supports_check_mode=True + ) + + result = {"changed": False} + + want = map_params_to_obj(module) + have = map_config_to_obj(module) + + commands = map_obj_to_commands(want, have, module) + result["commands"] = commands + + if commands: + commit = not module.check_mode + response = load_config(module, commands, commit=commit) + if response.get("diff") and module._diff: + result["diff"] = {"prepared": response.get("diff")} + result["session_name"] = response.get("session") + result["changed"] = True + + module.exit_json(**result) + + +if __name__ == "__main__": + main() diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/modules/eos_user.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/modules/eos_user.py new file mode 100644 index 00000000..a14331ae --- /dev/null +++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/modules/eos_user.py @@ -0,0 +1,477 @@ +#!/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/>. +# +from __future__ import absolute_import, division, print_function + +__metaclass__ = type + + +DOCUMENTATION = """ +module: eos_user +author: Peter Sprygada (@privateip) +short_description: Manage the collection of local users on EOS devices +description: +- This module provides declarative management of the local usernames configured on + Arista EOS devices. It allows playbooks to manage either individual usernames or + the collection of usernames in the current running config. It also supports purging + usernames from the configuration that are not explicitly defined. +version_added: 1.0.0 +extends_documentation_fragment: +- arista.eos.eos +notes: +- Tested against EOS 4.15 +options: + aggregate: + description: + - The set of username objects to be configured on the remote Arista EOS device. The + list entries can either be the username or a hash of username and properties. This + argument is mutually exclusive with the C(username) argument. + aliases: + - users + - collection + type: list + elements: dict + suboptions: + name: + description: + - The username to be configured on the remote Arista EOS device. This argument + accepts a stringv value and is mutually exclusive with the C(aggregate) argument. + Please note that this option is not same as C(provider username). + type: str + configured_password: + description: + - The password to be configured on the remote Arista EOS device. The password + needs to be provided in clear and it will be encrypted on the device. Please + note that this option is not same as C(provider password). + type: str + update_password: + description: + - Since passwords are encrypted in the device running config, this argument will + instruct the module when to change the password. When set to C(always), the + password will always be updated in the device and when set to C(on_create) the + password will be updated only if the username is created. + type: str + choices: + - on_create + - always + privilege: + description: + - The C(privilege) argument configures the privilege level of the user when logged + into the system. This argument accepts integer values in the range of 1 to + 15. + type: int + role: + description: + - Configures the role for the username in the device running configuration. The + argument accepts a string value defining the role name. This argument does + not check if the role has been configured on the device. + type: str + sshkey: + description: + - Specifies the SSH public key to configure for the given username. This argument + accepts a valid SSH key value. + type: str + nopassword: + description: + - Defines the username without assigning a password. This will allow the user + to login to the system without being authenticated by a password. + type: bool + state: + description: + - Configures the state of the username definition as it relates to the device + operational configuration. When set to I(present), the username(s) should be + configured in the device active configuration and when set to I(absent) the + username(s) should not be in the device active configuration + type: str + choices: + - present + - absent + name: + description: + - The username to be configured on the remote Arista EOS device. This argument + accepts a stringv value and is mutually exclusive with the C(aggregate) argument. + Please note that this option is not same as C(provider username). + type: str + configured_password: + description: + - The password to be configured on the remote Arista EOS device. The password + needs to be provided in clear and it will be encrypted on the device. Please + note that this option is not same as C(provider password). + type: str + update_password: + description: + - Since passwords are encrypted in the device running config, this argument will + instruct the module when to change the password. When set to C(always), the + password will always be updated in the device and when set to C(on_create) the + password will be updated only if the username is created. + default: always + type: str + choices: + - on_create + - always + privilege: + description: + - The C(privilege) argument configures the privilege level of the user when logged + into the system. This argument accepts integer values in the range of 1 to + 15. + type: int + role: + description: + - Configures the role for the username in the device running configuration. The + argument accepts a string value defining the role name. This argument does + not check if the role has been configured on the device. + type: str + sshkey: + description: + - Specifies the SSH public key to configure for the given username. This argument + accepts a valid SSH key value. + type: str + nopassword: + description: + - Defines the username without assigning a password. This will allow the user + to login to the system without being authenticated by a password. + type: bool + purge: + description: + - Instructs the module to consider the resource definition absolute. It will + remove any previously configured usernames on the device with the exception + of the `admin` user which cannot be deleted per EOS constraints. + type: bool + default: false + state: + description: + - Configures the state of the username definition as it relates to the device + operational configuration. When set to I(present), the username(s) should be + configured in the device active configuration and when set to I(absent) the + username(s) should not be in the device active configuration + type: str + default: present + choices: + - present + - absent +""" + +EXAMPLES = """ +- name: create a new user + arista.eos.eos_user: + name: ansible + sshkey: "{{ lookup('file', '~/.ssh/id_rsa.pub') }}" + state: present + +- name: remove all users except admin + arista.eos.eos_user: + purge: yes + +- name: set multiple users to privilege level 15 + arista.eos.eos_user: + aggregate: + - name: netop + - name: netend + privilege: 15 + state: present + +- name: Change Password for User netop + arista.eos.eos_user: + username: netop + configured_password: '{{ new_password }}' + update_password: always + state: present +""" + +RETURN = """ +commands: + description: The list of configuration mode commands to send to the device + returned: always + type: list + sample: + - name ansible secret password + - name admin secret admin +session_name: + description: The EOS config session name used to load the configuration + returned: when changed is True + type: str + sample: ansible_1479315771 +""" + +import re + +from copy import deepcopy +from functools import partial + +from ansible.module_utils.basic import AnsibleModule +from ansible_collections.ansible.netcommon.plugins.module_utils.network.common.utils import ( + remove_default_spec, +) +from ansible_collections.arista.eos.plugins.module_utils.network.eos.eos import ( + get_config, + load_config, +) +from ansible_collections.arista.eos.plugins.module_utils.network.eos.eos import ( + eos_argument_spec, +) +from ansible.module_utils.six import iteritems + + +def validate_privilege(value, module): + if not 1 <= value <= 15: + module.fail_json( + msg="privilege must be between 1 and 15, got %s" % value + ) + + +def map_obj_to_commands(updates, module): + commands = list() + update_password = module.params["update_password"] + + for update in updates: + want, have = update + + def needs_update(x): + return want.get(x) and (want.get(x) != have.get(x)) + + def add(x): + return commands.append("username %s %s" % (want["name"], x)) + + if want["state"] == "absent": + commands.append("no username %s" % want["name"]) + continue + + if needs_update("configured_password"): + if update_password == "always" or not have: + add("secret %s" % want["configured_password"]) + + if needs_update("role"): + add("role %s" % want["role"]) + + if needs_update("privilege"): + add("privilege %s" % want["privilege"]) + + if needs_update("sshkey"): + add("sshkey %s" % want["sshkey"]) + + if needs_update("nopassword"): + if want["nopassword"]: + add("nopassword") + else: + add("no username %s nopassword" % want["name"]) + + if want.get("state") == "present" and want.get("name"): + value = [ + want.get("configured_password"), + want.get("nopassword"), + want.get("sshkey"), + ] + if all(v is None for v in value) is True: + module.fail_json( + msg="configured_password, sshkey or nopassword should be provided" + ) + + return commands + + +def parse_role(data): + match = re.search(r"role (\S+)", data, re.M) + if match: + return match.group(1) + + +def parse_sshkey(data): + match = re.search(r"sshkey (.+)$", data, re.M) + if match: + return match.group(1) + + +def parse_privilege(data): + match = re.search(r"privilege (\S+)", data, re.M) + if match: + return int(match.group(1)) + + +def map_config_to_obj(module): + data = get_config(module, flags=["section username"]) + + match = re.findall(r"^username (\S+)", data, re.M) + if not match: + return list() + + instances = list() + + for user in set(match): + regex = r"username %s .+$" % user + cfg = re.findall(regex, data, re.M) + cfg = "\n".join(cfg) + obj = { + "name": user, + "state": "present", + "nopassword": "nopassword" in cfg, + "configured_password": None, + "sshkey": parse_sshkey(cfg), + "privilege": parse_privilege(cfg), + "role": parse_role(cfg), + } + instances.append(obj) + + return instances + + +def get_param_value(key, item, module): + # if key doesn't exist in the item, get it from module.params + if not item.get(key): + value = module.params[key] + + # if key does exist, do a type check on it to validate it + else: + value_type = module.argument_spec[key].get("type", "str") + type_checker = module._CHECK_ARGUMENT_TYPES_DISPATCHER[value_type] + type_checker(item[key]) + value = item[key] + + # validate the param value (if validator func exists) + validator = globals().get("validate_%s" % key) + if all((value, validator)): + validator(value, module) + + return value + + +def map_params_to_obj(module): + aggregate = module.params["aggregate"] + if not aggregate: + if not module.params["name"] and module.params["purge"]: + return list() + elif not module.params["name"]: + module.fail_json(msg="name is required") + else: + collection = [{"name": module.params["name"]}] + else: + collection = list() + for item in aggregate: + if not isinstance(item, dict): + collection.append({"name": item}) + elif "name" not in item: + module.fail_json(msg="name is required") + else: + collection.append(item) + + objects = list() + + for item in collection: + get_value = partial(get_param_value, item=item, module=module) + item["configured_password"] = get_value("configured_password") + item["nopassword"] = get_value("nopassword") + item["privilege"] = get_value("privilege") + item["role"] = get_value("role") + item["sshkey"] = get_value("sshkey") + item["state"] = get_value("state") + objects.append(item) + + return objects + + +def update_objects(want, have): + updates = list() + for entry in want: + if "name" in entry: + item = next((i for i in have if i["name"] == entry["name"]), None) + if all((item is None, entry["state"] == "present")): + updates.append((entry, {})) + elif item: + for key, value in iteritems(entry): + if value and value != item[key]: + updates.append((entry, item)) + return updates + + +def main(): + """ main entry point for module execution + """ + element_spec = dict( + name=dict(), + configured_password=dict(no_log=True), + nopassword=dict(type="bool"), + update_password=dict( + default="always", choices=["on_create", "always"] + ), + privilege=dict(type="int"), + role=dict(), + sshkey=dict(), + state=dict(default="present", choices=["present", "absent"]), + ) + + aggregate_spec = deepcopy(element_spec) + + # remove default in aggregate spec, to handle common arguments + remove_default_spec(aggregate_spec) + + argument_spec = dict( + aggregate=dict( + type="list", + elements="dict", + options=aggregate_spec, + aliases=["collection", "users"], + ), + purge=dict(type="bool", default=False), + ) + + argument_spec.update(element_spec) + argument_spec.update(eos_argument_spec) + mutually_exclusive = [("name", "aggregate")] + + module = AnsibleModule( + argument_spec=argument_spec, + mutually_exclusive=mutually_exclusive, + supports_check_mode=True, + ) + + warnings = list() + + result = {"changed": False} + if warnings: + result["warnings"] = warnings + + want = map_params_to_obj(module) + have = map_config_to_obj(module) + + commands = map_obj_to_commands(update_objects(want, have), module) + + if module.params["purge"]: + want_users = [x["name"] for x in want] + have_users = [x["name"] for x in have] + for item in set(have_users).difference(want_users): + if item != "admin": + commands.append("no username %s" % item) + + result["commands"] = commands + + # the eos cli prevents this by rule so capture it and display + # a nice failure message + if "no username admin" in commands: + module.fail_json(msg="cannot delete the `admin` account") + + if commands: + commit = not module.check_mode + response = load_config(module, commands, commit=commit) + if response.get("diff") and module._diff: + result["diff"] = {"prepared": response.get("diff")} + result["session_name"] = response.get("session") + result["changed"] = True + + module.exit_json(**result) + + +if __name__ == "__main__": + main() diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/modules/eos_vlan.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/modules/eos_vlan.py new file mode 100644 index 00000000..ea7d28f8 --- /dev/null +++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/modules/eos_vlan.py @@ -0,0 +1,463 @@ +#!/usr/bin/python +# -*- coding: utf-8 -*- + +# (c) 2017, Ansible by Red Hat, inc +# +# This file is part of Ansible by Red Hat +# +# 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/>. +# + +from __future__ import absolute_import, division, print_function + +__metaclass__ = type + +DOCUMENTATION = """ +module: eos_vlan +author: Ricardo Carrillo Cruz (@rcarrillocruz) +short_description: (deprecated, removed after 2022-06-01) Manage VLANs on Arista EOS + network devices +description: +- This module provides declarative management of VLANs on Arista EOS network devices. +version_added: 1.0.0 +deprecated: + alternative: eos_vlans + why: Updated modules released with more functionality + removed_at_date: '2022-06-01' +notes: +- Tested against EOS 4.15 +options: + name: + description: + - Name of the VLAN. + type: str + vlan_id: + description: + - ID of the VLAN. + type: int + interfaces: + description: + - List of interfaces that should be associated to the VLAN. The name of interface + is case sensitive and should be in expanded format and not abbreviated. + type: list + elements: str + associated_interfaces: + description: + - This is a intent option and checks the operational state of the for given vlan + C(name) for associated interfaces. The name of interface is case sensitive and + should be in expanded format and not abbreviated. If the value in the C(associated_interfaces) + does not match with the operational state of vlan interfaces on device it will + result in failure. + type: list + elements: str + delay: + description: + - Delay the play should wait to check for declarative intent params values. + default: 10 + type: int + aggregate: + description: List of VLANs definitions. + type: list + elements: dict + suboptions: + name: + description: + - Name of the VLAN. + type: str + vlan_id: + description: + - ID of the VLAN. + required: true + type: int + interfaces: + description: + - List of interfaces that should be associated to the VLAN. The name of interface + is case sensitive and should be in expanded format and not abbreviated. + type: list + elements: str + associated_interfaces: + description: + - This is a intent option and checks the operational state of the for given vlan + C(name) for associated interfaces. The name of interface is case sensitive and + should be in expanded format and not abbreviated. If the value in the C(associated_interfaces) + does not match with the operational state of vlan interfaces on device it will + result in failure. + type: list + elements: str + delay: + description: + - Delay the play should wait to check for declarative intent params values. + default: 10 + type: int + state: + description: + - State of the VLAN configuration. + type: str + default: present + choices: + - present + - absent + - active + - suspend + purge: + description: + - Purge VLANs not defined in the I(aggregate) parameter. + default: false + type: bool + state: + description: + - State of the VLAN configuration. + type: str + default: present + choices: + - present + - absent + - active + - suspend +extends_documentation_fragment: +- arista.eos.eos + +""" + +EXAMPLES = """ +- name: Create vlan + arista.eos.eos_vlan: + vlan_id: 4000 + name: vlan-4000 + state: present + +- name: Add interfaces to vlan + arista.eos.eos_vlan: + vlan_id: 4000 + state: present + interfaces: + - Ethernet1 + - Ethernet2 + +- name: Check if interfaces is assigned to vlan + arista.eos.eos_vlan: + vlan_id: 4000 + associated_interfaces: + - Ethernet1 + - Ethernet2 + +- name: Suspend vlan + arista.eos.eos_vlan: + vlan_id: 4000 + state: suspend + +- name: Unsuspend vlan + arista.eos.eos_vlan: + vlan_id: 4000 + state: active + +- name: Create aggregate of vlans + arista.eos.eos_vlan: + aggregate: + - vlan_id: 4000 + - {vlan_id: 4001, name: vlan-4001} +""" + +RETURN = """ +commands: + description: The list of configuration mode commands to send to the device + returned: always + type: list + sample: + - vlan 20 + - name test-vlan +""" +import time + +from copy import deepcopy + +from ansible.module_utils.basic import AnsibleModule +from ansible_collections.ansible.netcommon.plugins.module_utils.network.common.utils import ( + remove_default_spec, +) +from ansible_collections.arista.eos.plugins.module_utils.network.eos.eos import ( + load_config, + run_commands, +) +from ansible_collections.arista.eos.plugins.module_utils.network.eos.eos import ( + eos_argument_spec, +) + + +def search_obj_in_list(vlan_id, lst): + for o in lst: + if o["vlan_id"] == vlan_id: + return o + + +def map_obj_to_commands(updates, module): + commands = list() + want, have = updates + purge = module.params["purge"] + + for w in want: + vlan_id = w["vlan_id"] + state = w["state"] + + obj_in_have = search_obj_in_list(vlan_id, have) + + if state == "absent": + if obj_in_have: + commands.append("no vlan %s" % w["vlan_id"]) + elif state == "present": + if not obj_in_have: + commands.append("vlan %s" % w["vlan_id"]) + if w["name"]: + commands.append("name %s" % w["name"]) + + if w["interfaces"]: + for i in w["interfaces"]: + commands.append("interface %s" % i) + commands.append( + "switchport access vlan %s" % w["vlan_id"] + ) + else: + if w["name"] and w["name"] != obj_in_have["name"]: + commands.append("vlan %s" % w["vlan_id"]) + commands.append("name %s" % w["name"]) + + if w["interfaces"]: + if not obj_in_have["interfaces"]: + for i in w["interfaces"]: + commands.append("vlan %s" % w["vlan_id"]) + commands.append("interface %s" % i) + commands.append( + "switchport access vlan %s" % w["vlan_id"] + ) + elif set(w["interfaces"]) != obj_in_have["interfaces"]: + missing_interfaces = list( + set(w["interfaces"]) + - set(obj_in_have["interfaces"]) + ) + for i in missing_interfaces: + commands.append("vlan %s" % w["vlan_id"]) + commands.append("interface %s" % i) + commands.append( + "switchport access vlan %s" % w["vlan_id"] + ) + + superfluous_interfaces = list( + set(obj_in_have["interfaces"]) + - set(w["interfaces"]) + ) + for i in superfluous_interfaces: + commands.append("vlan %s" % w["vlan_id"]) + commands.append("interface %s" % i) + commands.append( + "no switchport access vlan %s" % w["vlan_id"] + ) + else: + if not obj_in_have: + commands.append("vlan %s" % w["vlan_id"]) + if w["name"]: + commands.append("name %s" % w["name"]) + commands.append("state %s" % w["state"]) + elif ( + w["name"] and obj_in_have["name"] != w["name"] + ) or obj_in_have["state"] != w["state"]: + commands.append("vlan %s" % w["vlan_id"]) + + if w["name"]: + if obj_in_have["name"] != w["name"]: + commands.append("name %s" % w["name"]) + + if obj_in_have["state"] != w["state"]: + commands.append("state %s" % w["state"]) + + if purge: + for h in have: + obj_in_want = search_obj_in_list(h["vlan_id"], want) + if not obj_in_want and h["vlan_id"] != "1": + commands.append("no vlan %s" % h["vlan_id"]) + + return commands + + +def map_config_to_obj(module): + objs = [] + vlans = run_commands(module, ["show vlan configured-ports | json"]) + + for vlan in vlans[0]["vlans"]: + obj = {} + obj["vlan_id"] = vlan + obj["name"] = vlans[0]["vlans"][vlan]["name"] + obj["state"] = vlans[0]["vlans"][vlan]["status"] + obj["interfaces"] = [] + + interfaces = vlans[0]["vlans"][vlan] + + for interface in interfaces["interfaces"]: + obj["interfaces"].append(interface) + + if obj["state"] == "suspended": + obj["state"] = "suspend" + + objs.append(obj) + + return objs + + +def map_params_to_obj(module): + obj = [] + aggregate = module.params.get("aggregate") + if aggregate: + for item in aggregate: + for key in item: + if item.get(key) is None: + item[key] = module.params[key] + + if item.get("interfaces"): + item["interfaces"] = [ + intf.replace(" ", "") + for intf in item.get("interfaces") + if intf + ] + + if item.get("associated_interfaces"): + item["associated_interfaces"] = [ + intf.replace(" ", "") + for intf in item.get("associated_interfaces") + if intf + ] + + d = item.copy() + d["vlan_id"] = str(d["vlan_id"]) + + obj.append(d) + else: + obj.append( + { + "vlan_id": str(module.params["vlan_id"]), + "name": module.params["name"], + "state": module.params["state"], + "interfaces": [ + intf.replace(" ", "") + for intf in module.params["interfaces"] + ] + if module.params["interfaces"] + else [], + "associated_interfaces": [ + intf.replace(" ", "") + for intf in module.params["associated_interfaces"] + ] + if module.params["associated_interfaces"] + else [], + } + ) + + return obj + + +def check_declarative_intent_params(want, module, result): + have = None + is_delay = False + + for w in want: + if w.get("associated_interfaces") is None: + continue + + if result["changed"] and not is_delay: + time.sleep(module.params["delay"]) + is_delay = True + + if have is None: + have = map_config_to_obj(module) + + for i in w["associated_interfaces"]: + obj_in_have = search_obj_in_list(w["vlan_id"], have) + + if ( + obj_in_have + and "interfaces" in obj_in_have + and i not in obj_in_have["interfaces"] + ): + module.fail_json( + msg="Interface %s not configured on vlan %s" + % (i, w["vlan_id"]) + ) + + +def main(): + """ main entry point for module execution + """ + element_spec = dict( + vlan_id=dict(type="int"), + name=dict(), + interfaces=dict(type="list", elements="str"), + associated_interfaces=dict(type="list", elements="str"), + delay=dict(default=10, type="int"), + state=dict( + default="present", + choices=["present", "absent", "active", "suspend"], + ), + ) + + aggregate_spec = deepcopy(element_spec) + aggregate_spec["vlan_id"] = dict(required=True, type="int") + + # remove default in aggregate spec, to handle common arguments + remove_default_spec(aggregate_spec) + + aggregate_spec["state"].update(default="present") + aggregate_spec["delay"].update(default=10) + argument_spec = dict( + aggregate=dict(type="list", elements="dict", options=aggregate_spec), + purge=dict(default=False, type="bool"), + ) + + argument_spec.update(element_spec) + argument_spec.update(eos_argument_spec) + + required_one_of = [["vlan_id", "aggregate"]] + mutually_exclusive = [["vlan_id", "aggregate"]] + module = AnsibleModule( + argument_spec=argument_spec, + supports_check_mode=True, + required_one_of=required_one_of, + mutually_exclusive=mutually_exclusive, + ) + + warnings = list() + + result = {"changed": False} + + if warnings: + result["warnings"] = warnings + + want = map_params_to_obj(module) + have = map_config_to_obj(module) + + commands = map_obj_to_commands((want, have), module) + result["commands"] = commands + + if commands: + commit = not module.check_mode + response = load_config(module, commands, commit=commit) + if response.get("diff") and module._diff: + result["diff"] = {"prepared": response.get("diff")} + result["session_name"] = response.get("session") + result["changed"] = True + + check_declarative_intent_params(want, module, result) + + module.exit_json(**result) + + +if __name__ == "__main__": + main() diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/modules/eos_vlans.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/modules/eos_vlans.py new file mode 100644 index 00000000..8ee01fab --- /dev/null +++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/modules/eos_vlans.py @@ -0,0 +1,327 @@ +#!/usr/bin/python +# -*- coding: utf-8 -*- +# Copyright 2019 Red Hat +# GNU General Public License v3.0+ +# (see COPYING or https://www.gnu.org/licenses/gpl-3.0.txt) + +############################################# +# WARNING # +############################################# +# +# This file is auto generated by the resource +# module builder playbook. +# +# Do not edit this file manually. +# +# Changes to this file will be over written +# by the resource module builder. +# +# Changes should be made in the model used to +# generate this file or in the resource module +# builder template. +# +############################################# + +""" +The module file for eos_vlans +""" + +from __future__ import absolute_import, division, print_function + +__metaclass__ = type + + +DOCUMENTATION = """ +module: eos_vlans +short_description: VLANs resource module +description: This module provides declarative management of VLANs on Arista EOS network + devices. +version_added: 1.0.0 +author: Nathaniel Case (@qalthos) +notes: +- Tested against Arista EOS 4.20.10M +- This module works with connection C(network_cli). See the L(EOS Platform Options,../network/user_guide/platform_eos.html). +options: + config: + description: A dictionary of VLANs options + type: list + elements: dict + suboptions: + name: + description: + - Name of the VLAN. + type: str + vlan_id: + description: + - ID of the VLAN. Range 1-4094 + type: int + required: true + state: + description: + - Operational state of the VLAN + type: str + choices: + - active + - suspend + running_config: + description: + - This option is used only with state I(parsed). + - The value of this option should be the output received from the EOS device + by executing the command B(show running-config | section vlan). + - The state I(parsed) reads the configuration from C(running_config) option + and transforms it into Ansible structured data as per the resource module's + argspec and the value + type: str + state: + description: + - The state of the configuration after module completion + type: str + choices: + - merged + - replaced + - overridden + - deleted + - rendered + - gathered + - parsed + default: merged + +""" +EXAMPLES = """ +# Using deleted + +# Before state: +# ------------- +# +# veos(config-vlan-20)#show running-config | section vlan +# vlan 10 +# name ten +# ! +# vlan 20 +# name twenty + +- name: Delete attributes of the given VLANs. + arista.eos.eos_vlans: + config: + - vlan_id: 20 + state: deleted + +# After state: +# ------------ +# +# veos(config-vlan-20)#show running-config | section vlan +# vlan 10 +# name ten + + +# Using merged + +# Before state: +# ------------- +# +# veos(config-vlan-20)#show running-config | section vlan +# vlan 10 +# name ten +# ! +# vlan 20 +# name twenty + +- name: Merge given VLAN attributes with device configuration + arista.eos.eos_vlans: + config: + - vlan_id: 20 + state: suspend + state: merged + +# After state: +# ------------ +# +# veos(config-vlan-20)#show running-config | section vlan +# vlan 10 +# name ten +# ! +# vlan 20 +# name twenty +# state suspend + + +# Using overridden + +# Before state: +# ------------- +# +# veos(config-vlan-20)#show running-config | section vlan +# vlan 10 +# name ten +# ! +# vlan 20 +# name twenty + +- name: Override device configuration of all VLANs with provided configuration + arista.eos.eos_vlans: + config: + - vlan_id: 20 + state: suspend + state: overridden + +# After state: +# ------------ +# +# veos(config-vlan-20)#show running-config | section vlan +# vlan 20 +# state suspend + + +# Using replaced + +# Before state: +# ------------- +# +# veos(config-vlan-20)#show running-config | section vlan +# vlan 10 +# name ten +# ! +# vlan 20 +# name twenty + +- name: Replace all attributes of specified VLANs with provided configuration + arista.eos.eos_vlans: + config: + - vlan_id: 20 + state: suspend + state: replaced + +# After state: +# ------------ +# +# veos(config-vlan-20)#show running-config | section vlan +# vlan 10 +# name ten +# ! +# vlan 20 +# state suspend + +# using parsed + +# parsed.cfg +# vlan 10 +# name ten +# ! +# vlan 20 +# name twenty +# state suspend + +- name: Use parsed to convert native configs to structured data + arista.eos.eos_vlans: + running_config: "{{ lookup('file', 'parsed.cfg') }}" + state: parsed + +# Output: +# ------- +# parsed: +# - vlan_id: 10 +# name: ten +# - vlan_id: 20 +# state: suspend + +# Using rendered: + +- name: Use Rendered to convert the structured data to native config + arista.eos.eos_vlans: + config: + - vlan_id: 10 + name: ten + - vlan_id: 20 + state: suspend + state: rendered + +# Output: +# ------ +# rendered: +# - "vlan 10" +# - "name ten" +# - "vlan 20" +# - "state suspend" + +# Using gathered: +# native_config: +# vlan 10 +# name ten +# ! +# vlan 20 +# name twenty +# state suspend + +- name: Gather vlans facts from the device + arista.eos.eos_vlans: + state: gathered + +# Output: +# ------ + +# gathered: +# - vlan_id: 10 +# name: ten +# - vlan_id: 20 +# state: suspend + +""" +RETURN = """ +before: + description: The configuration as structured data prior to module invocation. + returned: always + type: list + sample: > + The configuration returned will always be in the same format + of the parameters above. +after: + description: The configuration as structured data after module completion. + returned: when changed + type: list + sample: > + The configuration returned will always be in the same format + of the parameters above. +commands: + description: The set of commands pushed to the remote device. + returned: always + type: list + sample: ['vlan 10', 'no name', 'vlan 11', 'name Eleven'] +""" + + +from ansible.module_utils.basic import AnsibleModule +from ansible_collections.arista.eos.plugins.module_utils.network.eos.argspec.vlans.vlans import ( + VlansArgs, +) +from ansible_collections.arista.eos.plugins.module_utils.network.eos.config.vlans.vlans import ( + Vlans, +) + + +def main(): + """ + Main entry point for module execution + + :returns: the result form module invocation + """ + required_if = [ + ("state", "merged", ("config",)), + ("state", "replaced", ("config",)), + ("state", "overridden", ("config",)), + ("state", "rendered", ("config",)), + ("state", "parsed", ("running_config",)), + ] + mutually_exclusive = [("config", "running_config")] + module = AnsibleModule( + argument_spec=VlansArgs.argument_spec, + required_if=required_if, + supports_check_mode=True, + mutually_exclusive=mutually_exclusive, + ) + + result = Vlans(module).execute_module() + module.exit_json(**result) + + +if __name__ == "__main__": + main() diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/modules/eos_vrf.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/modules/eos_vrf.py new file mode 100644 index 00000000..23356bf7 --- /dev/null +++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/modules/eos_vrf.py @@ -0,0 +1,432 @@ +#!/usr/bin/python +# -*- coding: utf-8 -*- +from __future__ import absolute_import, division, print_function + +__metaclass__ = type + +# (c) 2017, Ansible by Red Hat, inc +# +# This file is part of Ansible by Red Hat +# +# 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/>. +# + + +DOCUMENTATION = """ +module: eos_vrf +author: Ricardo Carrillo Cruz (@rcarrillocruz) +short_description: Manage VRFs on Arista EOS network devices +description: +- This module provides declarative management of VRFs on Arista EOS network devices. +version_added: 1.0.0 +notes: +- Tested against EOS 4.15 +options: + name: + description: + - Name of the VRF. + type: str + rd: + description: + - Route distinguisher of the VRF + type: str + interfaces: + description: + - Identifies the set of interfaces that should be configured in the VRF. Interfaces + must be routed interfaces in order to be placed into a VRF. The name of interface + should be in expanded format and not abbreviated. + type: list + elements: str + associated_interfaces: + description: + - This is a intent option and checks the operational state of the for given vrf + C(name) for associated interfaces. If the value in the C(associated_interfaces) + does not match with the operational state of vrf interfaces on device it will + result in failure. + type: list + elements: str + aggregate: + description: List of VRFs definitions + type: list + elements: dict + suboptions: + name: + description: + - Name of the VRF. + required: true + type: str + rd: + description: + - Route distinguisher of the VRF + type: str + interfaces: + description: + - Identifies the set of interfaces that should be configured in the VRF. Interfaces + must be routed interfaces in order to be placed into a VRF. The name of interface + should be in expanded format and not abbreviated. + type: list + elements: str + associated_interfaces: + description: + - This is a intent option and checks the operational state of the for given vrf + C(name) for associated interfaces. If the value in the C(associated_interfaces) + does not match with the operational state of vrf interfaces on device it will + result in failure. + type: list + elements: str + delay: + description: + - Time in seconds to wait before checking for the operational state on remote + device. This wait is applicable for operational state arguments. + default: 10 + type: int + state: + description: + - State of the VRF configuration. + default: present + type: str + choices: + - present + - absent + purge: + description: + - Purge VRFs not defined in the I(aggregate) parameter. + default: false + type: bool + delay: + description: + - Time in seconds to wait before checking for the operational state on remote + device. This wait is applicable for operational state arguments. + default: 10 + type: int + state: + description: + - State of the VRF configuration. + default: present + type: str + choices: + - present + - absent +extends_documentation_fragment: +- arista.eos.eos +""" + +EXAMPLES = """ +- name: Create vrf + arista.eos.eos_vrf: + name: test + rd: 1:200 + interfaces: + - Ethernet2 + state: present + +- name: Delete VRFs + arista.eos.eos_vrf: + name: test + state: absent + +- name: Create aggregate of VRFs with purge + arista.eos.eos_vrf: + aggregate: + - name: test4 + rd: 1:204 + - name: test5 + rd: 1:205 + state: present + purge: yes + +- name: Delete aggregate of VRFs + arista.eos.eos_vrf: + aggregate: + - name: test2 + - name: test3 + - name: test4 + - name: test5 + state: absent +""" + +RETURN = """ +commands: + description: The list of configuration mode commands to send to the device + returned: always + type: list + sample: + - vrf definition test + - rd 1:100 + - interface Ethernet1 + - vrf forwarding test +""" +import re +import time + +from copy import deepcopy + +from ansible.module_utils.basic import AnsibleModule +from ansible_collections.ansible.netcommon.plugins.module_utils.network.common.utils import ( + remove_default_spec, +) +from ansible_collections.arista.eos.plugins.module_utils.network.eos.eos import ( + load_config, + run_commands, +) +from ansible_collections.arista.eos.plugins.module_utils.network.eos.eos import ( + eos_argument_spec, +) + + +def search_obj_in_list(name, lst): + for o in lst: + if o["name"] == name: + return o + + +def map_obj_to_commands(updates, module): + commands = list() + want, have = updates + state = module.params["state"] + purge = module.params["purge"] + + for w in want: + name = w["name"] + rd = w["rd"] + + obj_in_have = search_obj_in_list(name, have) + + if state == "absent": + if obj_in_have: + commands.append("no vrf definition %s" % name) + elif state == "present": + if not obj_in_have: + commands.append("vrf definition %s" % name) + + if rd is not None: + commands.append("rd %s" % rd) + + if w["interfaces"]: + for i in w["interfaces"]: + commands.append("interface %s" % i) + commands.append("vrf forwarding %s" % w["name"]) + else: + if w["rd"] is not None and w["rd"] != obj_in_have["rd"]: + commands.append("vrf definition %s" % w["name"]) + commands.append("rd %s" % w["rd"]) + + if w["interfaces"]: + if not obj_in_have["interfaces"]: + for i in w["interfaces"]: + commands.append("interface %s" % i) + commands.append("vrf forwarding %s" % w["name"]) + elif set(w["interfaces"]) != obj_in_have["interfaces"]: + missing_interfaces = list( + set(w["interfaces"]) + - set(obj_in_have["interfaces"]) + ) + + for i in missing_interfaces: + commands.append("interface %s" % i) + commands.append("vrf forwarding %s" % w["name"]) + + if purge: + for h in have: + obj_in_want = search_obj_in_list(h["name"], want) + if not obj_in_want: + commands.append("no vrf definition %s" % h["name"]) + + return commands + + +def map_config_to_obj(module): + objs = [] + output = run_commands(module, {"command": "show vrf", "output": "text"}) + + lines = output[0].strip().splitlines()[3:] + + out_len = len(lines) + index = 0 + while out_len > index: + line = lines[index] + if not line: + continue + + splitted_line = re.split(r"\s{2,}", line.strip()) + + if len(splitted_line) == 1: + index += 1 + continue + obj = dict() + obj["name"] = splitted_line[0] + obj["rd"] = splitted_line[1] + obj["interfaces"] = [] + + if len(splitted_line) > 4: + obj["interfaces"] = [] + interfaces = splitted_line[4] + if interfaces.endswith(","): + while interfaces.endswith(","): + # gather all comma separated interfaces + if out_len <= index: + break + index += 1 + line = lines[index] + vrf_line = re.split(r"\s{2,}", line.strip()) + interfaces += vrf_line[-1] + + for i in interfaces.split(","): + obj["interfaces"].append(i.strip().lower()) + index += 1 + objs.append(obj) + + return objs + + +def map_params_to_obj(module): + obj = [] + aggregate = module.params.get("aggregate") + if aggregate: + for item in aggregate: + for key in item: + if item.get(key) is None: + item[key] = module.params[key] + + if item.get("interfaces"): + item["interfaces"] = [ + intf.replace(" ", "").lower() + for intf in item.get("interfaces") + if intf + ] + + if item.get("associated_interfaces"): + item["associated_interfaces"] = [ + intf.replace(" ", "").lower() + for intf in item.get("associated_interfaces") + if intf + ] + + obj.append(item.copy()) + else: + obj.append( + { + "name": module.params["name"], + "state": module.params["state"], + "rd": module.params["rd"], + "interfaces": [ + intf.replace(" ", "").lower() + for intf in module.params["interfaces"] + ] + if module.params["interfaces"] + else [], + "associated_interfaces": [ + intf.replace(" ", "").lower() + for intf in module.params["associated_interfaces"] + ] + if module.params["associated_interfaces"] + else [], + } + ) + + return obj + + +def check_declarative_intent_params(want, module, result): + have = None + is_delay = False + + for w in want: + if w.get("associated_interfaces") is None: + continue + + if result["changed"] and not is_delay: + time.sleep(module.params["delay"]) + is_delay = True + + if have is None: + have = map_config_to_obj(module) + + for i in w["associated_interfaces"]: + obj_in_have = search_obj_in_list(w["name"], have) + + if obj_in_have: + interfaces = obj_in_have.get("interfaces") + if interfaces is not None and i not in interfaces: + module.fail_json( + msg="Interface %s not configured on vrf %s" + % (i, w["name"]) + ) + + +def main(): + """ main entry point for module execution + """ + element_spec = dict( + name=dict(), + interfaces=dict(type="list", elements="str"), + associated_interfaces=dict(type="list", elements="str"), + delay=dict(default=10, type="int"), + rd=dict(), + state=dict(default="present", choices=["present", "absent"]), + ) + + aggregate_spec = deepcopy(element_spec) + aggregate_spec["name"] = dict(required=True) + + # remove default in aggregate spec, to handle common arguments + remove_default_spec(aggregate_spec) + aggregate_spec["state"].update(default="present") + aggregate_spec["delay"].update(default=10) + + argument_spec = dict( + aggregate=dict(type="list", elements="dict", options=aggregate_spec), + purge=dict(default=False, type="bool"), + ) + + argument_spec.update(element_spec) + argument_spec.update(eos_argument_spec) + + required_one_of = [["name", "aggregate"]] + mutually_exclusive = [["name", "aggregate"]] + module = AnsibleModule( + argument_spec=argument_spec, + required_one_of=required_one_of, + mutually_exclusive=mutually_exclusive, + supports_check_mode=True, + ) + + warnings = list() + + result = {"changed": False} + + if warnings: + result["warnings"] = warnings + + want = map_params_to_obj(module) + have = map_config_to_obj(module) + + commands = map_obj_to_commands((want, have), module) + result["commands"] = commands + + if commands: + commit = not module.check_mode + response = load_config(module, commands, commit=commit) + if response.get("diff") and module._diff: + result["diff"] = {"prepared": response.get("diff")} + result["session_name"] = response.get("session") + result["changed"] = True + + check_declarative_intent_params(want, module, result) + + module.exit_json(**result) + + +if __name__ == "__main__": + main() diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/terminal/__init__.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/terminal/__init__.py new file mode 100644 index 00000000..e69de29b --- /dev/null +++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/terminal/__init__.py diff --git a/collections-debian-merged/ansible_collections/arista/eos/plugins/terminal/eos.py b/collections-debian-merged/ansible_collections/arista/eos/plugins/terminal/eos.py new file mode 100644 index 00000000..ee575bc0 --- /dev/null +++ b/collections-debian-merged/ansible_collections/arista/eos/plugins/terminal/eos.py @@ -0,0 +1,108 @@ +# +# (c) 2016 Red Hat Inc. +# +# 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/>. +# + +from __future__ import absolute_import, division, print_function + +__metaclass__ = type + +import re +import json + +from ansible.plugins.terminal import TerminalBase +from ansible.errors import AnsibleConnectionFailure +from ansible.module_utils._text import to_bytes, to_text + + +class TerminalModule(TerminalBase): + + terminal_stdout_re = [ + re.compile(br"[\r\n]?[\w+\-\.:\/\[\]]+(?:\([^\)]+\)){,3}(?:>|#) ?$"), + re.compile(br"\[\w+\@[\w\-\.]+(?: [^\]])\] ?[>#\$] ?$"), + ] + + terminal_stderr_re = [ + re.compile(br"% ?Error"), + # re.compile(br"^% \w+", re.M), + re.compile(br"% User not present"), + re.compile(br"% ?Bad secret"), + re.compile(br"invalid input", re.I), + re.compile(br"(?:incomplete|ambiguous) command", re.I), + re.compile(br"connection timed out", re.I), + # Strings like this regarding VLANs are not errors + re.compile(br"[^\r\n]+ not found(?! in current VLAN)", re.I), + re.compile(br"'[^']' +returned error code: ?\d+"), + re.compile(br"[^\r\n](?<! shell )\/bin\/(?:ba)?sh"), + re.compile(br"% More than \d+ OSPF instance", re.I), + re.compile(br"% Subnet [0-9a-f.:/]+ overlaps", re.I), + re.compile(br"Maximum number of pending sessions has been reached"), + re.compile(br"% Prefix length must be less than"), + # returned in response to 'channel-group <name> mode <mode>' + re.compile( + br"% Cannot change mode; remove all members and try again." + ), + ] + + def on_open_shell(self): + try: + for cmd in (b"terminal length 0", b"terminal width 512"): + self._exec_cli_command(cmd) + except AnsibleConnectionFailure: + raise AnsibleConnectionFailure("unable to set terminal parameters") + + def on_become(self, passwd=None): + if self._get_prompt().endswith(b"#"): + return + + cmd = {u"command": u"enable"} + if passwd: + cmd[u"prompt"] = to_text( + r"[\r\n]?password: $", errors="surrogate_or_strict" + ) + cmd[u"answer"] = passwd + cmd[u"prompt_retry_check"] = True + + try: + self._exec_cli_command( + to_bytes(json.dumps(cmd), errors="surrogate_or_strict") + ) + prompt = self._get_prompt() + if prompt is None or not prompt.endswith(b"#"): + raise AnsibleConnectionFailure( + "failed to elevate privilege to enable mode still at prompt [%s]" + % prompt + ) + except AnsibleConnectionFailure as e: + prompt = self._get_prompt() + raise AnsibleConnectionFailure( + "unable to elevate privilege to enable mode, at prompt [%s] with error: %s" + % (prompt, e.message) + ) + + def on_unbecome(self): + prompt = self._get_prompt() + if prompt is None: + # if prompt is None most likely the terminal is hung up at a prompt + return + + if b"(config" in prompt: + self._exec_cli_command(b"end") + self._exec_cli_command(b"disable") + + elif prompt.endswith(b"#"): + self._exec_cli_command(b"disable") |