Files
oikos/server/routes/backup.js
T
Ulas Kalayci 9b29d1847c feat: automatische geplante Backups mit Rotation
Phase 1.3 - Automatische Backups:
- Cron-basierter Scheduler (Standard: täglich 2 Uhr)
- Konfigurierbar über .env (Zeitplan, Verzeichnis, Anzahl)
- Automatische Rotation: behält nur letzte N Backups (Standard: 7)
- UI in Settings → Backup: Status-Anzeige und manueller Trigger
- Tests: 7 erfolgreiche Tests für Scheduler-Funktionalität

Neue Umgebungsvariablen:
- BACKUP_ENABLED (Standard: true)
- BACKUP_SCHEDULE (Standard: 0 2 * * *)
- BACKUP_DIR (Standard: ./backups)
- BACKUP_KEEP (Standard: 7)
- TZ (für Zeitzone)

Co-Authored-By: Claude Opus 4.7 <noreply@anthropic.com>
2026-05-04 07:02:38 +02:00

111 lines
3.4 KiB
JavaScript

/**
* Module: Database Backup
* Purpose: Authenticated admin-only database backup and restore endpoints.
* Dependencies: express, server/db.js
*/
import express from 'express';
import os from 'node:os';
import path from 'node:path';
import fs from 'node:fs/promises';
import { backupToFile, currentVersion, restoreFromFile } from '../db.js';
import { requireAdmin } from '../auth.js';
import { createLogger } from '../logger.js';
import { getStatus as getSchedulerStatus, triggerBackup } from '../services/backup-scheduler.js';
const router = express.Router();
const log = createLogger('Backup');
const RESTORE_LIMIT = process.env.BACKUP_UPLOAD_LIMIT || '100mb';
function backupFileName() {
const stamp = new Date().toISOString().replace(/[:.]/g, '-');
return `oikos-backup-${stamp}.db`;
}
router.get('/status', requireAdmin, (req, res) => {
const schedulerStatus = getSchedulerStatus();
res.json({
data: {
schema_version: currentVersion(),
restore_upload_limit: RESTORE_LIMIT,
scheduler: schedulerStatus,
},
});
});
router.get('/database', requireAdmin, async (req, res) => {
let tmpPath = null;
try {
const dir = await fs.mkdtemp(path.join(os.tmpdir(), 'oikos-backup-'));
tmpPath = path.join(dir, backupFileName());
await backupToFile(tmpPath);
res.setHeader('Cache-Control', 'no-store');
res.download(tmpPath, path.basename(tmpPath), async (err) => {
try { await fs.rm(dir, { recursive: true, force: true }); } catch { /* best effort */ }
if (err && !res.headersSent) {
log.error('Backup download failed:', err);
}
});
} catch (err) {
log.error('Database backup failed:', err);
if (tmpPath) {
try { await fs.rm(path.dirname(tmpPath), { recursive: true, force: true }); } catch { /* best effort */ }
}
res.status(500).json({ error: 'Database backup failed.', code: 500 });
}
});
router.post(
'/restore',
requireAdmin,
express.raw({ type: 'application/octet-stream', limit: RESTORE_LIMIT }),
async (req, res) => {
let dir = null;
try {
if (!Buffer.isBuffer(req.body) || req.body.length === 0) {
return res.status(400).json({ error: 'Backup file is required.', code: 400 });
}
dir = await fs.mkdtemp(path.join(os.tmpdir(), 'oikos-restore-'));
const uploadPath = path.join(dir, 'restore.db');
await fs.writeFile(uploadPath, req.body);
const result = await restoreFromFile(uploadPath);
res.json({
ok: true,
data: {
schema_version: result.schemaVersion,
},
});
} catch (err) {
log.error('Database restore failed:', err);
const message = err?.message || 'Database restore failed.';
res.status(400).json({ error: message, code: 400 });
} finally {
if (dir) {
try { await fs.rm(dir, { recursive: true, force: true }); } catch { /* best effort */ }
}
}
}
);
router.post('/trigger', requireAdmin, async (req, res) => {
try {
const result = await triggerBackup();
res.json({ data: result });
} catch (err) {
log.error('Manual backup trigger failed:', err);
res.status(500).json({ error: 'Backup trigger failed.', code: 500 });
}
});
router.use((err, req, res, next) => {
if (err?.type === 'entity.too.large') {
return res.status(413).json({ error: `Backup file is too large. Maximum upload size is ${RESTORE_LIMIT}.`, code: 413 });
}
next(err);
});
export default router;