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:
parent
11fd2cf115
commit
5e6ba4ff5c
@ -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
|
||||||
|
|
||||||
|
Loading…
Reference in New Issue
Block a user