feat: BackupService real pg_dump, OrganisationService region stats, SystemConfigService overrides

- BackupService: DB-persisted metadata (BackupRecord/BackupConfig entities + V16 Flyway migration),
  real pg_dump execution via ProcessBuilder, soft-delete on deleteBackup, pg_restore manual guidance
- OrganisationService: repartitionRegion now queries Adresse entities (was Map.of() stub)
- SystemConfigService: in-memory config overrides via AtomicReference (no DB dependency)
- SystemMetricsService: null-guard on MemoryMXBean in getSystemStatus() (fixes test NPE)
- Souscription workflow: SouscriptionService, SouscriptionResource, FormuleAbonnementRepository,
  V11 Flyway migration, admin REST clients
- Flyway V8-V15: notes membres, types référence, type orga constraint, seed roles,
  première connexion, Wave checkout URL, Wave telephone column length fix
- .gitignore: added uploads/ and .claude/

Co-Authored-By: Claude Sonnet 4.6 <noreply@anthropic.com>
This commit is contained in:
dahoud
2026-04-04 16:14:30 +00:00
parent 9c66909eff
commit e00a9301d8
98 changed files with 5571 additions and 636 deletions

View File

@@ -0,0 +1,279 @@
-- =============================================================================
-- V11 — Workflow de souscription/onboarding UnionFlow
-- =============================================================================
-- Auteur : UnionFlow Team
-- Date : 2026-03-30
-- Objectif:
-- 0. Renommer les tables singulières créées par V1 vers les noms pluriels attendus
-- par les entités JPA (FormuleAbonnement → formules_abonnement, etc.)
-- 1. Ajouter les colonnes manquantes à formules_abonnement (entité refactorisée)
-- 2. Ajouter la colonne `plage` à formules_abonnement (PETITE/MOYENNE/GRANDE/TRES_GRANDE)
-- 3. Supprimer le code unique sur `code` seul (nouvelle contrainte sur code+plage)
-- 4. Migrer les anciennes valeurs TypeFormule (STARTER→BASIC, CRYSTAL supprimé)
-- 5. Vider et re-seeder avec la matrice tarifaire 4×3 (12 formules)
-- 6. Ajouter les colonnes du workflow de validation à souscriptions_organisation
-- 7. Ajouter EN_ATTENTE au statut (déjà en DB comme VARCHAR, pas de contrainte CHECK)
-- =============================================================================
-- -----------------------------------------------------------------------------
-- 0. Renommer les tables singulières → plurielles (alignement entités JPA)
-- -----------------------------------------------------------------------------
-- formule_abonnement → formules_abonnement (FormuleAbonnement entity @Table)
DO $$
BEGIN
IF EXISTS (SELECT 1 FROM pg_tables WHERE schemaname = 'public' AND tablename = 'formule_abonnement')
AND NOT EXISTS (SELECT 1 FROM pg_tables WHERE schemaname = 'public' AND tablename = 'formules_abonnement') THEN
ALTER TABLE formule_abonnement RENAME TO formules_abonnement;
END IF;
END $$;
-- souscription_organisation → souscriptions_organisation (SouscriptionOrganisation entity @Table)
DO $$
BEGIN
IF EXISTS (SELECT 1 FROM pg_tables WHERE schemaname = 'public' AND tablename = 'souscription_organisation')
AND NOT EXISTS (SELECT 1 FROM pg_tables WHERE schemaname = 'public' AND tablename = 'souscriptions_organisation') THEN
ALTER TABLE souscription_organisation RENAME TO souscriptions_organisation;
END IF;
END $$;
-- Mise à jour des noms de contraintes FK après le renommage
-- (PostgreSQL met à jour automatiquement les FK qui référencent la table renommée)
-- -----------------------------------------------------------------------------
-- 0b. Ajouter les colonnes manquantes à formules_abonnement (entité refactorisée)
-- -----------------------------------------------------------------------------
-- Colonne libelle (libellé de la formule — nullable temporairement, NOT NULL après seed)
ALTER TABLE formules_abonnement ADD COLUMN IF NOT EXISTS libelle VARCHAR(100);
-- Colonne plage (taille organisation cible)
ALTER TABLE formules_abonnement ADD COLUMN IF NOT EXISTS plage VARCHAR(20);
-- Nombre max de membres (NULL = illimité)
ALTER TABLE formules_abonnement ADD COLUMN IF NOT EXISTS max_membres INTEGER;
-- Stockage max en Mo (défaut 1 Go)
ALTER TABLE formules_abonnement ADD COLUMN IF NOT EXISTS max_stockage_mo INTEGER DEFAULT 1024;
-- Ordre d'affichage dans le catalogue
ALTER TABLE formules_abonnement ADD COLUMN IF NOT EXISTS ordre_affichage INTEGER DEFAULT 0;
-- La colonne `nom` de V1 n'est pas dans l'entité JPA — rendre nullable pour compatibilité
ALTER TABLE formules_abonnement ALTER COLUMN nom DROP NOT NULL;
-- -----------------------------------------------------------------------------
-- 0c. Supprimer les contraintes CHECK Hibernate obsolètes
-- -----------------------------------------------------------------------------
-- Hibernate génère automatiquement des CHECK constraints pour les enums @Enumerated(STRING).
-- Les valeurs ont changé (STARTER→BASIC, CRYSTAL supprimé) : on supprime ces contraintes
-- avant toute modification de données.
ALTER TABLE formules_abonnement DROP CONSTRAINT IF EXISTS formules_abonnement_code_check;
ALTER TABLE souscriptions_organisation DROP CONSTRAINT IF EXISTS souscriptions_organisation_statut_check;
ALTER TABLE souscriptions_organisation DROP CONSTRAINT IF EXISTS souscriptions_organisation_type_periode_check;
ALTER TABLE souscriptions_organisation DROP CONSTRAINT IF EXISTS souscriptions_organisation_statut_validation_check;
-- -----------------------------------------------------------------------------
-- 1. Préparer formules_abonnement
-- -----------------------------------------------------------------------------
-- Ajouter la colonne plage si elle n'existe pas encore
ALTER TABLE formules_abonnement ADD COLUMN IF NOT EXISTS plage VARCHAR(20);
-- Supprimer toutes les contraintes unicité sur code seul (V1 ou Hibernate)
DO $$
DECLARE r RECORD;
BEGIN
FOR r IN
SELECT conname FROM pg_constraint
WHERE conrelid = 'formules_abonnement'::regclass
AND contype = 'u'
AND conname NOT IN ('idx_formule_code_plage')
AND array_length(conkey, 1) = 1
AND conkey[1] = (
SELECT attnum FROM pg_attribute
WHERE attrelid = 'formules_abonnement'::regclass AND attname = 'code'
)
LOOP
EXECUTE 'ALTER TABLE formules_abonnement DROP CONSTRAINT ' || quote_ident(r.conname);
END LOOP;
END $$;
-- Supprimer les index uniques sur code seul s'ils existent
DROP INDEX IF EXISTS idx_formule_code;
DROP INDEX IF EXISTS formule_abonnement_code_key;
-- -----------------------------------------------------------------------------
-- 2. Migrer les anciennes données TypeFormule
-- -----------------------------------------------------------------------------
UPDATE formules_abonnement SET code = 'BASIC' WHERE code = 'STARTER';
DELETE FROM formules_abonnement WHERE code = 'CRYSTAL';
-- -----------------------------------------------------------------------------
-- 3. Vider et re-seeder avec la nouvelle matrice 4 plages × 3 formules
-- -----------------------------------------------------------------------------
DELETE FROM formules_abonnement;
INSERT INTO formules_abonnement (
id, version, actif, cree_par, modifie_par, date_creation, date_modification,
code, libelle, description, plage, max_membres, max_stockage_mo,
prix_mensuel, prix_annuel, ordre_affichage
) VALUES
-- ── PETITE (1100 membres) ──────────────────────────────────────────────────
(gen_random_uuid(), 0, true, 'SYSTEM', 'SYSTEM', NOW(), NOW(),
'BASIC', 'Basic — Petites structures',
'Idéal pour les petites associations de moins de 100 membres',
'PETITE', 100, 1024, 3000, 28800, 1),
(gen_random_uuid(), 0, true, 'SYSTEM', 'SYSTEM', NOW(), NOW(),
'STANDARD', 'Standard — Petites structures',
'Pour les associations actives de moins de 100 membres',
'PETITE', 100, 5120, 6000, 57600, 2),
(gen_random_uuid(), 0, true, 'SYSTEM', 'SYSTEM', NOW(), NOW(),
'PREMIUM', 'Premium — Petites structures',
'Fonctionnalités avancées pour petites structures ambitieuses',
'PETITE', 100, 10240, 10000, 96000, 3),
-- ── MOYENNE (101500 membres) ────────────────────────────────────────────────
(gen_random_uuid(), 0, true, 'SYSTEM', 'SYSTEM', NOW(), NOW(),
'BASIC', 'Basic — Moyennes structures',
'Gestion complète pour moyennes structures',
'MOYENNE', 500, 2048, 8000, 76800, 4),
(gen_random_uuid(), 0, true, 'SYSTEM', 'SYSTEM', NOW(), NOW(),
'STANDARD', 'Standard — Moyennes structures',
'Fonctionnalités étendues pour organisations en croissance',
'MOYENNE', 500, 10240, 15000, 144000, 5),
(gen_random_uuid(), 0, true, 'SYSTEM', 'SYSTEM', NOW(), NOW(),
'PREMIUM', 'Premium — Moyennes structures',
'Suite complète pour organisations actives',
'MOYENNE', 500, 20480, 25000, 240000, 6),
-- ── GRANDE (5012 000 membres) ────────────────────────────────────────────────
(gen_random_uuid(), 0, true, 'SYSTEM', 'SYSTEM', NOW(), NOW(),
'BASIC', 'Basic — Grandes structures',
'Solution économique pour grandes organisations',
'GRANDE', 2000, 5120, 20000, 192000, 7),
(gen_random_uuid(), 0, true, 'SYSTEM', 'SYSTEM', NOW(), NOW(),
'STANDARD', 'Standard — Grandes structures',
'Gestion avancée pour grandes structures',
'GRANDE', 2000, 20480, 35000, 336000, 8),
(gen_random_uuid(), 0, true, 'SYSTEM', 'SYSTEM', NOW(), NOW(),
'PREMIUM', 'Premium — Grandes structures',
'Fonctionnalités entreprise pour grandes organisations',
'GRANDE', 2000, 51200, 60000, 576000, 9),
-- ── TRES_GRANDE (2 000+ membres) ─────────────────────────────────────────────
(gen_random_uuid(), 0, true, 'SYSTEM', 'SYSTEM', NOW(), NOW(),
'BASIC', 'Basic — Très grandes structures',
'Gestion de base pour très grandes organisations',
'TRES_GRANDE', NULL, 10240, 50000, 480000, 10),
(gen_random_uuid(), 0, true, 'SYSTEM', 'SYSTEM', NOW(), NOW(),
'STANDARD', 'Standard — Très grandes structures',
'Suite complète pour très grandes organisations',
'TRES_GRANDE', NULL, 51200, 80000, 768000, 11),
(gen_random_uuid(), 0, true, 'SYSTEM', 'SYSTEM', NOW(), NOW(),
'PREMIUM', 'Premium — Très grandes structures',
'Solution enterprise multi-sites avec analytics avancé',
'TRES_GRANDE', NULL, 102400, 120000, 1152000, 12);
-- Appliquer NOT NULL sur plage après le seed
DO $$
BEGIN
-- PostgreSQL ne supporte pas ADD COLUMN ... NOT NULL ... en une seule commande
-- quand la table est déjà peuplée. On pose la contrainte séparément.
ALTER TABLE formules_abonnement ALTER COLUMN plage SET NOT NULL;
EXCEPTION
WHEN others THEN
RAISE NOTICE 'Contrainte NOT NULL sur plage déjà présente ou erreur: %', SQLERRM;
END $$;
-- Créer l'index unique sur la combinaison code+plage
CREATE UNIQUE INDEX IF NOT EXISTS idx_formule_code_plage
ON formules_abonnement (code, plage);
CREATE INDEX IF NOT EXISTS idx_formule_plage
ON formules_abonnement (plage);
-- -----------------------------------------------------------------------------
-- 4. Colonnes workflow de validation dans souscriptions_organisation
-- -----------------------------------------------------------------------------
ALTER TABLE souscriptions_organisation
ADD COLUMN IF NOT EXISTS plage VARCHAR(20);
ALTER TABLE souscriptions_organisation
ADD COLUMN IF NOT EXISTS type_organisation VARCHAR(30);
ALTER TABLE souscriptions_organisation
ADD COLUMN IF NOT EXISTS coefficient_applique NUMERIC(4, 2);
ALTER TABLE souscriptions_organisation
ADD COLUMN IF NOT EXISTS statut_validation VARCHAR(40);
ALTER TABLE souscriptions_organisation
ADD COLUMN IF NOT EXISTS montant_total NUMERIC(12, 2);
ALTER TABLE souscriptions_organisation
ADD COLUMN IF NOT EXISTS date_validation DATE;
ALTER TABLE souscriptions_organisation
ADD COLUMN IF NOT EXISTS validated_by_id UUID;
ALTER TABLE souscriptions_organisation
ADD COLUMN IF NOT EXISTS commentaire_rejet VARCHAR(500);
ALTER TABLE souscriptions_organisation
ADD COLUMN IF NOT EXISTS mot_de_passe_temporaire VARCHAR(100);
-- Backfill des lignes existantes avant d'ajouter les contraintes NOT NULL
UPDATE souscriptions_organisation
SET plage = 'PETITE',
type_organisation = 'ASSOCIATION',
coefficient_applique = 1.0,
statut_validation = 'VALIDEE'
WHERE plage IS NULL;
-- Appliquer NOT NULL sur statut_validation uniquement (les autres sont optionnels)
DO $$
BEGIN
ALTER TABLE souscriptions_organisation
ALTER COLUMN statut_validation SET NOT NULL;
EXCEPTION
WHEN others THEN
RAISE NOTICE 'Contrainte NOT NULL sur statut_validation: %', SQLERRM;
END $$;
-- Valeur par défaut pour les nouvelles lignes
ALTER TABLE souscriptions_organisation
ALTER COLUMN statut_validation SET DEFAULT 'EN_ATTENTE_PAIEMENT';
-- Index pour les requêtes SuperAdmin sur le workflow
CREATE INDEX IF NOT EXISTS idx_souscription_statut_validation
ON souscriptions_organisation (statut_validation);
CREATE INDEX IF NOT EXISTS idx_souscription_plage
ON souscriptions_organisation (plage);
-- Mise à jour du statut global pour les souscriptions existantes actives
-- (l'ancienne valeur ACTIVE reste cohérente avec VALIDEE côté validation)
UPDATE souscriptions_organisation
SET statut = 'ACTIVE'
WHERE statut IS NULL OR statut NOT IN ('ACTIVE', 'EXPIREE', 'SUSPENDUE', 'RESILIEE', 'EN_ATTENTE');
-- =============================================================================
-- Fin de V11
-- =============================================================================