backup: fix handling target write error (like no disk space)

When writing process returns an error, prefer reporting that one,
instead of other process (which in most cases will be canceled, so
the error will be meaningless CancelledError).
Then, include sanitized stderr from VM process in the error message.
This commit is contained in:
Marek Marczykowski-Górecki 2017-11-06 20:50:02 +01:00
parent 11fd2cf115
commit 5e6ba4ff5c
No known key found for this signature in database
GPG Key ID: 063938BA42CFA724

View File

@ -23,6 +23,7 @@ from __future__ import unicode_literals
import itertools import itertools
import logging import logging
import functools import functools
import string
import termios import termios
import asyncio import asyncio
@ -665,6 +666,13 @@ class Backup(object):
raise raise
if proc.returncode: if proc.returncode:
if proc.stderr is not None:
proc_stderr = (yield from proc.stderr.read())
proc_stderr = proc_stderr.decode('ascii', errors='ignore')
proc_stderr = ''.join(
c for c in proc_stderr if c in string.printable and
c not in '\r\n%{}')
error_message += ': ' + proc_stderr
raise qubes.exc.QubesException(error_message) raise qubes.exc.QubesException(error_message)
@staticmethod @staticmethod
@ -717,7 +725,9 @@ class Backup(object):
# If APPVM, STDOUT is a PIPE # If APPVM, STDOUT is a PIPE
read_fd, write_fd = os.pipe() read_fd, write_fd = os.pipe()
vmproc = yield from self.target_vm.run_service('qubes.Backup', vmproc = yield from self.target_vm.run_service('qubes.Backup',
stdin=read_fd, stderr=subprocess.PIPE) stdin=read_fd,
stderr=subprocess.PIPE,
stdout=subprocess.DEVNULL)
os.close(read_fd) os.close(read_fd)
os.write(write_fd, (self.target_dir. os.write(write_fd, (self.target_dir.
replace("\r", "").replace("\n", "") + "\n").encode()) replace("\r", "").replace("\n", "") + "\n").encode())
@ -755,11 +765,12 @@ class Backup(object):
vmproc_task = None vmproc_task = None
if vmproc is not None: if vmproc is not None:
vmproc_task = asyncio.ensure_future(self._cancel_on_error( vmproc_task = asyncio.ensure_future(
self._monitor_process(vmproc, self._monitor_process(vmproc,
'Writing backup to VM {} failed'.format( 'Writing backup to VM {} failed'.format(
self.target_vm.name)), self.target_vm.name)))
send_task)) asyncio.ensure_future(self._cancel_on_error(
vmproc_task, send_task))
for file_name in header_files: for file_name in header_files:
yield from to_send.put(file_name) yield from to_send.put(file_name)
@ -783,10 +794,12 @@ class Backup(object):
except: except:
yield from to_send.put(QUEUE_ERROR) yield from to_send.put(QUEUE_ERROR)
# in fact we may be handling CancelledError, induced by # in fact we may be handling CancelledError, induced by
# exception in send_task (and propagated by # exception in send_task or vmproc_task (and propagated by
# self._cancel_on_error call above); in such a case this # self._cancel_on_error call above); in such a case this
# yield from will raise exception, covering CancelledError - # yield from will raise exception, covering CancelledError -
# this is intended behaviour # this is intended behaviour
if vmproc_task:
yield from vmproc_task
yield from send_task yield from send_task
raise raise