add 3-2-1 backup pipeline: weekly PVC file copy, NFS mirror, pfsense, offsite sync
- weekly-backup.sh: mounts LVM thin snapshots ro, rsyncs files to /mnt/backup/pvc-data
with --link-dest versioning (4 weeks). Also mirrors NFS backup dirs from TrueNAS,
backs up pfsense (config.xml + full tar), PVE host config, and prunes >7d snapshots.
- offsite-sync-backup.sh: rsync --files-from manifest to Synology (no full dir walk).
Monthly full --delete sync on 1st Sunday. After=weekly-backup.service dependency.
- lvm-pvc-snapshot.timer: changed to daily 03:00 (was 2x daily)
- Prometheus alerts: WeeklyBackupStale, WeeklyBackupFailing, PfsenseBackupStale,
OffsiteBackupSyncStale, BackupDiskFull. LVMSnapshotStale threshold 24h→48h.
2026-04-06 14:53:28 +03:00
|
|
|
#!/usr/bin/env bash
|
2026-04-13 18:06:20 +00:00
|
|
|
# offsite-sync-backup — Sync backups to Synology NAS
|
add 3-2-1 backup pipeline: weekly PVC file copy, NFS mirror, pfsense, offsite sync
- weekly-backup.sh: mounts LVM thin snapshots ro, rsyncs files to /mnt/backup/pvc-data
with --link-dest versioning (4 weeks). Also mirrors NFS backup dirs from TrueNAS,
backs up pfsense (config.xml + full tar), PVE host config, and prunes >7d snapshots.
- offsite-sync-backup.sh: rsync --files-from manifest to Synology (no full dir walk).
Monthly full --delete sync on 1st Sunday. After=weekly-backup.service dependency.
- lvm-pvc-snapshot.timer: changed to daily 03:00 (was 2x daily)
- Prometheus alerts: WeeklyBackupStale, WeeklyBackupFailing, PfsenseBackupStale,
OffsiteBackupSyncStale, BackupDiskFull. LVMSnapshotStale threshold 24h→48h.
2026-04-06 14:53:28 +03:00
|
|
|
# Deploy to PVE host at /usr/local/bin/offsite-sync-backup
|
2026-04-13 18:37:04 +00:00
|
|
|
# Schedule: Daily 06:00 via systemd timer (After=daily-backup.service)
|
2026-04-13 18:06:20 +00:00
|
|
|
#
|
|
|
|
|
# Two sync paths:
|
|
|
|
|
# Step 1: sda (/mnt/backup) → Synology pve-backup/ (PVC snapshots, pfsense, pve-config, sqlite)
|
|
|
|
|
# Step 2: NFS (/srv/nfs, /srv/nfs-ssd) → Synology nfs/, nfs-ssd/ (inotify change-tracked)
|
add 3-2-1 backup pipeline: weekly PVC file copy, NFS mirror, pfsense, offsite sync
- weekly-backup.sh: mounts LVM thin snapshots ro, rsyncs files to /mnt/backup/pvc-data
with --link-dest versioning (4 weeks). Also mirrors NFS backup dirs from TrueNAS,
backs up pfsense (config.xml + full tar), PVE host config, and prunes >7d snapshots.
- offsite-sync-backup.sh: rsync --files-from manifest to Synology (no full dir walk).
Monthly full --delete sync on 1st Sunday. After=weekly-backup.service dependency.
- lvm-pvc-snapshot.timer: changed to daily 03:00 (was 2x daily)
- Prometheus alerts: WeeklyBackupStale, WeeklyBackupFailing, PfsenseBackupStale,
OffsiteBackupSyncStale, BackupDiskFull. LVMSnapshotStale threshold 24h→48h.
2026-04-06 14:53:28 +03:00
|
|
|
set -euo pipefail
|
|
|
|
|
|
|
|
|
|
# --- Configuration ---
|
|
|
|
|
BACKUP_ROOT="/mnt/backup"
|
2026-04-13 18:06:20 +00:00
|
|
|
SYNOLOGY="Administrator@192.168.1.13"
|
|
|
|
|
PVE_BACKUP_DEST="${SYNOLOGY}:/volume1/Backup/Viki/pve-backup"
|
|
|
|
|
NFS_DEST="${SYNOLOGY}:/volume1/Backup/Viki/nfs"
|
|
|
|
|
NFS_SSD_DEST="${SYNOLOGY}:/volume1/Backup/Viki/nfs-ssd"
|
add 3-2-1 backup pipeline: weekly PVC file copy, NFS mirror, pfsense, offsite sync
- weekly-backup.sh: mounts LVM thin snapshots ro, rsyncs files to /mnt/backup/pvc-data
with --link-dest versioning (4 weeks). Also mirrors NFS backup dirs from TrueNAS,
backs up pfsense (config.xml + full tar), PVE host config, and prunes >7d snapshots.
- offsite-sync-backup.sh: rsync --files-from manifest to Synology (no full dir walk).
Monthly full --delete sync on 1st Sunday. After=weekly-backup.service dependency.
- lvm-pvc-snapshot.timer: changed to daily 03:00 (was 2x daily)
- Prometheus alerts: WeeklyBackupStale, WeeklyBackupFailing, PfsenseBackupStale,
OffsiteBackupSyncStale, BackupDiskFull. LVMSnapshotStale threshold 24h→48h.
2026-04-06 14:53:28 +03:00
|
|
|
MANIFEST="${BACKUP_ROOT}/.changed-files"
|
2026-04-13 18:06:20 +00:00
|
|
|
NFS_CHANGE_LOG="${BACKUP_ROOT}/.nfs-changes.log"
|
add 3-2-1 backup pipeline: weekly PVC file copy, NFS mirror, pfsense, offsite sync
- weekly-backup.sh: mounts LVM thin snapshots ro, rsyncs files to /mnt/backup/pvc-data
with --link-dest versioning (4 weeks). Also mirrors NFS backup dirs from TrueNAS,
backs up pfsense (config.xml + full tar), PVE host config, and prunes >7d snapshots.
- offsite-sync-backup.sh: rsync --files-from manifest to Synology (no full dir walk).
Monthly full --delete sync on 1st Sunday. After=weekly-backup.service dependency.
- lvm-pvc-snapshot.timer: changed to daily 03:00 (was 2x daily)
- Prometheus alerts: WeeklyBackupStale, WeeklyBackupFailing, PfsenseBackupStale,
OffsiteBackupSyncStale, BackupDiskFull. LVMSnapshotStale threshold 24h→48h.
2026-04-06 14:53:28 +03:00
|
|
|
PUSHGATEWAY="${OFFSITE_SYNC_PUSHGATEWAY:-http://10.0.20.100:30091}"
|
|
|
|
|
PUSHGATEWAY_JOB="offsite-backup-sync"
|
|
|
|
|
LOCKFILE="/run/offsite-sync-backup.lock"
|
|
|
|
|
|
|
|
|
|
# --- Logging ---
|
|
|
|
|
log() { echo "[$(date '+%Y-%m-%d %H:%M:%S')] $*"; }
|
|
|
|
|
warn() { log "WARN: $*" >&2; }
|
|
|
|
|
|
|
|
|
|
# --- Locking ---
|
|
|
|
|
cleanup() { rm -f "${LOCKFILE}"; }
|
|
|
|
|
trap cleanup EXIT
|
|
|
|
|
if ! ( set -o noclobber; echo $$ > "${LOCKFILE}" ) 2>/dev/null; then
|
|
|
|
|
log "FATAL: Another instance running"; exit 1
|
|
|
|
|
fi
|
|
|
|
|
|
|
|
|
|
# --- Main ---
|
|
|
|
|
log "=== Offsite sync starting ==="
|
|
|
|
|
STATUS=0
|
|
|
|
|
|
|
|
|
|
if ! mountpoint -q "${BACKUP_ROOT}"; then
|
|
|
|
|
log "FATAL: ${BACKUP_ROOT} is not mounted"; exit 1
|
|
|
|
|
fi
|
|
|
|
|
|
2026-04-13 18:06:20 +00:00
|
|
|
if ! timeout 10 ssh -o BatchMode=yes -o ConnectTimeout=5 "${SYNOLOGY}" true 2>/dev/null; then
|
|
|
|
|
log "FATAL: Cannot SSH to Synology"
|
add 3-2-1 backup pipeline: weekly PVC file copy, NFS mirror, pfsense, offsite sync
- weekly-backup.sh: mounts LVM thin snapshots ro, rsyncs files to /mnt/backup/pvc-data
with --link-dest versioning (4 weeks). Also mirrors NFS backup dirs from TrueNAS,
backs up pfsense (config.xml + full tar), PVE host config, and prunes >7d snapshots.
- offsite-sync-backup.sh: rsync --files-from manifest to Synology (no full dir walk).
Monthly full --delete sync on 1st Sunday. After=weekly-backup.service dependency.
- lvm-pvc-snapshot.timer: changed to daily 03:00 (was 2x daily)
- Prometheus alerts: WeeklyBackupStale, WeeklyBackupFailing, PfsenseBackupStale,
OffsiteBackupSyncStale, BackupDiskFull. LVMSnapshotStale threshold 24h→48h.
2026-04-06 14:53:28 +03:00
|
|
|
echo "backup_last_success_timestamp 0" | \
|
|
|
|
|
curl -s --connect-timeout 5 --max-time 10 --data-binary @- \
|
|
|
|
|
"${PUSHGATEWAY}/metrics/job/${PUSHGATEWAY_JOB}" 2>/dev/null || true
|
|
|
|
|
exit 1
|
|
|
|
|
fi
|
|
|
|
|
|
|
|
|
|
DAY_OF_MONTH=$(date +%d)
|
|
|
|
|
|
2026-04-13 18:06:20 +00:00
|
|
|
# ============================================================
|
|
|
|
|
# STEP 1: sda → Synology pve-backup/ (PVC snapshots, pfsense, pve-config)
|
|
|
|
|
# ============================================================
|
|
|
|
|
log "--- Step 1: sda → Synology pve-backup/ ---"
|
|
|
|
|
|
add 3-2-1 backup pipeline: weekly PVC file copy, NFS mirror, pfsense, offsite sync
- weekly-backup.sh: mounts LVM thin snapshots ro, rsyncs files to /mnt/backup/pvc-data
with --link-dest versioning (4 weeks). Also mirrors NFS backup dirs from TrueNAS,
backs up pfsense (config.xml + full tar), PVE host config, and prunes >7d snapshots.
- offsite-sync-backup.sh: rsync --files-from manifest to Synology (no full dir walk).
Monthly full --delete sync on 1st Sunday. After=weekly-backup.service dependency.
- lvm-pvc-snapshot.timer: changed to daily 03:00 (was 2x daily)
- Prometheus alerts: WeeklyBackupStale, WeeklyBackupFailing, PfsenseBackupStale,
OffsiteBackupSyncStale, BackupDiskFull. LVMSnapshotStale threshold 24h→48h.
2026-04-06 14:53:28 +03:00
|
|
|
if [ "${DAY_OF_MONTH}" -le 7 ]; then
|
|
|
|
|
log "Monthly full sync (1st Sunday)..."
|
2026-04-06 16:01:42 +03:00
|
|
|
rsync -rltz --delete --chmod=Du=rwx,Dgo=rx,Fu=rw,Fog=r \
|
add 3-2-1 backup pipeline: weekly PVC file copy, NFS mirror, pfsense, offsite sync
- weekly-backup.sh: mounts LVM thin snapshots ro, rsyncs files to /mnt/backup/pvc-data
with --link-dest versioning (4 weeks). Also mirrors NFS backup dirs from TrueNAS,
backs up pfsense (config.xml + full tar), PVE host config, and prunes >7d snapshots.
- offsite-sync-backup.sh: rsync --files-from manifest to Synology (no full dir walk).
Monthly full --delete sync on 1st Sunday. After=weekly-backup.service dependency.
- lvm-pvc-snapshot.timer: changed to daily 03:00 (was 2x daily)
- Prometheus alerts: WeeklyBackupStale, WeeklyBackupFailing, PfsenseBackupStale,
OffsiteBackupSyncStale, BackupDiskFull. LVMSnapshotStale threshold 24h→48h.
2026-04-06 14:53:28 +03:00
|
|
|
--exclude='.changed-files' \
|
|
|
|
|
--exclude='.last-offsite-sync' \
|
|
|
|
|
--exclude='.lv-pvc-mapping.json' \
|
2026-04-13 18:06:20 +00:00
|
|
|
--exclude='.nfs-changes.log' \
|
|
|
|
|
"${BACKUP_ROOT}/" "${PVE_BACKUP_DEST}/" 2>&1 || STATUS=1
|
add 3-2-1 backup pipeline: weekly PVC file copy, NFS mirror, pfsense, offsite sync
- weekly-backup.sh: mounts LVM thin snapshots ro, rsyncs files to /mnt/backup/pvc-data
with --link-dest versioning (4 weeks). Also mirrors NFS backup dirs from TrueNAS,
backs up pfsense (config.xml + full tar), PVE host config, and prunes >7d snapshots.
- offsite-sync-backup.sh: rsync --files-from manifest to Synology (no full dir walk).
Monthly full --delete sync on 1st Sunday. After=weekly-backup.service dependency.
- lvm-pvc-snapshot.timer: changed to daily 03:00 (was 2x daily)
- Prometheus alerts: WeeklyBackupStale, WeeklyBackupFailing, PfsenseBackupStale,
OffsiteBackupSyncStale, BackupDiskFull. LVMSnapshotStale threshold 24h→48h.
2026-04-06 14:53:28 +03:00
|
|
|
elif [ -s "${MANIFEST}" ]; then
|
|
|
|
|
MANIFEST_LINES=$(wc -l < "${MANIFEST}")
|
|
|
|
|
log "Incremental sync (${MANIFEST_LINES} files from manifest)..."
|
2026-04-13 18:06:20 +00:00
|
|
|
rsync -rltz --chmod=Du=rwx,Dgo=rx,Fu=rw,Fog=r --files-from="${MANIFEST}" \
|
|
|
|
|
"${BACKUP_ROOT}/" "${PVE_BACKUP_DEST}/" 2>&1 || STATUS=1
|
add 3-2-1 backup pipeline: weekly PVC file copy, NFS mirror, pfsense, offsite sync
- weekly-backup.sh: mounts LVM thin snapshots ro, rsyncs files to /mnt/backup/pvc-data
with --link-dest versioning (4 weeks). Also mirrors NFS backup dirs from TrueNAS,
backs up pfsense (config.xml + full tar), PVE host config, and prunes >7d snapshots.
- offsite-sync-backup.sh: rsync --files-from manifest to Synology (no full dir walk).
Monthly full --delete sync on 1st Sunday. After=weekly-backup.service dependency.
- lvm-pvc-snapshot.timer: changed to daily 03:00 (was 2x daily)
- Prometheus alerts: WeeklyBackupStale, WeeklyBackupFailing, PfsenseBackupStale,
OffsiteBackupSyncStale, BackupDiskFull. LVMSnapshotStale threshold 24h→48h.
2026-04-06 14:53:28 +03:00
|
|
|
else
|
|
|
|
|
log "No changed files in manifest, nothing to sync"
|
|
|
|
|
fi
|
|
|
|
|
|
2026-04-13 15:47:56 +00:00
|
|
|
# ============================================================
|
2026-04-13 18:06:20 +00:00
|
|
|
# STEP 2: NFS → Synology nfs/ + nfs-ssd/ (inotify change-tracked)
|
2026-04-13 15:47:56 +00:00
|
|
|
# ============================================================
|
2026-04-13 18:06:20 +00:00
|
|
|
log "--- Step 2: NFS → Synology (change-tracked) ---"
|
|
|
|
|
|
|
|
|
|
if [ "${DAY_OF_MONTH}" -le 7 ]; then
|
|
|
|
|
# Monthly: full sync with --delete for cleanup
|
|
|
|
|
log "Monthly full NFS sync..."
|
|
|
|
|
rsync -rltz --delete /srv/nfs/ "${NFS_DEST}/" 2>&1 \
|
|
|
|
|
&& log " OK: nfs/ full sync" || { warn "nfs/ full sync failed"; STATUS=1; }
|
|
|
|
|
rsync -rltz --delete /srv/nfs-ssd/ "${NFS_SSD_DEST}/" 2>&1 \
|
|
|
|
|
&& log " OK: nfs-ssd/ full sync" || { warn "nfs-ssd/ full sync failed"; STATUS=1; }
|
|
|
|
|
> "${NFS_CHANGE_LOG}"
|
|
|
|
|
elif [ -s "${NFS_CHANGE_LOG}" ]; then
|
|
|
|
|
# Incremental: only sync files logged by inotifywait
|
|
|
|
|
sort -u "${NFS_CHANGE_LOG}" > /tmp/nfs-changes-deduped
|
|
|
|
|
|
|
|
|
|
# HDD NFS
|
|
|
|
|
grep '^/srv/nfs/' /tmp/nfs-changes-deduped | \
|
|
|
|
|
while IFS= read -r f; do [ -f "$f" ] && echo "${f#/srv/nfs/}"; done \
|
|
|
|
|
> /tmp/sync-nfs.list 2>/dev/null
|
|
|
|
|
NFS_COUNT=$(wc -l < /tmp/sync-nfs.list 2>/dev/null || echo 0)
|
|
|
|
|
if [ "${NFS_COUNT:-0}" -gt 0 ]; then
|
|
|
|
|
rsync -rltz --files-from=/tmp/sync-nfs.list /srv/nfs/ "${NFS_DEST}/" 2>&1 \
|
|
|
|
|
&& log " OK: nfs/ (${NFS_COUNT} files)" \
|
|
|
|
|
|| { warn "nfs/ incremental failed"; STATUS=1; }
|
2026-04-13 15:47:56 +00:00
|
|
|
fi
|
2026-04-13 18:06:20 +00:00
|
|
|
|
|
|
|
|
# SSD NFS
|
|
|
|
|
grep '^/srv/nfs-ssd/' /tmp/nfs-changes-deduped | \
|
|
|
|
|
while IFS= read -r f; do [ -f "$f" ] && echo "${f#/srv/nfs-ssd/}"; done \
|
|
|
|
|
> /tmp/sync-nfs-ssd.list 2>/dev/null || true
|
|
|
|
|
SSD_COUNT=$(wc -l < /tmp/sync-nfs-ssd.list 2>/dev/null || echo 0)
|
|
|
|
|
if [ "${SSD_COUNT:-0}" -gt 0 ]; then
|
|
|
|
|
rsync -rltz --files-from=/tmp/sync-nfs-ssd.list /srv/nfs-ssd/ "${NFS_SSD_DEST}/" 2>&1 \
|
|
|
|
|
&& log " OK: nfs-ssd/ (${SSD_COUNT} files)" \
|
|
|
|
|
|| { warn "nfs-ssd/ incremental failed"; STATUS=1; }
|
2026-04-13 15:47:56 +00:00
|
|
|
fi
|
2026-04-13 18:06:20 +00:00
|
|
|
|
|
|
|
|
TOTAL=$(wc -l < /tmp/nfs-changes-deduped)
|
|
|
|
|
log " Processed ${TOTAL} change events (${NFS_COUNT} nfs + ${SSD_COUNT} nfs-ssd files synced)"
|
|
|
|
|
> "${NFS_CHANGE_LOG}"
|
|
|
|
|
rm -f /tmp/nfs-changes-deduped /tmp/sync-nfs.list /tmp/sync-nfs-ssd.list
|
|
|
|
|
else
|
|
|
|
|
log " No NFS changes to sync"
|
|
|
|
|
fi
|
2026-04-13 15:47:56 +00:00
|
|
|
|
|
|
|
|
# ============================================================
|
|
|
|
|
# Finish
|
|
|
|
|
# ============================================================
|
add 3-2-1 backup pipeline: weekly PVC file copy, NFS mirror, pfsense, offsite sync
- weekly-backup.sh: mounts LVM thin snapshots ro, rsyncs files to /mnt/backup/pvc-data
with --link-dest versioning (4 weeks). Also mirrors NFS backup dirs from TrueNAS,
backs up pfsense (config.xml + full tar), PVE host config, and prunes >7d snapshots.
- offsite-sync-backup.sh: rsync --files-from manifest to Synology (no full dir walk).
Monthly full --delete sync on 1st Sunday. After=weekly-backup.service dependency.
- lvm-pvc-snapshot.timer: changed to daily 03:00 (was 2x daily)
- Prometheus alerts: WeeklyBackupStale, WeeklyBackupFailing, PfsenseBackupStale,
OffsiteBackupSyncStale, BackupDiskFull. LVMSnapshotStale threshold 24h→48h.
2026-04-06 14:53:28 +03:00
|
|
|
if [ "${STATUS}" -eq 0 ]; then
|
|
|
|
|
touch "${BACKUP_ROOT}/.last-offsite-sync"
|
|
|
|
|
> "${MANIFEST}"
|
|
|
|
|
log "=== Offsite sync complete (success) ==="
|
|
|
|
|
else
|
|
|
|
|
warn "Offsite sync had errors — manifest preserved for retry"
|
|
|
|
|
log "=== Offsite sync complete (with errors) ==="
|
|
|
|
|
fi
|
|
|
|
|
|
|
|
|
|
cat <<EOF | curl -s --connect-timeout 5 --max-time 10 --data-binary @- "${PUSHGATEWAY}/metrics/job/${PUSHGATEWAY_JOB}" 2>/dev/null || true
|
|
|
|
|
backup_last_success_timestamp $(date +%s)
|
|
|
|
|
offsite_sync_last_status ${STATUS}
|
|
|
|
|
EOF
|
|
|
|
|
|
|
|
|
|
exit "${STATUS}"
|