mirror of
https://github.com/ansible-collections/community.general.git
synced 2024-09-14 20:13:21 +02:00
Added pct.py
This commit is contained in:
parent
da2c87ce0d
commit
f8db305523
1 changed files with 203 additions and 0 deletions
203
plugins/connection/pct.py
Normal file
203
plugins/connection/pct.py
Normal file
|
@ -0,0 +1,203 @@
|
||||||
|
# -*- coding: utf-8 -*-
|
||||||
|
# Based on lxc.py (c) 2015, Joerg Thalheim <joerg@higgsboson.tk>
|
||||||
|
# Based on lxd.py (c) 2016 Matt Clay <matt@mystile.com>
|
||||||
|
# Copyright (c) 2024 Nils Stein <github.nstein@mailbox.org>
|
||||||
|
# Copyright (c) 2024 Ansible Project
|
||||||
|
# GNU General Public License v3.0+ (see LICENSES/GPL-3.0-or-later.txt or https://www.gnu.org/licenses/gpl-3.0.txt)
|
||||||
|
# SPDX-License-Identifier: GPL-3.0-or-later
|
||||||
|
|
||||||
|
from __future__ import (absolute_import, division, print_function)
|
||||||
|
__metaclass__ = type
|
||||||
|
|
||||||
|
DOCUMENTATION = '''
|
||||||
|
author: Nils Stein (@mietzen) <github.nstein@mailbox.org>
|
||||||
|
name: pct
|
||||||
|
short_description: Run tasks in Proxmox LXC container instances via pct CLI
|
||||||
|
description:
|
||||||
|
- Run commands or put/fetch files to an existing Proxmox LXC container instance using pct CLI.
|
||||||
|
options:
|
||||||
|
remote_addr:
|
||||||
|
description:
|
||||||
|
- Container Name
|
||||||
|
default: inventory_hostname
|
||||||
|
vars:
|
||||||
|
- name: inventory_hostname
|
||||||
|
- name: ansible_host
|
||||||
|
- name: ansible_pct_host
|
||||||
|
vmid:
|
||||||
|
description:
|
||||||
|
- Container ID
|
||||||
|
default: proxmox_vmid
|
||||||
|
vars:
|
||||||
|
- name: proxmox_vmid
|
||||||
|
executable:
|
||||||
|
default: /bin/sh
|
||||||
|
description:
|
||||||
|
- Shell executable
|
||||||
|
vars:
|
||||||
|
- name: ansible_executable
|
||||||
|
- name: ansible_pct_executable
|
||||||
|
'''
|
||||||
|
|
||||||
|
import os
|
||||||
|
from subprocess import Popen, PIPE
|
||||||
|
|
||||||
|
from ansible.errors import AnsibleError, AnsibleConnectionFailure, AnsibleFileNotFound
|
||||||
|
from ansible.module_utils.common.process import get_bin_path
|
||||||
|
from ansible.module_utils.common.text.converters import to_bytes, to_text
|
||||||
|
from ansible.plugins.connection import ConnectionBase
|
||||||
|
|
||||||
|
|
||||||
|
class Connection(ConnectionBase):
|
||||||
|
""" pct based connections """
|
||||||
|
|
||||||
|
transport = 'community.general.pct'
|
||||||
|
has_pipelining = True
|
||||||
|
default_user = 'root'
|
||||||
|
|
||||||
|
def __init__(self, play_context, new_stdin, *args, **kwargs):
|
||||||
|
super(Connection, self).__init__(play_context, new_stdin, *args, **kwargs)
|
||||||
|
|
||||||
|
self.container_name = None
|
||||||
|
self.container_vmid = None
|
||||||
|
|
||||||
|
try:
|
||||||
|
self._pct_cmd = get_bin_path("pct")
|
||||||
|
except ValueError:
|
||||||
|
raise AnsibleError("pct command not found in PATH")
|
||||||
|
try:
|
||||||
|
self._pvesh_cmd = get_bin_path("pvesh")
|
||||||
|
except ValueError:
|
||||||
|
raise AnsibleError("pvesh command not found in PATH")
|
||||||
|
|
||||||
|
if self._play_context.remote_user is not None and self._play_context.remote_user != 'root':
|
||||||
|
self._display.warning('pct does not support remote_user, using default: root')
|
||||||
|
|
||||||
|
def _vmid(self):
|
||||||
|
""" Get the vmid via hostname """
|
||||||
|
if not self.container_vmid:
|
||||||
|
self.container_vmid = self.get_option('vmid')
|
||||||
|
return self.container_vmid
|
||||||
|
|
||||||
|
# def _vmid(self):
|
||||||
|
# """ Get the vmid via hostname """
|
||||||
|
# if not self.container_vmid:
|
||||||
|
# local_cmd = [self._pvesh_cmd]
|
||||||
|
# local_cmd.extend([
|
||||||
|
# "get", "/cluster/resources", "--output-format", "json"])
|
||||||
|
# self._display.vvvvv(u"EXEC {0}".format(local_cmd), host=self._host())
|
||||||
|
|
||||||
|
# local_cmd = [to_bytes(i, errors='surrogate_or_strict') for i in local_cmd]
|
||||||
|
|
||||||
|
# process = Popen(local_cmd, stdin=PIPE, stdout=PIPE, stderr=PIPE)
|
||||||
|
# stdout, stderr = process.communicate()
|
||||||
|
|
||||||
|
# stdout = to_text(stdout)
|
||||||
|
# stderr = to_text(stderr)
|
||||||
|
|
||||||
|
# self._display.vvvvv(u"EXEC pvesh output: {0} {1}".format(stdout, stderr), host=self._host())
|
||||||
|
# try:
|
||||||
|
# cluster_resources = json.loads(stdout)
|
||||||
|
# self.container_vmid = [x['vmid'] for x in cluster_resources if x['name'] == self.container_name][0]
|
||||||
|
# except KeyError:
|
||||||
|
# raise AnsibleError(f"Got unknow json structure, expected keys name name vmid to exist, instead got: {json.dumps(cluster_resources)}")
|
||||||
|
# except JSONDecodeError:
|
||||||
|
# raise AnsibleError(f"Output not parsable as json: {stdout}")
|
||||||
|
# if not self.container_vmid:
|
||||||
|
# raise AnsibleError(f"Containter: {self.container_name}, not found!")
|
||||||
|
# return self.container_vmid
|
||||||
|
|
||||||
|
def _host(self):
|
||||||
|
""" translate remote_addr to pct (short) hostname """
|
||||||
|
if not self.container_name:
|
||||||
|
self.container_name = self.get_option('remote_addr')
|
||||||
|
return self.container_name
|
||||||
|
|
||||||
|
def _connect(self):
|
||||||
|
"""connect to proxmox (nothing to do here) """
|
||||||
|
super(Connection, self)._connect()
|
||||||
|
|
||||||
|
if not self._connected:
|
||||||
|
self._display.vvv(u"ESTABLISH PCT CONNECTION FOR USER: root", host=self._host())
|
||||||
|
self._connected = True
|
||||||
|
|
||||||
|
def exec_command(self, cmd, in_data=None, sudoable=True):
|
||||||
|
""" execute a command inside the proxmox container """
|
||||||
|
super(Connection, self).exec_command(cmd, in_data=in_data, sudoable=sudoable)
|
||||||
|
|
||||||
|
self._display.vvv(u"EXEC {0}".format(cmd), host=self._host())
|
||||||
|
|
||||||
|
local_cmd = [self._pct_cmd]
|
||||||
|
local_cmd.extend([
|
||||||
|
"exec",
|
||||||
|
self._vmid(),
|
||||||
|
"--",
|
||||||
|
self.get_option("executable"), "-c", cmd
|
||||||
|
])
|
||||||
|
|
||||||
|
self._display.vvvvv(u"EXEC {0}".format(local_cmd), host=self._host())
|
||||||
|
|
||||||
|
local_cmd = [to_bytes(i, errors='surrogate_or_strict') for i in local_cmd]
|
||||||
|
in_data = to_bytes(in_data, errors='surrogate_or_strict', nonstring='passthru')
|
||||||
|
|
||||||
|
process = Popen(local_cmd, stdin=PIPE, stdout=PIPE, stderr=PIPE)
|
||||||
|
stdout, stderr = process.communicate(in_data)
|
||||||
|
|
||||||
|
stdout = to_text(stdout)
|
||||||
|
stderr = to_text(stderr)
|
||||||
|
|
||||||
|
self._display.vvvvv(u"EXEC ptc output: {0} {1}".format(stdout, stderr), host=self._host())
|
||||||
|
|
||||||
|
if "not running!" in stderr:
|
||||||
|
raise AnsibleConnectionFailure("instance not running: %s" % self._host())
|
||||||
|
|
||||||
|
if "does not exist" in stderr:
|
||||||
|
raise AnsibleConnectionFailure("instance not found: %s" % self._host())
|
||||||
|
|
||||||
|
return process.returncode, stdout, stderr
|
||||||
|
|
||||||
|
def put_file(self, in_path, out_path):
|
||||||
|
""" put a file from local into a proxmox container """
|
||||||
|
super(Connection, self).put_file(in_path, out_path)
|
||||||
|
|
||||||
|
self._display.vvv(u"PUT {0} TO {1}".format(in_path, out_path), host=self._host())
|
||||||
|
|
||||||
|
if not os.path.isfile(to_bytes(in_path, errors='surrogate_or_strict')):
|
||||||
|
raise AnsibleFileNotFound("input path is not a file: %s" % in_path)
|
||||||
|
|
||||||
|
local_cmd = [self._pct_cmd]
|
||||||
|
local_cmd.extend([
|
||||||
|
"push",
|
||||||
|
self._vmid(),
|
||||||
|
in_path,
|
||||||
|
out_path
|
||||||
|
])
|
||||||
|
|
||||||
|
local_cmd = [to_bytes(i, errors='surrogate_or_strict') for i in local_cmd]
|
||||||
|
|
||||||
|
process = Popen(local_cmd, stdin=PIPE, stdout=PIPE, stderr=PIPE)
|
||||||
|
process.communicate()
|
||||||
|
|
||||||
|
def fetch_file(self, in_path, out_path):
|
||||||
|
""" fetch a file from a proxmox container to local """
|
||||||
|
super(Connection, self).fetch_file(in_path, out_path)
|
||||||
|
|
||||||
|
self._display.vvv(u"FETCH {0} TO {1}".format(in_path, out_path), host=self._host())
|
||||||
|
|
||||||
|
local_cmd = [self._pct_cmd]
|
||||||
|
local_cmd.extend([
|
||||||
|
"pull",
|
||||||
|
self._vmid(),
|
||||||
|
in_path,
|
||||||
|
out_path
|
||||||
|
])
|
||||||
|
local_cmd = [to_bytes(i, errors='surrogate_or_strict') for i in local_cmd]
|
||||||
|
|
||||||
|
process = Popen(local_cmd, stdin=PIPE, stdout=PIPE, stderr=PIPE)
|
||||||
|
process.communicate()
|
||||||
|
|
||||||
|
def close(self):
|
||||||
|
""" close the connection (nothing to do here) """
|
||||||
|
super(Connection, self).close()
|
||||||
|
|
||||||
|
self._connected = False
|
Loading…
Add table
Reference in a new issue