A flexible systemd unit template for using Restic to backup your data to multiple repositories. It was designed to be lightweight and customizable with systemd overrides. If you want to have notifications on backup failures you will need to setup Blazon (a simple systemd unit template for setting up OnFailure=
notifications). See the Notifications section in this README for more info.
Note: Make sure you have restic installed!
Automatically:
- Run
sudo mkdir /etc/restic
- Run
sudo ./install
Manually:
- Run
sudo mkdir /etc/restic
sudo cp mv units/* /etc/systemd/system/
- run
systemctl daemon-reload
Below are examples for setting up Local, AWS, and Google Cloud backups. Be sure to create a /etc/restic/exclude.conf
file and customize it to exclude files you do not want to backup. See the examples/
directory for inspiration.
Create /etc/restic/local.conf
and configure your local backups as you see fit. Since your configuration file might contain your RESTIC_PASSWORD
or other sensitive information you will want to chmod 0600 /etc/restic/local.conf
this file. Also make sure the path to RESTIC_REPOSITORY
exits and your user has permissions to write to it.
RESTIC_BACKUP_EXCLUDES="--exclude-file /etc/restic/exclude.conf --exclude-if-present .restic_exclude"
RESTIC_BACKUP_PATHS="/home/bingo"
RESTIC_RETENTION_POLICY="--keep-within-hourly 24h --keep-within-daily 7d --keep-within-weekly 1m --keep-within-monthly 1y --keep-within-yearly 7y"
RESTIC_PASSWORD=mysupersecretbackuppassword
RESTIC_REPOSITORY=/mnt/storage/backups/bluey
Setup an AWS S3 bucket with an IAM user and policy that has access to the bucket. See the Restic AWS documentation for more information. Then create /etc/restic/aws.conf
and add the below configuration:
AWS_ACCESS_KEY_ID="AAAAAAAAAAAAAAAAAAAA"
AWS_DEFAULT_REGION="us-east-2"
AWS_SECRET_ACCESS_KEY="BBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBB"
RESTIC_BACKUP_EXCLUDES="--exclude-file /etc/restic/exclude.conf --exclude-if-present .restic_exclude"
RESTIC_BACKUP_PATHS="/home/bingo"
RESTIC_PASSWORD="anothersuperscretpassword"
RESTIC_REPOSITORY="s3:https://s3.amazonaws.com/mycool-restic-backups-s3-bucket/bluey"
RESTIC_RETENTION_POLICY="--keep-within-hourly 24h --keep-within-daily 7d --keep-within-weekly 1m --keep-within-monthly 1y --keep-within-yearly 7y"
Since your /etc/restic/aws.conf
configuration file might contain your RESTIC_PASSWORD
or other sensitive information you will want to chmod 0600 /etc/restic/aws.conf
this file.
Create a Google Cloud Storage bucket and then create a new Service Account that has access to that bucket. See the Restic GCP documentation for more information. Then create /etc/restic/gcp.conf
and add the below configuration:
GOOGLE_PROJECT_ID=123123123123
GOOGLE_APPLICATION_CREDENTIALS=/etc/restic/gcs-secret-restic-key.json
RESTIC_BACKUP_EXCLUDES="--exclude-file /etc/restic/exclude.conf --exclude-if-present .restic_exclude"
RESTIC_BACKUP_PATHS="/home/bingo"
RESTIC_PASSWORD="themostsecretpasswordever"
RESTIC_REPOSITORY="gs:mycool-restic-backups-gcs-bucket:/bluey"
RESTIC_RETENTION_POLICY="--keep-within-hourly 24h --keep-within-daily 7d --keep-within-weekly 1m --keep-within-monthly 1y --keep-within-yearly 7y"
Since your /etc/restic/gcp.conf
configuration file and your /etc/restic/gs-secret-restic-key.json
Service Account JSON file might contain your RESTIC_PASSWORD
or other sensitive information you will want to chmod 0600 /etc/restic/aws.conf /etc/restic/gs-secret-restic-key.json
these two files.
sudo ./bin/restic-wrapper /etc/restic/local.conf init
sudo ./bin/restic-wrapper /etc/restic/aws.conf init
sudo ./bin/restic-wrapper /etc/restic/gcp.conf init
sudo systemctl enable --now [email protected]
sudo systemctl enable --now [email protected]
sudo systemctl enable --now [email protected]
You can use ./bin/restic-wrapper /path/to/your.config
to run manual commands on your repositories. Be sure you have permissions to read the config file as your user.
Run a check
sudo ./bin/restic-wrapper /etc/restic/local.conf check
using temporary cache in /tmp/restic-check-cache-2107648289
repository 63c2e3d8 opened successfully, password is correct
created new cache in /tmp/restic-check-cache-2107648289
create exclusive lock for repository
load indexes
check all packs
check snapshots, trees and blobs
[0:08] 100.00% 14 / 14 snapshots
no errors were found
Run a dry run forget
sudo ./bin/restic-wrapper /etc/restic/local.conf forget --dry-run --verbose --tag local --group-by "paths,tags" --keep-within-hourly 24h --keep-within-daily 7d --keep-within-weekly 1m --keep-within-monthly 1y --keep-within-yearly 7y
repository 63c2e3d8 opened successfully, password is correct
Applying Policy: keep hourly snapshots within 24h, daily snapshots within 7d, weekly snapshots within 1m, monthly snapshots within 1y, yearly snapshots within 7y
snapshots for (tags [local], paths [/home/bingo]):
keep 14 snapshots:
ID Time Host Tags Reasons Paths
-------------------------------------------------------------------------------------
e2f01b0d 2022-06-16 00:37:45 bluey local hourly within 24h /home/bingo
e0af1b23 2022-06-16 01:28:03 bluey local hourly within 24h /home/bingo
f4fa0b64 2022-06-16 02:37:18 bluey local hourly within 24h /home/bingo
8e05e9a9 2022-06-16 03:50:34 bluey local hourly within 24h /home/bingo
d2b2f067 2022-06-16 04:50:52 bluey local hourly within 24h /home/bingo
65ff1396 2022-06-16 06:00:22 bluey local hourly within 24h /home/bingo
c205a7d1 2022-06-16 07:07:37 bluey local hourly within 24h /home/bingo
c5630498 2022-06-16 08:17:23 bluey local hourly within 24h /home/bingo
62a05617 2022-06-16 09:23:10 bluey local hourly within 24h /home/bingo
e155b037 2022-06-16 10:37:52 bluey local hourly within 24h /home/bingo
0bd39494 2022-06-16 11:46:23 bluey local hourly within 24h /home/bingo
6bfccf17 2022-06-16 13:00:17 bluey local hourly within 24h /home/bingo
73de7644 2022-06-16 14:07:52 bluey local hourly within 24h /home/bingo
4421dcd0 2022-06-16 15:03:47 bluey local hourly within 24h /home/bingo
daily within 7d
weekly within 1m
monthly within 1y
yearly within 7y
-------------------------------------------------------------------------------------
14 snapshots
We will be setting up local desktop and Slack notifications in this example. You need to install the Blazon systemd units and you will need to create a Slack Incoming Webhook on your Slack org that posts messages to a channel. See the repository for more information on using Blazon for configuring this systemd template unit.
Configure Blazon for local and AWS backups
Create systemd overrides for each blazon@restic@
and service:
systemctl edit blazon@[email protected]
systemctl edit blazon@[email protected]
and add the following to each:
[Service]
Environment=BLAZON_NOTIFY_DESKTOP=true
Environment=BLAZON_NOTIFY_SLACK=true
Environment=BLAZON_SLACK_WEBHOOK_URL="https://hooks.slack.com/services/AAAAAAAAAAA/AAAAAAAAAAA/HHHHHHHHHHHHHHHHHHHHHHHHH"
Create overrides for each [email protected]
:
systemctl edit [email protected]
systemctl edit [email protected]
and add the following to each:
[Unit]
OnFailure=blazon@%N.service
Now you should get desktop notifications and a Slack message if any of your backups units fail!
You may see failures on your backups like the one below.
[root@sw-0608 ~]# journalctl --since "10 minutes ago" -u [email protected]
Jul 13 10:32:32 sw-0608 systemd[1]: Starting [email protected] - Restic backup service...
Jul 13 10:32:32 sw-0608 restic[1369163]: open repository
Jul 13 10:32:33 sw-0608 restic[1369163]: lock repository
Jul 13 10:32:33 sw-0608 restic[1369163]: load index files
Jul 13 10:32:34 sw-0608 restic[1369163]: using parent snapshot 44eafc84
Jul 13 10:32:34 sw-0608 restic[1369163]: start scan on [/home/jdoss]
Jul 13 10:32:34 sw-0608 restic[1369163]: start backup on [/home/jdoss]
Jul 13 10:32:58 sw-0608 restic[1369163]: error: lstat /home/jdoss/Documents/projects/iotpod/victoriametrics/data/data/small/2022_07/143344_4355_20220713152308.202_20220713152853.292_1700E4A397107AD4: no such file >
Jul 13 10:32:58 sw-0608 restic[1369163]: error: lstat /home/jdoss/Documents/projects/iotpod/victoriametrics/data/data/small/2022_07/146308_4355_20220713145838.202_20220713150413.292_1700E4A3971079A0: no such file >
Jul 13 10:32:58 sw-0608 restic[1369163]: error: lstat /home/jdoss/Documents/projects/iotpod/victoriametrics/data/data/small/2022_07/146344_4355_20220713145138.202_20220713145713.292_1700E4A397107946: no such file >
Jul 13 10:32:58 sw-0608 restic[1369163]: error: lstat /home/jdoss/Documents/projects/iotpod/victoriametrics/data/data/small/2022_07/172967_4355_20220713151608.202_20220713152323.292_1700E4A397107A8D: no such file >
Jul 13 10:32:58 sw-0608 restic[1369163]: error: lstat /home/jdoss/Documents/projects/iotpod/victoriametrics/data/data/small/2022_07/173003_4355_20220713150908.202_20220713151643.292_1700E4A397107A3B: no such file >
Jul 13 10:32:58 sw-0608 restic[1369163]: error: lstat /home/jdoss/Documents/projects/iotpod/victoriametrics/data/data/small/2022_07/180628_4355_20220713144358.202_20220713145133.292_1700E4A3971078FE: no such file >
Jul 13 10:32:58 sw-0608 restic[1369163]: error: lstat /home/jdoss/Documents/projects/iotpod/victoriametrics/data/data/small/2022_07/26623_4355_20220713153108.202_20220713153208.202_1700E4A397107AFC: no such file o>
Jul 13 10:32:58 sw-0608 restic[1369163]: error: lstat /home/jdoss/Documents/projects/iotpod/victoriametrics/data/data/small/2022_07/27296_4355_20220713152758.202_20220713152958.202_1700E4A397107AE2: no such file o>
Jul 13 10:32:58 sw-0608 restic[1369163]: error: lstat /home/jdoss/Documents/projects/iotpod/victoriametrics/data/data/small/2022_07/29596_4355_20220713153002.736_20220713153103.292_1700E4A397107AEF: no such file o>
Jul 13 10:32:58 sw-0608 restic[1369163]: error: lstat /home/jdoss/Documents/projects/iotpod/victoriametrics/data/data/small/2022_07/3646_3646_20220713153212.736_20220713153213.292_1700E4A397107AFD: no such file or>
Jul 13 10:32:58 sw-0608 restic[1369163]: error: lstat /home/jdoss/Documents/projects/iotpod/victoriametrics/data/data/small/2022_07/3646_3646_20220713153222.736_20220713153223.292_1700E4A397107AFF: no such file or>
Jul 13 10:32:58 sw-0608 restic[1369163]: error: lstat /home/jdoss/Documents/projects/iotpod/victoriametrics/data/data/small/2022_07/3655_3655_20220713153235.218_20220713153243.292_1700E4A397107B03: no such file or>
Jul 13 10:33:52 sw-0608 restic[1369163]: scan finished in 79.638s: 1167623 files, 143.294 GiB
Jul 13 10:34:59 sw-0608 restic[1369163]: Files: 134 new, 408 changed, 1167037 unmodified
Jul 13 10:34:59 sw-0608 restic[1369163]: Dirs: 50 new, 236 changed, 186695 unmodified
Jul 13 10:34:59 sw-0608 restic[1369163]: Data Blobs: 775 new
Jul 13 10:34:59 sw-0608 restic[1369163]: Tree Blobs: 271 new
Jul 13 10:34:59 sw-0608 restic[1369163]: Added to the repo: 329.317 MiB
Jul 13 10:34:59 sw-0608 restic[1369163]: processed 1167579 files, 143.293 GiB in 2:26
Jul 13 10:34:59 sw-0608 restic[1369163]: snapshot db22c2c7 saved
Jul 13 10:34:59 sw-0608 restic[1369163]: Warning: at least one source file could not be read
Jul 13 10:34:59 sw-0608 systemd[1]: [email protected]: Main process exited, code=exited, status=3/NOTIMPLEMENTED
Jul 13 10:34:59 sw-0608 systemd[1]: [email protected]: Failed with result 'exit-code'.
Jul 13 10:34:59 sw-0608 systemd[1]: Failed to start [email protected] - Restic backup service.
Jul 13 10:34:59 sw-0608 systemd[1]: [email protected]: Triggering OnFailure= dependencies.
This is when the restic backup run exits with an exit code of 3/NOTIMPLEMENTED
when it scans your files at the start but one or more files are removed by the time it gets to them to back them up. You can ignore this exit code by adding:
[Service]
SuccessExitStatus=3
as a systemd override to your backup unit. This will still log the errors about the files not being able to be backed up, but it will not fail the systemd unit.
MIT License
Copyright (c) 2022 QuickVM
Permission is hereby granted, free of charge, to any person obtaining a copy of this software and associated documentation files (the "Software"), to deal in the Software without restriction, including without limitation the rights to use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of the Software, and to permit persons to whom the Software is furnished to do so, subject to the following conditions:
The above copyright notice and this permission notice shall be included in all copies or substantial portions of the Software.
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.