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

@@ -16,7 +16,7 @@ class FormuleAbonnementTest {
@DisplayName("getters/setters")
void gettersSetters() {
FormuleAbonnement f = new FormuleAbonnement();
f.setCode(TypeFormule.STARTER);
f.setCode(TypeFormule.BASIC);
f.setLibelle("Starter");
f.setDescription("Pour petites structures");
f.setMaxMembres(50);
@@ -25,7 +25,7 @@ class FormuleAbonnementTest {
f.setPrixAnnuel(new BigDecimal("50000.00"));
f.setOrdreAffichage(1);
assertThat(f.getCode()).isEqualTo(TypeFormule.STARTER);
assertThat(f.getCode()).isEqualTo(TypeFormule.BASIC);
assertThat(f.getLibelle()).isEqualTo("Starter");
assertThat(f.getMaxMembres()).isEqualTo(50);
assertThat(f.getMaxStockageMo()).isEqualTo(1024);
@@ -37,7 +37,7 @@ class FormuleAbonnementTest {
@DisplayName("isIllimitee: true si maxMembres null")
void isIllimitee() {
FormuleAbonnement f = new FormuleAbonnement();
f.setCode(TypeFormule.CRYSTAL);
f.setCode(TypeFormule.PREMIUM);
f.setLibelle("Crystal");
f.setPrixMensuel(BigDecimal.ZERO);
f.setPrixAnnuel(BigDecimal.ZERO);
@@ -51,7 +51,7 @@ class FormuleAbonnementTest {
@DisplayName("accepteNouveauMembre")
void accepteNouveauMembre() {
FormuleAbonnement f = new FormuleAbonnement();
f.setCode(TypeFormule.STARTER);
f.setCode(TypeFormule.BASIC);
f.setLibelle("S");
f.setPrixMensuel(BigDecimal.ONE);
f.setPrixAnnuel(BigDecimal.TEN);
@@ -68,13 +68,13 @@ class FormuleAbonnementTest {
UUID id = UUID.randomUUID();
FormuleAbonnement a = new FormuleAbonnement();
a.setId(id);
a.setCode(TypeFormule.STARTER);
a.setCode(TypeFormule.BASIC);
a.setLibelle("S");
a.setPrixMensuel(BigDecimal.ONE);
a.setPrixAnnuel(BigDecimal.TEN);
FormuleAbonnement b = new FormuleAbonnement();
b.setId(id);
b.setCode(TypeFormule.STARTER);
b.setCode(TypeFormule.BASIC);
b.setLibelle("S");
b.setPrixMensuel(BigDecimal.ONE);
b.setPrixAnnuel(BigDecimal.TEN);
@@ -86,7 +86,7 @@ class FormuleAbonnementTest {
@DisplayName("toString non null")
void toString_nonNull() {
FormuleAbonnement f = new FormuleAbonnement();
f.setCode(TypeFormule.STARTER);
f.setCode(TypeFormule.BASIC);
f.setLibelle("S");
f.setPrixMensuel(BigDecimal.ONE);
f.setPrixAnnuel(BigDecimal.TEN);

View File

@@ -104,6 +104,15 @@ class MembreTest {
assertThat(a.hashCode()).isEqualTo(b.hashCode());
}
@Test
@DisplayName("notes getter/setter")
void notes() {
Membre m = new Membre();
assertThat(m.getNotes()).isNull();
m.setNotes("Ma biographie de test");
assertThat(m.getNotes()).isEqualTo("Ma biographie de test");
}
@Test
@DisplayName("toString non null")
void toString_nonNull() {

View File

@@ -1,6 +1,7 @@
package dev.lions.unionflow.server.entity;
import dev.lions.unionflow.server.api.enums.abonnement.StatutSouscription;
import dev.lions.unionflow.server.api.enums.abonnement.StatutValidationSouscription;
import dev.lions.unionflow.server.api.enums.abonnement.TypePeriodeAbonnement;
import org.junit.jupiter.api.DisplayName;
import org.junit.jupiter.api.Test;
@@ -47,6 +48,24 @@ class SouscriptionOrganisationBranchTest {
assertThat(s.getQuotaUtilise()).isEqualTo(0);
}
/**
* Branch manquante : statutValidation == null → la valeur par défaut EN_ATTENTE_PAIEMENT est affectée.
*/
@Test
@DisplayName("onCreate: statutValidation null → initialisé à EN_ATTENTE_PAIEMENT")
void onCreate_statutValidationNull_setsDefault() throws Exception {
SouscriptionOrganisation s = new SouscriptionOrganisation();
s.setDateDebut(LocalDate.now());
s.setDateFin(LocalDate.now().plusMonths(1));
s.setStatutValidation(null); // forcer null pour couvrir le then-branch de L168
Method onCreate = SouscriptionOrganisation.class.getDeclaredMethod("onCreate");
onCreate.setAccessible(true);
onCreate.invoke(s);
assertThat(s.getStatutValidation()).isEqualTo(StatutValidationSouscription.EN_ATTENTE_PAIEMENT);
}
// ── decrementerQuota() ────────────────────────────────────────────────────
/**

View File

@@ -1,6 +1,7 @@
package dev.lions.unionflow.server.entity;
import dev.lions.unionflow.server.api.enums.abonnement.StatutSouscription;
import dev.lions.unionflow.server.api.enums.abonnement.PlageMembres;
import dev.lions.unionflow.server.api.enums.abonnement.TypeFormule;
import dev.lions.unionflow.server.api.enums.abonnement.TypePeriodeAbonnement;
import org.junit.jupiter.api.DisplayName;
@@ -28,8 +29,9 @@ class SouscriptionOrganisationTest {
private static FormuleAbonnement newFormule() {
FormuleAbonnement f = new FormuleAbonnement();
f.setId(UUID.randomUUID());
f.setCode(TypeFormule.STARTER);
f.setLibelle("Starter");
f.setCode(TypeFormule.BASIC);
f.setPlage(PlageMembres.PETITE);
f.setLibelle("Basic");
f.setMaxMembres(100);
f.setPrixMensuel(java.math.BigDecimal.valueOf(5000));
f.setPrixAnnuel(java.math.BigDecimal.valueOf(50000));