Aller au contenu principal

Schéma de base de données

Ce document décrit le schéma de base de données SQLite utilisé par duplistatus pour stocker les données des opérations de sauvegarde.

Emplacement de la base de données

La base de données est stockée dans le répertoire des données d'application :

  • Emplacement par défaut : /app/data/backups.db
  • Volume Docker : duplistatus_data:/app/data
  • Nom du fichier : backups.db

Système de Migration de Base de Données

duplistatus utilise un système de migration automatisé pour gérer les modifications du schéma de base de données entre les versions.

Historique des versions de migration

Les versions de migration historiques suivantes ont amené la base de données à son état actuel :

  • Schema v1.0 (Application v0.6.x et antérieures) : Schéma de base de données initial avec tables machines et sauvegardes
  • Schema v2.0 (Application v0.7.x) : Ajout de colonnes manquantes et table configurations
  • Schema v3.0 (Application v0.7.x) : Renommage de la table machines en serveurs, ajout de la colonne server_url
  • Schema v3.1 (Application v0.8.x) : Amélioration des champs de données de sauvegarde, ajout de la colonne server_password
  • Schema v4.0 (Application v0.9.x / v1.0.x) : Ajout du contrôle d'accès utilisateur (tables utilisateurs, sessions, audit_log)

La version actuelle de l'application (v1.3.x) utilise Schema v4.0 comme dernière version du schéma de base de données.

Processus de Migration

  1. Sauvegarde Automatique : Crée une sauvegarde avant la migration
  2. Mise à Jour du Schéma : Met à jour la structure de la base de données
  3. Migration des Données : Préserve les données existantes
  4. Vérification : Confirme la migration réussie

Tableaux

Tableau des serveurs

Stocke les informations sur les serveurs Duplicati en cours de surveillance.

Champs

ChampTypeDescription
idTEXT PRIMARY KEYIdentifiant unique du serveur
nameTEXT NOT NULLNom du serveur depuis Duplicati
server_urlTEXTURL du serveur Duplicati
aliasTEXTNom convivial défini par l'utilisateur
noteTEXTNotes/Description définies par l'utilisateur
server_passwordTEXTMot de passe du serveur pour l'authentification
created_atDATETIMEHorodatage de création du serveur

Tableau des sauvegardes

Stocke les données d'opération de sauvegarde reçues des serveurs Duplicati.

Champs clés

ChampTypeDescription
idTEXT PRIMARY KEYIdentifiant unique de sauvegarde
server_idTEXT NOT NULLRéférence à la table serveurs
backup_nameTEXT NOT NULLNom du travail de sauvegarde
backup_idTEXT NOT NULLID de sauvegarde de Duplicati
dateDATETIME NOT NULLHeure d'exécution de la sauvegarde
statusTEXT NOT NULLStatut de la sauvegarde (Succès, Avertissement, Erreur, Fatal)
duration_secondsINTEGER NOT NULLDurée en secondes
sizeINTEGERTaille des fichiers source
uploaded_sizeINTEGERTaille des données téléversées
examined_filesINTEGERNombre de fichiers examinés
warningsINTEGERNombre d'avertissements
errorsINTEGERNombre d'erreurs
created_atDATETIMEHorodatage de création de l'enregistrement

Tableaux de Messages (Stockage JSON)

ChampTypeDescription
messages_arrayTEXTTableau JSON de messages de journal
warnings_arrayTEXTTableau JSON de messages d'avertissement
errors_arrayTEXTTableau JSON de messages d'erreur
available_backupsTEXTTableau JSON de versions de sauvegarde disponibles

Champs d'opération de fichier

ChampTypeDescription
examined_filesINTEGERFichiers examinés lors de la sauvegarde
opened_filesINTEGERFichiers ouverts pour la sauvegarde
added_filesINTEGERNouveaux fichiers ajoutés à la sauvegarde
modified_filesINTEGERFichiers modifiés dans la sauvegarde
deleted_filesINTEGERFichiers supprimés de la sauvegarde
deleted_foldersINTEGERDossiers supprimés de la sauvegarde
added_foldersINTEGERDossiers ajoutés à la sauvegarde
modified_foldersINTEGERDossiers modifiés dans la sauvegarde
not_processed_filesINTEGERFichiers non traités
too_large_filesINTEGERFichiers trop volumineux pour être traités
files_with_errorINTEGERFichiers avec erreurs
added_symlinksINTEGERLiens symboliques ajoutés
modified_symlinksINTEGERLiens symboliques modifiés
deleted_symlinksINTEGERLiens symboliques supprimés

Champs de Taille des fichiers

FieldTypeDescription
size_of_examined_filesINTEGERTaille des fichiers examinés lors de la sauvegarde
size_of_opened_filesINTEGERTaille des fichiers ouverts pour la sauvegarde
size_of_added_filesINTEGERTaille des nouveaux fichiers ajoutés à la sauvegarde
size_of_modified_filesINTEGERTaille des fichiers modifiés dans la sauvegarde

Champs de Statut d'Opération

ChampTypeDescription
parsed_resultTEXT NOT NULLRésultat de l'opération analysé
main_operationTEXT NOT NULLType d'opération principal
interruptedBOOLEANIndique si la sauvegarde a été interrompue
partial_backupBOOLEANIndique si la sauvegarde a été partielle
dryrunBOOLEANIndique si la sauvegarde a été un essai à blanc
versionTEXTVersion de Duplicati utilisée
begin_timeDATETIME NOT NULLHeure de début de la sauvegarde
end_timeDATETIME NOT NULLHeure de fin de la sauvegarde
warnings_actual_lengthINTEGERNombre réel d'avertissements
errors_actual_lengthINTEGERNombre réel d'erreurs
messages_actual_lengthINTEGERNombre réel de messages

Champs de Statistiques Backend

ChampTypeDescription
bytes_downloadedINTEGEROctets téléchargés de la destination
known_file_sizeINTEGERTaille des fichiers connue sur la destination
last_backup_dateDATETIMEDate de la dernière sauvegarde sur la destination
backup_list_countINTEGERNombre de versions de sauvegarde
reported_quota_errorBOOLEANErreur de quota signalée
reported_quota_warningBOOLEANAvertissement de quota signalé
backend_main_operationTEXTOpération principale du backend
backend_parsed_resultTEXTRésultat analysé du backend
backend_interruptedBOOLEANOpération du backend interrompue
backend_versionTEXTVersion du backend
backend_begin_timeDATETIMEHeure de début de l'opération du backend
backend_durationTEXTDurée de l'opération du backend
backend_warnings_actual_lengthINTEGERNombre d'avertissements du backend
backend_errors_actual_lengthINTEGERNombre d'erreurs du backend

Tableau des configurations

Stocke les paramètres de configuration de l'application.

Champs

ChampTypeDescription
keyTEXT PRIMARY KEY NOT NULLClé de configuration
valueTEXTValeur de configuration (JSON)

Clés de Configuration Communes

  • email_config: Paramètres de notification par e-mail
  • ntfy_config: Paramètres de notification NTFY
  • overdue_tolerance: Paramètres de tolérance de sauvegarde en retard
  • notification_templates: Modèles de message de notification
  • audit_retention_days: Période de rétention du journal d'audit (par défaut : 90 jours)

Tableau des versions de base de données

Suit la version du schéma de base de données à des fins de migration.

Champs

ChampTypeDescription
versionTEXT PRIMARY KEYVersion de la base de données
applied_atDATETIMEQuand la migration a été appliquée

Tableau Utilisateurs

Stocke les informations de compte utilisateur pour l'authentification et le contrôle d'accès.

Champs

ChampTypeDescription
idTEXT PRIMARY KEYIdentifiant utilisateur unique
usernameTEXT UNIQUE NOT NULLNom d'utilisateur pour la connexion
password_hashTEXT NOT NULLMot de passe haché avec Bcrypt
is_adminBOOLEAN NOT NULLSi l'utilisateur a les privilèges admin
must_change_passwordBOOLEANSi le changement de mot de passe est requis
created_atDATETIMEHorodatage de création du compte
updated_atDATETIMEHorodatage de dernière mise à jour
last_login_atDATETIMEHorodatage de dernière connexion réussie
last_login_ipTEXTAdresse IP de la dernière connexion
failed_login_attemptsINTEGERNombre de tentatives de connexion échouées
locked_untilDATETIMEExpiration du verrouillage du compte (si verrouillé)

Tableau Sessions

Stocke les données de session utilisateur pour l'authentification et la sécurité.

Champs

ChampTypeDescription
idTEXT PRIMARY KEYIdentifiant de session
user_idTEXTRéférence à la table utilisateurs (nullable pour les sessions non authentifiées)
created_atDATETIMEHorodatage de création de session
last_accessedDATETIMEHorodatage du Dernière accès
expires_atDATETIME NOT NULLHorodatage d'expiration de session
ip_addressTEXTAdresse IP d'origine de la session
user_agentTEXTChaîne Agent utilisateur
csrf_tokenTEXTJeton CSRF pour la session
csrf_expires_atDATETIMEExpiration du jeton CSRF

Journal d'Audit

Stocke la piste d'audit des actions des utilisateurs et des événements système.

Champs

ChampTypeDescription
idINTEGER PRIMARY KEY AUTOINCREMENTIdentifiant unique de l'entrée du journal d'audit
timestampDATETIMEHorodatage de l'événement
user_idTEXTRéférence à la table Utilisateurs (nullable)
usernameTEXTNom d'utilisateur au moment de l'action
actionTEXT NOT NULLAction effectuée
categoryTEXT NOT NULLCatégorie de l'action (par ex. « authentication », « Paramètres », « sauvegarde »)
target_typeTEXTType de cible (par ex. « Serveur », « sauvegarde », « Utilisateur »)
target_idTEXTIdentifiant de la cible
detailsTEXTDétails supplémentaires (JSON)
ip_addressTEXTAdresse IP du demandeur
user_agentTEXTChaîne Agent utilisateur
statusTEXT NOT NULLStatut de l'action (« Succès », « Échec », « Erreur »)
error_messageTEXTMessage d'erreur si l'action a échoué

Gestion des sessions

Stockage de session sauvegardé par base de données

Les sessions sont stockées dans la base de données avec secours en mémoire :

  • Stockage Principal : Table de sessions sauvegardée en base de données
  • Secours : Stockage en mémoire (support hérité ou cas d'erreur)
  • ID de Session : Chaîne aléatoire cryptographiquement sécurisée
  • Expiration : Délai dépassé de session configurable
  • Protection CSRF : Protection contre les attaques par falsification de requête intersite
  • Nettoyage Automatique : Les sessions expirées sont automatiquement supprimées

Points de terminaison de l'API de session

  • POST /api/session : Créer une nouvelle session
  • GET /api/session : Valider une session existante
  • DELETE /api/session : Détruire la session
  • GET /api/csrf : Obtenir le jeton CSRF

Index

La base de données comprend plusieurs index pour des performances de requête optimales :

  • Clés primaires : Tous les tableaux ont des index de clé primaire
  • Clés étrangères : Références de Serveur dans la table sauvegardes, références d'Utilisateur dans sessions et audit_log
  • Optimisation des requêtes : Index sur les champs fréquemment interrogés
  • Index de Date : Index sur les champs de date pour les requêtes basées sur le temps
  • Index d'Utilisateur : Index de nom d'utilisateur pour les recherches rapides d'utilisateurs
  • Index de Session : Index d'expiration et user_id pour la gestion des sessions
  • Index d'audit : Index d'horodatage, user_id, action, catégorie et statut pour les requêtes d'audit

Relations

  • Serveurs → Sauvegardes : Relation un-à-plusieurs
  • Utilisateurs → Sessions : Relation un-à-plusieurs (les sessions peuvent exister sans utilisateurs)
  • Utilisateurs → Journal d'Audit : Relation un-à-plusieurs (les entrées d'audit peuvent exister sans utilisateurs)
  • Sauvegardes → Messages : Tableaux JSON intégrés
  • Configurations : Stockage clé-valeur

Types de données

  • TEXT : Données de chaîne de caractères, tableaux JSON
  • INTEGER : Données numériques, nombre de fichiers, tailles
  • REAL : Nombres à virgule flottante, durées
  • DATETIME : Données d'horodatage
  • BOOLEAN : Valeurs vrai/faux

Valeurs de Statut de Sauvegarde

  • Succès : Sauvegarde terminée avec succès
  • Avertissement : Sauvegarde terminée avec avertissements
  • Erreur : Sauvegarde terminée avec erreurs
  • Fatal : Sauvegarde échouée fatalement

Requêtes courantes

Obtenir la dernière sauvegarde pour un serveur

SELECT * FROM backups 
WHERE server_id = ?
ORDER BY date DESC
LIMIT 1;

Obtenir toutes les sauvegardes pour un serveur

SELECT * FROM backups 
WHERE server_id = ?
ORDER BY date DESC;

Obtenir le résumé du serveur

SELECT 
s.name,
s.alias,
COUNT(b.id) as backup_count,
MAX(b.date) as last_backup,
b.status as last_status
FROM servers s
LEFT JOIN backups b ON s.id = b.server_id
GROUP BY s.id;

Obtenir le Résumé Général

SELECT 
COUNT(DISTINCT s.id) as total_servers,
COUNT(b.id) as total_backups_runs,
COUNT(DISTINCT s.id || ':' || b.backup_name) as total_backups,
COALESCE(SUM(b.uploaded_size), 0) as total_uploaded_size,
(
SELECT COALESCE(SUM(b2.known_file_size), 0)
FROM backups b2
INNER JOIN (
SELECT server_id, MAX(date) as max_date
FROM backups
GROUP BY server_id
) latest ON b2.server_id = latest.server_id AND b2.date = latest.max_date
) as total_storage_used,
(
SELECT COALESCE(SUM(b2.size_of_examined_files), 0)
FROM backups b2
INNER JOIN (
SELECT server_id, MAX(date) as max_date
FROM backups
GROUP BY server_id
) latest ON b2.server_id = latest.server_id AND b2.date = latest.max_date
) as total_backuped_size
FROM servers s
LEFT JOIN backups b ON b.server_id = s.id;

Nettoyage de la base de données

-- Delete old backups (older than 30 days)
DELETE FROM backups
WHERE date < datetime('now', '-30 days');

-- Delete servers with no backups
DELETE FROM servers
WHERE id NOT IN (SELECT DISTINCT server_id FROM backups);

Mappage JSON vers base de données

Mappage du corps de la requête API aux colonnes de la base de données

Quand Duplicati envoie les données de sauvegarde via HTTP POST, la structure JSON est mappée aux colonnes de la base de données :

{
"Data": {
"ExaminedFiles": 15399, // → examined_files
"OpenedFiles": 1861, // → opened_files
"AddedFiles": 1861, // → added_files
"SizeOfExaminedFiles": 11086692615, // → size_of_examined_files
"SizeOfOpenedFiles": 13450481, // → size_of_opened_files
"SizeOfAddedFiles": 13450481, // → size_of_added_files
"SizeOfModifiedFiles": 0, // → size_of_modified_files
"ParsedResult": "Success", // → status
"BeginTime": "2025-04-21T23:45:46.9712217Z", // → begin_time and date
"Duration": "00:00:51.3856057", // → duration_seconds (calculated)
"WarningsActualLength": 0, // → warnings_actual_length
"ErrorsActualLength": 0 // → errors_actual_length
},
"Extra": {
"machine-id": "66f5ffc7ff474a73a3c9cba4ac7bfb65", // → server_id
"machine-name": "WSJ-SER5", // → server name
"backup-name": "WSJ-SER5 Local files", // → backup_name
"backup-id": "DB-2" // → backup_id
}
}

Note : Le champ size dans la table des sauvegardes stocke SizeOfExaminedFiles et uploaded_size stocke la taille réelle téléversée/transférée à partir de l'opération de sauvegarde.