diff options
Diffstat (limited to 'collections-debian-merged/ansible_collections/dellemc/os10/plugins/action')
-rw-r--r-- | collections-debian-merged/ansible_collections/dellemc/os10/plugins/action/os10.py | 94 | ||||
-rw-r--r-- | collections-debian-merged/ansible_collections/dellemc/os10/plugins/action/textfsm_parser.py | 81 |
2 files changed, 175 insertions, 0 deletions
diff --git a/collections-debian-merged/ansible_collections/dellemc/os10/plugins/action/os10.py b/collections-debian-merged/ansible_collections/dellemc/os10/plugins/action/os10.py new file mode 100644 index 00000000..5669001c --- /dev/null +++ b/collections-debian-merged/ansible_collections/dellemc/os10/plugins/action/os10.py @@ -0,0 +1,94 @@ +# +# (c) 2020 Red Hat Inc. +# +# (c) 2020 Dell 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.module_utils._text import to_text +from ansible.module_utils.connection import Connection +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_collections.dellemc.os10.plugins.module_utils.network.os10 import os10_provider_spec +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 + + self._config_module = True if self._task.action == 'os10_config' else False + socket_path = None + + if self._play_context.connection == 'network_cli': + provider = self._task.args.get('provider', {}) + if any(provider.values()): + display.warning('provider is unnecessary when using network_cli and will be ignored') + del self._task.args['provider'] + elif self._play_context.connection == 'local': + provider = load_provider(os10_provider_spec, self._task.args) + pc = copy.deepcopy(self._play_context) + pc.connection = 'network_cli' + pc.network_os = 'dellemc.os10.os10' + 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 + command_timeout = int(provider['timeout'] or C.PERSISTENT_COMMAND_TIMEOUT) + pc.become = provider['authorize'] or False + if pc.become: + pc.become_method = 'enable' + pc.become_pass = provider['auth_pass'] + + display.vvv('using connection plugin %s' % pc.connection, pc.remote_addr) + connection = self._shared_loader_obj.connection_loader.get('persistent', pc, sys.stdin) + 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 + + # make sure we are in the right cli context which should be + # enable mode and not config module + if socket_path is None: + socket_path = self._connection.socket_path + + conn = Connection(socket_path) + out = conn.get_prompt() + while to_text(out, errors='surrogate_then_replace').strip().endswith(')#'): + display.vvvv('wrong context, sending exit to device', self._play_context.remote_addr) + conn.send_command('exit') + out = conn.get_prompt() + + result = super(ActionModule, self).run(task_vars=task_vars) + return result diff --git a/collections-debian-merged/ansible_collections/dellemc/os10/plugins/action/textfsm_parser.py b/collections-debian-merged/ansible_collections/dellemc/os10/plugins/action/textfsm_parser.py new file mode 100644 index 00000000..602186c8 --- /dev/null +++ b/collections-debian-merged/ansible_collections/dellemc/os10/plugins/action/textfsm_parser.py @@ -0,0 +1,81 @@ +# -*- coding: utf-8 -*- + +# (c) 2020, Ansible by Red Hat, inc +# GNU General Public License v3.0+ (see COPYING or https://www.gnu.org/licenses/gpl-3.0.txt) +# +# 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 + +from ansible.module_utils.six import StringIO, string_types + +from ansible.plugins.action import ActionBase +from ansible.errors import AnsibleError + +try: + import textfsm + HAS_TEXTFSM = True +except ImportError: + HAS_TEXTFSM = False + + +class ActionModule(ActionBase): + + def run(self, tmp=None, task_vars=None): + ''' handler for textfsm action ''' + + if task_vars is None: + task_vars = dict() + + result = super(ActionModule, self).run(tmp, task_vars) + del tmp # tmp no longer has any effect + + try: + if not HAS_TEXTFSM: + raise AnsibleError('textfsm_parser engine requires the TextFSM library to be installed') + + try: + filename = self._task.args.get('file') + src = self._task.args.get('src') + content = self._task.args['content'] + name = self._task.args.get('name') + except KeyError as exc: + raise AnsibleError('missing required argument: %s' % exc) + + if src and filename: + raise AnsibleError('`src` and `file` are mutually exclusive arguments') + + if not isinstance(content, string_types): + return {'failed': True, 'msg': '`content` must be of type str, got %s' % type(content)} + + if filename: + tmpl = open(filename) + else: + tmpl = StringIO() + tmpl.write(src.strip()) + tmpl.seek(0) + + try: + re_table = textfsm.TextFSM(tmpl) + fsm_results = re_table.ParseText(content) + + except Exception as exc: + raise AnsibleError(str(exc)) + + final_facts = [] + for item in fsm_results: + facts = {} + facts.update(dict(zip(re_table.header, item))) + final_facts.append(facts) + + if name: + result['ansible_facts'] = {name: final_facts} + else: + result['ansible_facts'] = {} + + finally: + self._remove_tmp_path(self._connection._shell.tmpdir) + + return result |