All checks were successful
AI Code Review / ai-review (pull_request) Successful in 6s
Per reviewer (danijel): forte-drop's DB deployment belongs in apps/, not infra/. Straight relocation — same structure (Application + resources/ subdir), source.path updated to apps/base/forte-drop-postgresql/resources, wired into apps/overlays/upc-dev. Backup CronJob + RESTORE.md + sealed pg creds move with it. Consolidates the whole forte-drop deployment (postgres + web + mcp) under apps/. The infra PR (#17) is now superseded by this.
94 lines
3.3 KiB
YAML
94 lines
3.3 KiB
YAML
# Nightly logical backup of the forte-drop Postgres → UpCloud Managed Object Storage.
|
|
# Dumps to s3://drops/_pgbackups/ (the `_` prefix is collision-proof: app slugs match
|
|
# /^[a-z0-9][a-z0-9-]{0,62}$/ and can never start with `_`). Retains 30 days.
|
|
#
|
|
# Pod shape: initContainer pg_dump → shared emptyDir → mc upload + retention prune.
|
|
# Both images pinned. S3 creds reuse forte-drop-secrets (the app's UpCloud user has
|
|
# s3:* on the drops bucket). PG creds from forte-drop-pg-creds.
|
|
apiVersion: batch/v1
|
|
kind: CronJob
|
|
metadata:
|
|
name: forte-drop-pg-backup
|
|
namespace: forte-drop
|
|
labels:
|
|
app.kubernetes.io/name: postgresql
|
|
app.kubernetes.io/instance: forte-drop
|
|
app.kubernetes.io/component: backup
|
|
spec:
|
|
schedule: "0 2 * * *" # 02:00 UTC daily
|
|
concurrencyPolicy: Forbid
|
|
successfulJobsHistoryLimit: 3
|
|
failedJobsHistoryLimit: 3
|
|
jobTemplate:
|
|
spec:
|
|
backoffLimit: 2
|
|
template:
|
|
metadata:
|
|
labels:
|
|
app.kubernetes.io/name: postgresql
|
|
app.kubernetes.io/instance: forte-drop
|
|
app.kubernetes.io/component: backup
|
|
spec:
|
|
restartPolicy: Never
|
|
securityContext:
|
|
runAsNonRoot: true
|
|
runAsUser: 65532
|
|
fsGroup: 65532
|
|
volumes:
|
|
- name: work
|
|
emptyDir: {}
|
|
initContainers:
|
|
- name: dump
|
|
image: postgres:16-alpine
|
|
command:
|
|
- sh
|
|
- -c
|
|
- |
|
|
set -euo pipefail
|
|
TS=$(date -u +%Y%m%dT%H%M%SZ)
|
|
echo "dumping to /work/forte-drop-${TS}.sql.gz"
|
|
PGPASSWORD="$PGPASSWORD" pg_dump \
|
|
-h forte-drop-postgresql.forte-drop.svc \
|
|
-p 5432 -U "$PGUSER" -d drops \
|
|
--no-owner --no-privileges \
|
|
| gzip -9 > "/work/forte-drop-${TS}.sql.gz"
|
|
echo "dump complete: $(ls -lh /work/)"
|
|
env:
|
|
- name: PGUSER
|
|
valueFrom:
|
|
secretKeyRef: { name: forte-drop-pg-creds, key: pgusername }
|
|
- name: PGPASSWORD
|
|
valueFrom:
|
|
secretKeyRef: { name: forte-drop-pg-creds, key: pgpassword }
|
|
volumeMounts:
|
|
- name: work
|
|
mountPath: /work
|
|
containers:
|
|
- name: upload
|
|
image: quay.io/minio/mc:RELEASE.2024-11-21T17-21-54Z
|
|
command:
|
|
- sh
|
|
- -c
|
|
- |
|
|
set -euo pipefail
|
|
mc alias set obj "$S3_ENDPOINT" "$S3_KEY" "$S3_SECRET"
|
|
mc cp /work/*.sql.gz "obj/${S3_BUCKET}/_pgbackups/"
|
|
echo "uploaded. pruning backups older than 30d:"
|
|
mc rm --recursive --force --older-than 30d "obj/${S3_BUCKET}/_pgbackups/" || true
|
|
echo "backup retention pass complete"
|
|
env:
|
|
- name: S3_ENDPOINT
|
|
valueFrom:
|
|
secretKeyRef: { name: forte-drop-secrets, key: S3_ENDPOINT }
|
|
- name: S3_BUCKET
|
|
value: "drops"
|
|
- name: S3_KEY
|
|
valueFrom:
|
|
secretKeyRef: { name: forte-drop-secrets, key: S3_KEY }
|
|
- name: S3_SECRET
|
|
valueFrom:
|
|
secretKeyRef: { name: forte-drop-secrets, key: S3_SECRET }
|
|
volumeMounts:
|
|
- name: work
|
|
mountPath: /work
|