My backup scripts and tools
Go to file
2021-10-17 19:31:36 -04:00
.gitignore misc: ignore .venv dir 2021-10-16 09:37:04 -04:00
backup.py backup: tweak types for python 3.7 compatibility 2021-10-17 19:31:36 -04:00
Borg.bin borg: update binary to fix upstream bug 6009 2021-10-16 09:37:04 -04:00
borg.sh borg.sh: fix ssh option for read-write mode 2021-10-17 08:56:53 -04:00
config.yaml config: remove /efi, it probably doesn't exist 2021-10-15 23:21:28 -04:00
initial-setup.sh setup: pick host-dependent start time 2021-10-17 09:26:41 -04:00
Makefile makefile: reload systemd unit files after rebase 2021-10-16 23:47:34 -04:00
notify.sh notify: fix notify.sh to work with server side; adjust text 2021-10-16 09:37:03 -04:00
Pipfile Implement filesystem scanning with configurable filters 2021-10-11 16:50:08 -04:00
Pipfile.lock Implement filesystem scanning with configurable filters 2021-10-11 16:50:08 -04:00
prune.sh prune: use new vars.sh 2021-10-16 09:37:03 -04:00
README.md setup: allow hostname to be overridden 2021-10-16 18:48:43 -04:00

Initial setup

Run on client:

sudo git clone https://git.jim.sh/jim/borg-setup.git /opt/borg
sudo /opt/borg/initial-setup.sh

Customize /opt/borg/config.yaml as desired.

Cheat sheet

After setup, the copy of this file on the client will have the variables in this section filled in automatically

Configuration

Hostname: ${HOSTNAME}
Base directory: ${BORG_DIR}
Destination: ${BACKUP_USER}@${BACKUP_HOST}
Repository: ${BACKUP_REPO}

Commands

See when next backup is scheduled:

systemctl list-timers borg-backup.timer

See status of most recent backup:

systemctl status --full --lines 999999 --no-pager --all borg-backup

Watch log:

journalctl --all --follow --unit borg-backup

Start backup now:

sudo systemctl start borg-backup

Interrupt backup in progress:

sudo systemctl stop borg-backup

Show backups and related info:

sudo ${BORG_DIR}/borg.sh info
sudo ${BORG_DIR}/borg.sh list

Run Borg using the read-write SSH key:

sudo ${BORG_DIR}/borg.sh --rw list

Mount and look at files:

mkdir mnt
sudo ${BORG_DIR}/borg.sh mount :: mnt
sudo -s # to explore as root
sudo umount mnt

Prune old backups. Only run if sure local system was never compromised, as object deletion could have been queued during append-only operations. Requires SSH key password from bitwarden.

sudo ${BORG_DIR}/prune.sh

Design

  • On server, we have a separate user account "jim-backups". Password for this account is in bitwarden in the "Backups" folder, under ssh backup.jim.sh.

  • Repository keys are repokeys, which get stored on the server, inside the repo. Passphrases are stored:

    • on clients (in /opt/borg/passphrase, for making backups)
    • in bitwarden (under borg <hostname>, user repo key)
  • Each client has two SSH keys for connecting to the server:

    • /opt/borg/ssh/id_ecdsa_appendonly
      • configured on server for append-only operation
      • used for making backups
      • no password
    • /opt/borg/ssh/id_ecdsa
      • configured on server for read-write operation
      • used for manual recovery, management, pruning
      • password in bitwarden (under borg <hostname>, user read-write ssh key)
  • Pruning requires the password and is a manual operation, and should only be run when the client has not been compromised.

    sudo /opt/borg/prune.sh
    
  • Systemd timers start daily backups:

    /etc/systemd/system/borg-backup.service -> /opt/borg/borg-backup.service
    /etc/systemd/system/borg-backup.timer -> /opt/borg/borg-backup.timer
    
  • Backup script /opt/borg/backup.py uses configuration in /opt/borg/config.yaml to generate our own list of files, excluding anything that's too large by default. This requires borg 1.2.0b1 or newer.

Notes

Building Borg.bin binary from git

git clone https://github.com/borgbackup/borg.git
cd borg
virtualenv --python=python3 borg-env
source borg-env/bin/activate
pip install -r requirements.d/development.txt
pip install pyinstaller
pip install llfuse
pip install -e .[llfuse]
pyinstaller --clean --noconfirm scripts/borg.exe.spec

Then see dist/borg.exe. Confirm the version with dist/borg.exe --version.

Note: This uses the deprecated llfuse instead of the newer pyfuse3. pyfuse3 doesn't work because, at minimum, it pulls in trio which requires ssl which is explicitly excluded by scripts/borg.exe.spec.