Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Look in to using vzdump --stdout #1

Open
farthinder opened this issue Jan 5, 2020 · 1 comment
Open

Look in to using vzdump --stdout #1

farthinder opened this issue Jan 5, 2020 · 1 comment
Labels
enhancement New feature or request

Comments

@farthinder
Copy link
Owner

This might allow for skipping the .vma and .log file and go straight to .enc.

Potential issues are

  1. task log output
  2. dealing with the fact it´s two files given to stdout

https://pve.proxmox.com/pve-docs/vzdump.1.html

@farthinder farthinder added the enhancement New feature or request label Jan 5, 2020
@farthinder
Copy link
Owner Author

I've dug in a bit to this.
This POC appears to work, however Proxmox does not accept the --stdout in the vzdump crontab

vzdump 105 --stdout true | /opt/scripting/encryptedRemoteBackup/test2.py

`#!/usr/bin/python3

-- coding: utf-8 --

#Source of class:
#https://gist.github.com/tgwizard/1287d724afe7086d0f4882a4724caec7
import subprocess
from threading import Thread
from time import time
from typing import Union
from queue import Queue
import sys

class SubprocessInputStreamer:
_end_of_stream = object()

def __init__(self, popen_args, popen_kwargs=None, timeout=10):
    self.popen_args = popen_args
    self.popen_kwargs = popen_kwargs
    self.running = False
    self.timeout = timeout

def start(self):
    assert not self.running

    self.stdout = None
    self.stderr = None
    self.error = None
    self.exit_code = None

    self._process = subprocess.Popen(self.popen_args, **(self.popen_kwargs or {}))

    self._queue = Queue()
    self.running = True

    if self._process.stdin is not None:
        self._stdin_thread = Thread(target=self._thread_write)
        self._stdin_thread.daemon = True
        self._stdin_thread.start()

    if self._process.stdout is not None:
        self._stdout_thread = Thread(
            target=self._thread_read, args=(self._process.stdout, 'stdout')
        )
        self._stdout_thread.daemon = True
        self._stdout_thread.start()

    if self._process.stderr is not None:
        self._stderr_thread = Thread(
            target=self._thread_read, args=(self._process.stderr, 'stderr')
        )
        self._stderr_thread.daemon = True
        self._stderr_thread.start()

def stop(self):
    if not self.running:
        return

    end_time = time() + self.timeout

    self._queue.put(self._end_of_stream)
    self._queue.join()
    self._process.stdin.flush()
    self._process.stdin.close()

    self.running = False

    try:
        if self._process.stdin:
            self._stdin_thread.join(timeout=self._remaining_time(end_time))
            if self._stdin_thread.is_alive():
                raise subprocess.TimeoutExpired(self._process.args, self.timeout)
        if self._process.stdout:
            self._stdout_thread.join(timeout=self._remaining_time(end_time))
            if self._stdout_thread.is_alive():
                raise subprocess.TimeoutExpired(self._process.args, self.timeout)
        if self._process.stderr:
            self._stderr_thread.join(timeout=self._remaining_time(end_time))
            if self._stderr_thread.is_alive():
                raise subprocess.TimeoutExpired(self._process.args, self.timeout)

        self.exit_code = self._process.wait(timeout=self._remaining_time(end_time))
    except Exception as e:
        self.error = e
        self._process.kill()
        self._process.wait()
    finally:
        self._process = None
        self._queue = None

@classmethod
def _remaining_time(cls, end_time):
    return end_time - time()

def __enter__(self):
    self.start()
    return self

def __exit__(self, exc_type, exc_val, exc_tb):
    self.stop()

def write(self, b: Union[bytes, memoryview]):
    assert self.running, 'Must start before writing'
    if isinstance(b, memoryview):
        b = bytes(b)
    self._queue.put(b)

def _thread_write(self):
    while self.running:
        b = self._queue.get()
        if b is self._end_of_stream:
            self._queue.task_done()
            return

        try:
            self._process.stdin.write(b)
        except BrokenPipeError:
            # Failed to write to subprocess stdin (broken pipe, stdin likely closed.
            pass
        except Exception:
            # Failed to write to subprocess stdin. Handle appropriately (but make
            # sure to consume the queue).
            pass
        finally:
            self._queue.task_done()

def _thread_read(self, fp, out_name):
    setattr(self, out_name, fp.read())
    fp.close()

sub = SubprocessInputStreamer(popen_args=['/usr/bin/gpg', '--output', '/opt/scripting/encryptedRemoteBackup/temp/test2.gpg', '-e', '--recipient', '[email protected]'])

sub.start()
sub.write(sys.stdin)`

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
enhancement New feature or request
Projects
None yet
Development

No branches or pull requests

1 participant