2016-08-10 19:45:54 +02:00
|
|
|
# This code is part of Ansible, but is an independent component.
|
|
|
|
# This particular file snippet, and this file snippet only, is BSD licensed.
|
|
|
|
# Modules you write using this snippet, which is embedded dynamically by Ansible
|
|
|
|
# still belong to the author of the module, and may assign their own license
|
|
|
|
# to the complete work.
|
2016-06-20 18:11:48 +02:00
|
|
|
#
|
2016-08-10 19:45:54 +02:00
|
|
|
# Copyright (c) 2015 Peter Sprygada, <psprygada@ansible.com>
|
2016-06-20 18:11:48 +02:00
|
|
|
#
|
2016-08-10 19:45:54 +02:00
|
|
|
# Redistribution and use in source and binary forms, with or without modification,
|
|
|
|
# are permitted provided that the following conditions are met:
|
2016-06-20 18:11:48 +02:00
|
|
|
#
|
2016-08-10 19:45:54 +02:00
|
|
|
# * Redistributions of source code must retain the above copyright
|
|
|
|
# notice, this list of conditions and the following disclaimer.
|
|
|
|
# * Redistributions in binary form must reproduce the above copyright notice,
|
|
|
|
# this list of conditions and the following disclaimer in the documentation
|
|
|
|
# and/or other materials provided with the distribution.
|
2016-06-20 18:11:48 +02:00
|
|
|
#
|
2016-08-10 19:45:54 +02:00
|
|
|
# THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND
|
|
|
|
# ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
|
|
|
|
# WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED.
|
|
|
|
# IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT,
|
|
|
|
# INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
|
|
|
|
# PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
|
|
|
|
# INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
|
|
|
|
# LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE
|
|
|
|
# USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
2017-08-12 05:23:17 +02:00
|
|
|
|
|
|
|
import traceback
|
|
|
|
|
2016-06-20 18:11:48 +02:00
|
|
|
from ansible.module_utils.basic import AnsibleModule
|
2017-08-12 05:23:17 +02:00
|
|
|
from ansible.module_utils.basic import env_fallback
|
|
|
|
from ansible.module_utils.netcli import Cli
|
2016-10-28 19:48:16 +02:00
|
|
|
from ansible.module_utils._text import to_native
|
2017-01-07 02:22:17 +01:00
|
|
|
from ansible.module_utils.six import iteritems
|
2016-06-20 18:11:48 +02:00
|
|
|
|
2017-08-12 05:23:17 +02:00
|
|
|
|
2016-06-20 18:11:48 +02:00
|
|
|
NET_TRANSPORT_ARGS = dict(
|
|
|
|
host=dict(required=True),
|
|
|
|
port=dict(type='int'),
|
2016-08-20 13:57:39 +02:00
|
|
|
|
2016-06-20 18:11:48 +02:00
|
|
|
username=dict(fallback=(env_fallback, ['ANSIBLE_NET_USERNAME'])),
|
|
|
|
password=dict(no_log=True, fallback=(env_fallback, ['ANSIBLE_NET_PASSWORD'])),
|
|
|
|
ssh_keyfile=dict(fallback=(env_fallback, ['ANSIBLE_NET_SSH_KEYFILE']), type='path'),
|
2016-08-20 13:57:39 +02:00
|
|
|
|
2016-06-20 18:11:48 +02:00
|
|
|
authorize=dict(default=False, fallback=(env_fallback, ['ANSIBLE_NET_AUTHORIZE']), type='bool'),
|
|
|
|
auth_pass=dict(no_log=True, fallback=(env_fallback, ['ANSIBLE_NET_AUTH_PASS'])),
|
2016-08-20 13:57:39 +02:00
|
|
|
|
2017-03-13 20:21:46 +01:00
|
|
|
provider=dict(type='dict', no_log=True),
|
2016-06-20 18:11:48 +02:00
|
|
|
transport=dict(choices=list()),
|
2016-08-20 13:57:39 +02:00
|
|
|
|
2016-06-20 18:11:48 +02:00
|
|
|
timeout=dict(default=10, type='int')
|
|
|
|
)
|
|
|
|
|
|
|
|
NET_CONNECTION_ARGS = dict()
|
|
|
|
|
|
|
|
NET_CONNECTIONS = dict()
|
|
|
|
|
2017-06-02 13:14:11 +02:00
|
|
|
|
2017-01-07 02:22:17 +01:00
|
|
|
def _transitional_argument_spec():
|
|
|
|
argument_spec = {}
|
|
|
|
for key, value in iteritems(NET_TRANSPORT_ARGS):
|
|
|
|
value['required'] = False
|
|
|
|
argument_spec[key] = value
|
|
|
|
return argument_spec
|
2016-06-20 18:11:48 +02:00
|
|
|
|
2017-06-02 13:14:11 +02:00
|
|
|
|
2016-06-20 18:11:48 +02:00
|
|
|
def to_list(val):
|
|
|
|
if isinstance(val, (list, tuple)):
|
|
|
|
return list(val)
|
|
|
|
elif val is not None:
|
|
|
|
return [val]
|
|
|
|
else:
|
|
|
|
return list()
|
|
|
|
|
|
|
|
|
2016-07-07 20:32:19 +02:00
|
|
|
class ModuleStub(object):
|
|
|
|
def __init__(self, argument_spec, fail_json):
|
|
|
|
self.params = dict()
|
|
|
|
for key, value in argument_spec.items():
|
|
|
|
self.params[key] = value.get('default')
|
|
|
|
self.fail_json = fail_json
|
|
|
|
|
2017-06-02 13:14:11 +02:00
|
|
|
|
2016-06-20 18:11:48 +02:00
|
|
|
class NetworkError(Exception):
|
|
|
|
|
|
|
|
def __init__(self, msg, **kwargs):
|
|
|
|
super(NetworkError, self).__init__(msg)
|
|
|
|
self.kwargs = kwargs
|
|
|
|
|
2017-06-02 13:14:11 +02:00
|
|
|
|
2017-01-05 05:23:08 +01:00
|
|
|
class Config(object):
|
|
|
|
|
|
|
|
def __init__(self, connection):
|
|
|
|
self.connection = connection
|
|
|
|
|
|
|
|
def __call__(self, commands, **kwargs):
|
|
|
|
lines = to_list(commands)
|
|
|
|
return self.connection.configure(lines, **kwargs)
|
|
|
|
|
|
|
|
def load_config(self, commands, **kwargs):
|
|
|
|
commands = to_list(commands)
|
|
|
|
return self.connection.load_config(commands, **kwargs)
|
|
|
|
|
|
|
|
def get_config(self, **kwargs):
|
|
|
|
return self.connection.get_config(**kwargs)
|
|
|
|
|
|
|
|
def save_config(self):
|
|
|
|
return self.connection.save_config()
|
|
|
|
|
2016-06-20 18:11:48 +02:00
|
|
|
|
|
|
|
class NetworkModule(AnsibleModule):
|
|
|
|
|
|
|
|
def __init__(self, *args, **kwargs):
|
2016-07-07 03:24:24 +02:00
|
|
|
connect_on_load = kwargs.pop('connect_on_load', True)
|
|
|
|
|
|
|
|
argument_spec = NET_TRANSPORT_ARGS.copy()
|
|
|
|
argument_spec['transport']['choices'] = NET_CONNECTIONS.keys()
|
|
|
|
argument_spec.update(NET_CONNECTION_ARGS.copy())
|
|
|
|
|
|
|
|
if kwargs.get('argument_spec'):
|
|
|
|
argument_spec.update(kwargs['argument_spec'])
|
|
|
|
kwargs['argument_spec'] = argument_spec
|
|
|
|
|
2016-06-20 18:11:48 +02:00
|
|
|
super(NetworkModule, self).__init__(*args, **kwargs)
|
2016-07-09 13:03:18 +02:00
|
|
|
|
2016-06-20 18:11:48 +02:00
|
|
|
self.connection = None
|
|
|
|
self._cli = None
|
|
|
|
self._config = None
|
|
|
|
|
2016-07-07 03:24:24 +02:00
|
|
|
try:
|
|
|
|
transport = self.params['transport'] or '__default__'
|
|
|
|
cls = NET_CONNECTIONS[transport]
|
|
|
|
self.connection = cls()
|
|
|
|
except KeyError:
|
|
|
|
self.fail_json(msg='Unknown transport or no default transport specified')
|
2017-08-12 05:23:17 +02:00
|
|
|
except (TypeError, NetworkError) as exc:
|
|
|
|
self.fail_json(msg=to_native(exc), exception=traceback.format_exc())
|
2016-07-07 03:24:24 +02:00
|
|
|
|
|
|
|
if connect_on_load:
|
|
|
|
self.connect()
|
|
|
|
|
2016-06-20 18:11:48 +02:00
|
|
|
@property
|
|
|
|
def cli(self):
|
|
|
|
if not self.connected:
|
2016-07-09 13:03:18 +02:00
|
|
|
self.connect()
|
2016-06-20 18:11:48 +02:00
|
|
|
if self._cli:
|
|
|
|
return self._cli
|
|
|
|
self._cli = Cli(self.connection)
|
|
|
|
return self._cli
|
|
|
|
|
|
|
|
@property
|
|
|
|
def config(self):
|
|
|
|
if not self.connected:
|
2016-07-09 13:03:18 +02:00
|
|
|
self.connect()
|
2016-06-20 18:11:48 +02:00
|
|
|
if self._config:
|
|
|
|
return self._config
|
|
|
|
self._config = Config(self.connection)
|
|
|
|
return self._config
|
|
|
|
|
|
|
|
@property
|
|
|
|
def connected(self):
|
|
|
|
return self.connection._connected
|
|
|
|
|
|
|
|
def _load_params(self):
|
|
|
|
super(NetworkModule, self)._load_params()
|
|
|
|
provider = self.params.get('provider') or dict()
|
|
|
|
for key, value in provider.items():
|
|
|
|
for args in [NET_TRANSPORT_ARGS, NET_CONNECTION_ARGS]:
|
|
|
|
if key in args:
|
|
|
|
if self.params.get(key) is None and value is not None:
|
|
|
|
self.params[key] = value
|
|
|
|
|
2016-07-07 03:24:24 +02:00
|
|
|
def connect(self):
|
|
|
|
try:
|
|
|
|
if not self.connected:
|
|
|
|
self.connection.connect(self.params)
|
|
|
|
if self.params['authorize']:
|
|
|
|
self.connection.authorize(self.params)
|
2016-10-13 03:47:58 +02:00
|
|
|
self.log('connected to %s:%s using %s' % (self.params['host'],
|
|
|
|
self.params['port'], self.params['transport']))
|
2017-08-12 05:23:17 +02:00
|
|
|
except NetworkError as exc:
|
|
|
|
self.fail_json(msg=to_native(exc), exception=traceback.format_exc())
|
2016-07-07 03:24:24 +02:00
|
|
|
|
|
|
|
def disconnect(self):
|
|
|
|
try:
|
|
|
|
if self.connected:
|
|
|
|
self.connection.disconnect()
|
2016-10-13 03:47:58 +02:00
|
|
|
self.log('disconnected from %s' % self.params['host'])
|
2017-08-12 05:23:17 +02:00
|
|
|
except NetworkError as exc:
|
|
|
|
self.fail_json(msg=to_native(exc), exception=traceback.format_exc())
|
2016-07-07 03:24:24 +02:00
|
|
|
|
2017-06-02 13:14:11 +02:00
|
|
|
|
2016-06-20 18:11:48 +02:00
|
|
|
def register_transport(transport, default=False):
|
|
|
|
def register(cls):
|
|
|
|
NET_CONNECTIONS[transport] = cls
|
|
|
|
if default:
|
|
|
|
NET_CONNECTIONS['__default__'] = cls
|
|
|
|
return cls
|
|
|
|
return register
|
|
|
|
|
2017-06-02 13:14:11 +02:00
|
|
|
|
2016-06-20 18:11:48 +02:00
|
|
|
def add_argument(key, value):
|
|
|
|
NET_CONNECTION_ARGS[key] = value
|