2021-08-08 01:02:21 +12:00
|
|
|
# -*- coding: utf-8 -*-
|
2020-03-09 09:11:07 +00:00
|
|
|
# Based on local.py (c) 2012, Michael DeHaan <michael.dehaan@gmail.com>
|
|
|
|
# Based on chroot.py (c) 2013, Maykel Moya <mmoya@speedyrails.com>
|
|
|
|
# Copyright (c) 2013, Michael Scherer <misc@zarb.org>
|
|
|
|
# Copyright (c) 2017 Ansible Project
|
|
|
|
# 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 = '''
|
2021-07-26 11:44:41 +02:00
|
|
|
author: Michael Scherer (@mscherer) <misc@zarb.org>
|
2021-01-12 07:12:03 +01:00
|
|
|
name: funcd
|
2020-03-09 09:11:07 +00:00
|
|
|
short_description: Use funcd to connect to target
|
|
|
|
description:
|
|
|
|
- This transport permits you to use Ansible over Func.
|
|
|
|
- For people who have already setup func and that wish to play with ansible,
|
|
|
|
this permit to move gradually to ansible without having to redo completely the setup of the network.
|
|
|
|
options:
|
|
|
|
remote_addr:
|
|
|
|
description:
|
|
|
|
- The path of the chroot you want to access.
|
|
|
|
default: inventory_hostname
|
|
|
|
vars:
|
|
|
|
- name: ansible_host
|
|
|
|
- name: ansible_func_host
|
|
|
|
'''
|
|
|
|
|
|
|
|
HAVE_FUNC = False
|
|
|
|
try:
|
|
|
|
import func.overlord.client as fc
|
|
|
|
HAVE_FUNC = True
|
|
|
|
except ImportError:
|
|
|
|
pass
|
|
|
|
|
|
|
|
import os
|
|
|
|
import tempfile
|
|
|
|
import shutil
|
|
|
|
|
|
|
|
from ansible.errors import AnsibleError
|
2021-04-14 03:25:57 -04:00
|
|
|
from ansible.plugins.connection import ConnectionBase
|
2020-03-09 09:11:07 +00:00
|
|
|
from ansible.utils.display import Display
|
|
|
|
|
|
|
|
display = Display()
|
|
|
|
|
|
|
|
|
2021-04-14 03:25:57 -04:00
|
|
|
class Connection(ConnectionBase):
|
2021-05-16 23:24:37 +12:00
|
|
|
""" Func-based connections """
|
2020-03-09 09:11:07 +00:00
|
|
|
|
|
|
|
has_pipelining = False
|
|
|
|
|
|
|
|
def __init__(self, runner, host, port, *args, **kwargs):
|
|
|
|
self.runner = runner
|
|
|
|
self.host = host
|
|
|
|
# port is unused, this go on func
|
|
|
|
self.port = port
|
2021-05-16 23:24:37 +12:00
|
|
|
self.client = None
|
2020-03-09 09:11:07 +00:00
|
|
|
|
|
|
|
def connect(self, port=None):
|
|
|
|
if not HAVE_FUNC:
|
|
|
|
raise AnsibleError("func is not installed")
|
|
|
|
|
|
|
|
self.client = fc.Client(self.host)
|
|
|
|
return self
|
|
|
|
|
|
|
|
def exec_command(self, cmd, become_user=None, sudoable=False, executable='/bin/sh', in_data=None):
|
2021-05-16 23:24:37 +12:00
|
|
|
""" run a command on the remote minion """
|
2020-03-09 09:11:07 +00:00
|
|
|
|
|
|
|
if in_data:
|
|
|
|
raise AnsibleError("Internal Error: this module does not support optimized module pipelining")
|
|
|
|
|
|
|
|
# totally ignores privlege escalation
|
2021-05-16 23:24:37 +12:00
|
|
|
display.vvv("EXEC %s" % cmd, host=self.host)
|
2020-03-09 09:11:07 +00:00
|
|
|
p = self.client.command.run(cmd)[self.host]
|
2021-05-16 23:24:37 +12:00
|
|
|
return p[0], p[1], p[2]
|
2020-03-09 09:11:07 +00:00
|
|
|
|
2021-05-16 23:24:37 +12:00
|
|
|
@staticmethod
|
|
|
|
def _normalize_path(path, prefix):
|
2020-03-09 09:11:07 +00:00
|
|
|
if not path.startswith(os.path.sep):
|
|
|
|
path = os.path.join(os.path.sep, path)
|
|
|
|
normpath = os.path.normpath(path)
|
|
|
|
return os.path.join(prefix, normpath[1:])
|
|
|
|
|
|
|
|
def put_file(self, in_path, out_path):
|
2021-05-16 23:24:37 +12:00
|
|
|
""" transfer a file from local to remote """
|
2020-03-09 09:11:07 +00:00
|
|
|
|
|
|
|
out_path = self._normalize_path(out_path, '/')
|
|
|
|
display.vvv("PUT %s TO %s" % (in_path, out_path), host=self.host)
|
|
|
|
self.client.local.copyfile.send(in_path, out_path)
|
|
|
|
|
|
|
|
def fetch_file(self, in_path, out_path):
|
2021-05-16 23:24:37 +12:00
|
|
|
""" fetch a file from remote to local """
|
2020-03-09 09:11:07 +00:00
|
|
|
|
|
|
|
in_path = self._normalize_path(in_path, '/')
|
|
|
|
display.vvv("FETCH %s TO %s" % (in_path, out_path), host=self.host)
|
|
|
|
# need to use a tmp dir due to difference of semantic for getfile
|
|
|
|
# ( who take a # directory as destination) and fetch_file, who
|
|
|
|
# take a file directly
|
|
|
|
tmpdir = tempfile.mkdtemp(prefix="func_ansible")
|
|
|
|
self.client.local.getfile.get(in_path, tmpdir)
|
|
|
|
shutil.move(os.path.join(tmpdir, self.host, os.path.basename(in_path)), out_path)
|
|
|
|
shutil.rmtree(tmpdir)
|
|
|
|
|
|
|
|
def close(self):
|
2021-05-16 23:24:37 +12:00
|
|
|
""" terminate the connection; nothing to do here """
|
2020-03-09 09:11:07 +00:00
|
|
|
pass
|