Compare commits

...

4 Commits

Author SHA1 Message Date
46f9f98860 backup: show errors at top of email notification 2021-10-26 13:24:39 -04:00
dc7d72b2da initial-setup: make systemd units restart on failure 2021-10-26 13:24:28 -04:00
cb12e09c46 backup: rework output to make notification emails easier to read 2021-10-26 13:20:21 -04:00
1115d1f821 backup: rename pstr() helper to b2s()
The helper is just bytes->str conversion with errors=backslashreplace,
which we can use for more than just paths.
2021-10-26 12:54:41 -04:00
2 changed files with 65 additions and 48 deletions

111
backup.py
View File

@ -23,8 +23,8 @@ import yaml
import wcmatch.glob # type: ignore import wcmatch.glob # type: ignore
import humanfriendly # type: ignore import humanfriendly # type: ignore
def pstr(path: bytes) -> str: def b2s(raw: bytes) -> str:
return path.decode(errors='backslashreplace') return raw.decode(errors='backslashreplace')
def format_size(n: int) -> str: def format_size(n: int) -> str:
return humanfriendly.format_size(n, keep_width=True, binary=True) return humanfriendly.format_size(n, keep_width=True, binary=True)
@ -111,28 +111,31 @@ class Backup:
self.dry_run = dry_run self.dry_run = dry_run
self.root_seen: typing.Dict[bytes, bool] = {} self.root_seen: typing.Dict[bytes, bool] = {}
# Saved log messages # Saved log messages (which includes borg output)
self.logs: typing.List[typing.Tuple[str, str]] = [] self.logs: typing.List[typing.Tuple[str, str]] = []
# All captured borg output
self.captured_output: typing.List[bytes] = []
def out(self, path: bytes): def out(self, path: bytes):
self.outfile.write(path + (b'\n' if self.dry_run else b'\0')) self.outfile.write(path + (b'\n' if self.dry_run else b'\0'))
def log(self, letter: str, msg: str, bold: bool=False): def log(self, letter: str, msg: str, bold: bool=False):
colors = { 'E': 31, 'W': 33, 'I': 36 }; colors = {
'E': 31, # red: error
'W': 33, # yellow: warning
'N': 34, # blue: notice, a weaker warning (no email generated)
'I': 36, # cyan: info, backup.py script output
'O': 37, # white: regular output from borg
};
c = colors[letter] if letter in colors else 0 c = colors[letter] if letter in colors else 0
b = "" if bold else "\033[22m" b = "" if bold else "\033[22m"
sys.stderr.write(f"\033[1;{c}m{letter}:{b} {msg}\033[0m\n") sys.stdout.write(f"\033[1;{c}m{letter}:{b} {msg}\033[0m\n")
sys.stderr.flush() sys.stdout.flush()
self.logs.append((letter, msg)) self.logs.append((letter, msg))
def run(self, outfile: typing.IO[bytes]): def run(self, outfile: typing.IO[bytes]):
self.outfile = outfile self.outfile = outfile
for root in self.config.roots: for root in self.config.roots:
if root in self.root_seen: if root in self.root_seen:
self.log('I', f"ignoring root, already seen: {pstr(root)}") self.log('I', f"ignoring root, already seen: {b2s(root)}")
continue continue
try: try:
@ -140,13 +143,13 @@ class Backup:
if not stat.S_ISDIR(st.st_mode): if not stat.S_ISDIR(st.st_mode):
raise NotADirectoryError raise NotADirectoryError
except FileNotFoundError: except FileNotFoundError:
self.log('E', f"root does not exist: {pstr(root)}") self.log('E', f"root does not exist: {b2s(root)}")
continue continue
except NotADirectoryError: except NotADirectoryError:
self.log('E', f"root is not a directory: {pstr(root)}") self.log('E', f"root is not a directory: {b2s(root)}")
continue continue
self.log('I', f"processing root {pstr(root)}") self.log('I', f"processing root {b2s(root)}")
self.scan(root) self.scan(root)
def scan(self, path: bytes, parent_st: os.stat_result=None): def scan(self, path: bytes, parent_st: os.stat_result=None):
@ -199,7 +202,7 @@ class Backup:
force = self.config.match_re(self.config.unexclude, decorated_path) force = self.config.match_re(self.config.unexclude, decorated_path)
if exclude_reason and not force: if exclude_reason and not force:
self.log(exclude_reason[0], self.log(exclude_reason[0],
f"{exclude_reason[1]}: {pstr(path)}") f"{exclude_reason[1]}: {b2s(path)}")
return return
# Print path for Borg # Print path for Borg
@ -221,7 +224,7 @@ class Backup:
with open(path + b'/CACHEDIR.TAG', 'rb') as f: with open(path + b'/CACHEDIR.TAG', 'rb') as f:
if f.read(len(tag)) == tag: if f.read(len(tag)) == tag:
self.log( self.log(
'I', f"skipping, cache dir: {pstr(path)}") 'I', f"skipping, cache dir: {b2s(path)}")
return return
except: except:
pass pass
@ -235,7 +238,7 @@ class Backup:
IsADirectoryError, IsADirectoryError,
NotADirectoryError, NotADirectoryError,
PermissionError) as e: PermissionError) as e:
self.log('E', f"can't read {pstr(path)}: {str(e)}") self.log('E', f"can't read {b2s(path)}: {str(e)}")
return return
def run_borg(self, argv: typing.List[str], def run_borg(self, argv: typing.List[str],
@ -251,6 +254,9 @@ class Backup:
if borg.stdin is None: if borg.stdin is None:
raise Exception("no pipe") raise Exception("no pipe")
# Count warnings and errors from Borg, so we can interpret its
# error codes correctly (e.g. ignoring exit codes if warnings
# were all harmless).
borg_saw_warnings = 0 borg_saw_warnings = 0
borg_saw_errors = 0 borg_saw_errors = 0
@ -262,24 +268,30 @@ class Backup:
for line in fh: for line in fh:
try: try:
data = json.loads(line) data = json.loads(line)
if data['type'] == 'log_message': if data['type'] == 'log_message':
# Count warnings and errors, but ignore some.
changed_msg = "file changed while we backed it up" changed_msg = "file changed while we backed it up"
if data['levelname'] == 'WARNING': if data['levelname'] == 'WARNING':
prefix = "warning: " if changed_msg in data['message']:
if changed_msg not in data['message']: # harmless; don't count as a Borg warning
outlevel = 'N'
else:
borg_saw_warnings += 1 borg_saw_warnings += 1
outlevel = 'W'
output = "warning: "
elif data['levelname'] not in ('DEBUG', 'INFO'): elif data['levelname'] not in ('DEBUG', 'INFO'):
prefix = "error: "
borg_saw_errors += 1 borg_saw_errors += 1
outlevel = 'E'
output = "error: "
else: else:
prefix = "" outlevel = 'O'
output = ""
line = (prefix + data['message'] + '\n').encode() output += data['message']
elif (data['type'] == 'progress_message' elif (data['type'] == 'progress_message'
and 'message' in data): and 'message' in data):
line = (data['message'] + '\n').encode() outlevel = 'O'
output = data['message']
elif data['type'] == 'archive_progress': elif data['type'] == 'archive_progress':
now = time.time() now = time.time()
@ -287,12 +299,12 @@ class Backup:
last_progress = now last_progress = now
def size(short: str, full: str) -> str: def size(short: str, full: str) -> str:
return f" {short}={format_size(data[full])}" return f" {short}={format_size(data[full])}"
line = (f"progress:" + outlevel = 'O'
f" files={data['nfiles']}" + output = (f"progress:" +
size('orig', 'original_size') + f" files={data['nfiles']}" +
size('comp', 'compressed_size') + size('orig', 'original_size') +
size('dedup', 'deduplicated_size') + size('comp', 'compressed_size') +
"\n").encode() size('dedup', 'deduplicated_size'))
else: else:
continue continue
else: else:
@ -300,10 +312,9 @@ class Backup:
continue continue
except Exception as e: except Exception as e:
# on error, print raw line with exception # on error, print raw line with exception
line = f"[exception: {str(e)} ]".encode() + line outlevel = 'E'
sys.stdout.buffer.write(line) output = f"[exception: {str(e)}] " + b2s(line).rstrip()
sys.stdout.flush() self.log(outlevel, output)
self.captured_output.append(line)
fh.close() fh.close()
def _reader_thread(fh): def _reader_thread(fh):
try: try:
@ -322,8 +333,7 @@ class Backup:
time.sleep(1) time.sleep(1)
stdin_writer(borg.stdin) stdin_writer(borg.stdin)
except BrokenPipeError: except BrokenPipeError:
sys.stderr.write(f"broken pipe\n") self.log('E', "<broken pipe>")
sys.stderr.flush()
finally: finally:
try: try:
borg.stdin.close() borg.stdin.close()
@ -460,19 +470,24 @@ def main(argv: typing.List[str]):
if email and not args.dry_run: if email and not args.dry_run:
backup.log('I', f"sending error notification to {email}") backup.log('I', f"sending error notification to {email}")
# Show all of our warnings and errors. Use a ">" prefix def write_logs(title, only_include=None):
# so warnings and errors get highlighted by the mail reader. body = [ title ]
body = [ "Logs from backup.py:" ] for (letter, msg) in backup.logs:
for (letter, msg) in backup.logs: if only_include and letter not in only_include:
if letter == "E" or letter == "W": continue
prefix = ">" # Use a ":" prefix for warnings/errors/notices so that
else: # the mail reader highlights them.
prefix = " " if letter in "EWN":
body.append(f"{prefix}{letter}: {msg}") prefix = ":"
body_text = "\n".join(body).encode() else:
prefix = " "
body.append(f"{prefix}{letter}: {msg}")
return "\n".join(body).encode()
# Followed by borg output
body_text += b"\n\nBorg output:\n" + b"".join(backup.captured_output) body_text = write_logs("Logged errors and warnings:", "EWN")
body_text += "\n"
body_text += write_logs("All logs:")
# Subject summary # Subject summary
if errmsg and warnmsg: if errmsg and warnmsg:

View File

@ -278,6 +278,8 @@ ExecStart=${BORG_DIR}/backup.py
Nice=10 Nice=10
IOSchedulingClass=best-effort IOSchedulingClass=best-effort
IOSchedulingPriority=6 IOSchedulingPriority=6
Restart=on-failure
RestartSec=600
EOF EOF
log "Setting up systemd" log "Setting up systemd"