mirror of
https://github.com/ansible-collections/community.general.git
synced 2026-05-07 13:52:54 +00:00
adds support for using connection=netconf (#33400)
* adds support for using connection=netconf This change updates the module to provide support for using connection=netconf instead of connection=local. If connection=netconf is used, then the various connection arguments will be silently ignored. * adds netconf plugin default This adds a default implementation for netconf plugins if the network_os is not specified. The default plugin will implement only the standard netconf rpcs * fix up pep8 issues
This commit is contained in:
committed by
Ganesh Nalawade
parent
bd09e67438
commit
b4fa68555d
@@ -238,8 +238,7 @@ class Connection(ConnectionBase):
|
||||
if network_os:
|
||||
display.display('discovered network_os %s' % network_os, log_only=True)
|
||||
|
||||
if not network_os:
|
||||
raise AnsibleConnectionFailure('Unable to automatically determine host network os. Please ansible_network_os value')
|
||||
device_params = {'name': (network_os or 'default')}
|
||||
|
||||
ssh_config = os.getenv('ANSIBLE_NETCONF_SSH_CONFIG', False)
|
||||
if ssh_config in BOOLEANS_TRUE:
|
||||
@@ -256,9 +255,9 @@ class Connection(ConnectionBase):
|
||||
key_filename=str(key_filename),
|
||||
hostkey_verify=C.HOST_KEY_CHECKING,
|
||||
look_for_keys=C.PARAMIKO_LOOK_FOR_KEYS,
|
||||
device_params=device_params,
|
||||
allow_agent=self._play_context.allow_agent,
|
||||
timeout=self._play_context.timeout,
|
||||
device_params={'name': network_os},
|
||||
ssh_config=ssh_config
|
||||
)
|
||||
except SSHUnknownHostError as exc:
|
||||
|
||||
@@ -102,7 +102,7 @@ class NetconfBase(with_metaclass(ABCMeta, object)):
|
||||
:source: name of the configuration datastore being queried
|
||||
:filter: specifies the portion of the configuration to retrieve
|
||||
(by default entire configuration is retrieved)"""
|
||||
pass
|
||||
return self.m.get_config(*args, **kwargs).data_xml
|
||||
|
||||
@ensure_connected
|
||||
def get(self, *args, **kwargs):
|
||||
@@ -110,7 +110,7 @@ class NetconfBase(with_metaclass(ABCMeta, object)):
|
||||
*filter* specifies the portion of the configuration to retrieve
|
||||
(by default entire configuration is retrieved)
|
||||
"""
|
||||
pass
|
||||
return self.m.get(*args, **kwargs).data_xml
|
||||
|
||||
@ensure_connected
|
||||
def edit_config(self, *args, **kwargs):
|
||||
@@ -124,7 +124,7 @@ class NetconfBase(with_metaclass(ABCMeta, object)):
|
||||
:error_option: if specified must be one of { `"stop-on-error"`, `"continue-on-error"`, `"rollback-on-error"` }
|
||||
The `"rollback-on-error"` *error_option* depends on the `:rollback-on-error` capability.
|
||||
"""
|
||||
pass
|
||||
return self.m.edit_config(*args, **kwargs).xml
|
||||
|
||||
@ensure_connected
|
||||
def validate(self, *args, **kwargs):
|
||||
@@ -132,7 +132,7 @@ class NetconfBase(with_metaclass(ABCMeta, object)):
|
||||
:source: is the name of the configuration datastore being validated or `config`
|
||||
element containing the configuration subtree to be validated
|
||||
"""
|
||||
pass
|
||||
return self.m.validate(*args, **kwargs).xml
|
||||
|
||||
@ensure_connected
|
||||
def copy_config(self, *args, **kwargs):
|
||||
@@ -141,27 +141,27 @@ class NetconfBase(with_metaclass(ABCMeta, object)):
|
||||
:source: is the name of the configuration datastore to use as the source of the
|
||||
copy operation or `config` element containing the configuration subtree to copy
|
||||
:target: is the name of the configuration datastore to use as the destination of the copy operation"""
|
||||
return self.m.copy_config(*args, **kwargs).data_xml
|
||||
return self.m.copy_config(*args, **kwargs).xml
|
||||
|
||||
@ensure_connected
|
||||
def lock(self, *args, **kwargs):
|
||||
"""Allows the client to lock the configuration system of a device.
|
||||
*target* is the name of the configuration datastore to lock
|
||||
"""
|
||||
return self.m.lock(*args, **kwargs).data_xml
|
||||
return self.m.lock(*args, **kwargs).xml
|
||||
|
||||
@ensure_connected
|
||||
def unlock(self, *args, **kwargs):
|
||||
"""Release a configuration lock, previously obtained with the lock operation.
|
||||
:target: is the name of the configuration datastore to unlock
|
||||
"""
|
||||
return self.m.unlock(*args, **kwargs).data_xml
|
||||
return self.m.unlock(*args, **kwargs).xml
|
||||
|
||||
@ensure_connected
|
||||
def discard_changes(self, *args, **kwargs):
|
||||
"""Revert the candidate configuration to the currently running configuration.
|
||||
Any uncommitted changes are discarded."""
|
||||
pass
|
||||
return self.m.discard_changes(*args, **kwargs).xml
|
||||
|
||||
@ensure_connected
|
||||
def commit(self, *args, **kwargs):
|
||||
@@ -175,19 +175,19 @@ class NetconfBase(with_metaclass(ABCMeta, object)):
|
||||
:confirmed: whether this is a confirmed commit
|
||||
:timeout: specifies the confirm timeout in seconds
|
||||
"""
|
||||
pass
|
||||
return self.m.commit(*args, **kwargs).xml
|
||||
|
||||
@ensure_connected
|
||||
def validate(self, *args, **kwargs):
|
||||
"""Validate the contents of the specified configuration.
|
||||
:source: name of configuration data store"""
|
||||
return self.m.validate(*args, **kwargs).data_xml
|
||||
return self.m.validate(*args, **kwargs).xml
|
||||
|
||||
@ensure_connected
|
||||
def get_schema(self, *args, **kwargs):
|
||||
"""Retrieves the required schema from the device
|
||||
"""
|
||||
return self.m.get_schema(*args, **kwargs)
|
||||
return self.m.get_schema(*args, **kwargs).xml
|
||||
|
||||
@ensure_connected
|
||||
def locked(self, *args, **kwargs):
|
||||
@@ -220,4 +220,4 @@ class NetconfBase(with_metaclass(ABCMeta, object)):
|
||||
"""Fetch file over scp from remote device"""
|
||||
pass
|
||||
|
||||
# TODO Restore .data_xml, when ncclient supports it for all platforms
|
||||
# TODO Restore .xml, when ncclient supports it for all platforms
|
||||
|
||||
51
lib/ansible/plugins/netconf/default.py
Normal file
51
lib/ansible/plugins/netconf/default.py
Normal file
@@ -0,0 +1,51 @@
|
||||
#
|
||||
# (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
|
||||
|
||||
import json
|
||||
|
||||
from ansible.module_utils._text import to_text, to_bytes
|
||||
from ansible.plugins.netconf import NetconfBase
|
||||
|
||||
|
||||
class Netconf(NetconfBase):
|
||||
|
||||
def get_text(self, ele, tag):
|
||||
try:
|
||||
return to_text(ele.find(tag).text, errors='surrogate_then_replace').strip()
|
||||
except AttributeError:
|
||||
pass
|
||||
|
||||
def get_device_info(self):
|
||||
device_info = dict()
|
||||
device_info['network_os'] = 'default'
|
||||
return device_info
|
||||
|
||||
def get_capabilities(self):
|
||||
result = dict()
|
||||
result['rpc'] = self.get_base_rpc() + ['commit', 'discard_changes', 'validate', 'lock', 'unlock', 'copy_copy',
|
||||
'execute_rpc', 'load_configuration', 'get_configuration', 'command',
|
||||
'reboot', 'halt']
|
||||
result['network_api'] = 'netconf'
|
||||
result['device_info'] = self.get_device_info()
|
||||
result['server_capabilities'] = [c for c in self.m.server_capabilities]
|
||||
result['client_capabilities'] = [c for c in self.m.client_capabilities]
|
||||
result['session_id'] = self.m.session_id
|
||||
return json.dumps(result)
|
||||
Reference in New Issue
Block a user