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:
@@ -42,6 +42,10 @@ public class MembreService {
|
||||
MembreRepository membreRepository;
|
||||
@Inject
|
||||
dev.lions.unionflow.server.repository.MembreRoleRepository membreRoleRepository;
|
||||
@Inject
|
||||
dev.lions.unionflow.server.repository.RoleRepository roleRepository;
|
||||
@Inject
|
||||
dev.lions.unionflow.server.repository.MembreOrganisationRepository membreOrganisationRepository;
|
||||
|
||||
@Inject
|
||||
dev.lions.unionflow.server.repository.TypeReferenceRepository typeReferenceRepository;
|
||||
@@ -58,6 +62,12 @@ public class MembreService {
|
||||
@Inject
|
||||
io.quarkus.security.identity.SecurityIdentity securityIdentity;
|
||||
|
||||
@Inject
|
||||
dev.lions.unionflow.server.repository.InscriptionEvenementRepository inscriptionEvenementRepository;
|
||||
|
||||
@Inject
|
||||
dev.lions.unionflow.server.messaging.KafkaEventProducer kafkaEventProducer;
|
||||
|
||||
/** Crée un nouveau membre en attente de validation admin */
|
||||
@Transactional
|
||||
public Membre creerMembre(Membre membre) {
|
||||
@@ -91,6 +101,20 @@ public class MembreService {
|
||||
|
||||
membreRepository.persist(membre);
|
||||
LOG.infof("Membre créé en attente de validation: %s (ID: %s)", membre.getNomComplet(), membre.getId());
|
||||
|
||||
// Publier l'événement Kafka pour mise à jour temps réel
|
||||
try {
|
||||
Map<String, Object> memberData = new HashMap<>();
|
||||
memberData.put("memberId", membre.getId().toString());
|
||||
memberData.put("nomComplet", membre.getNomComplet());
|
||||
memberData.put("email", membre.getEmail());
|
||||
memberData.put("numeroMembre", membre.getNumeroMembre());
|
||||
memberData.put("statutCompte", membre.getStatutCompte());
|
||||
kafkaEventProducer.publishMemberCreated(membre.getId(), null, memberData);
|
||||
} catch (Exception e) {
|
||||
LOG.warnf("Kafka event publication failed (non-blocking): %s", e.getMessage());
|
||||
}
|
||||
|
||||
return membre;
|
||||
}
|
||||
|
||||
@@ -115,6 +139,45 @@ public class MembreService {
|
||||
membreRepository.persist(membre);
|
||||
|
||||
LOG.infof("Membre activé avec succès: %s (ID: %s)", membre.getNomComplet(), membreId);
|
||||
|
||||
try {
|
||||
Map<String, Object> memberData = new HashMap<>();
|
||||
memberData.put("memberId", membre.getId().toString());
|
||||
memberData.put("nomComplet", membre.getNomComplet());
|
||||
memberData.put("statutCompte", "ACTIF");
|
||||
kafkaEventProducer.publishMemberUpdated(membre.getId(), null, memberData);
|
||||
} catch (Exception e) {
|
||||
LOG.warnf("Kafka event publication failed (non-blocking): %s", e.getMessage());
|
||||
}
|
||||
|
||||
return membre;
|
||||
}
|
||||
|
||||
/**
|
||||
* Affecte un membre existant à une organisation.
|
||||
* Crée le lien MembreOrganisation (statut EN_ATTENTE_VALIDATION) si inexistant.
|
||||
* Si le lien existe déjà, la méthode est idempotente.
|
||||
*
|
||||
* @param membreId UUID du membre
|
||||
* @param organisationId UUID de l'organisation cible
|
||||
* @return Le membre mis à jour
|
||||
*/
|
||||
@Transactional
|
||||
public Membre affecterOrganisation(UUID membreId, UUID organisationId) {
|
||||
LOG.infof("Affectation du membre %s à l'organisation %s", membreId, organisationId);
|
||||
|
||||
Membre membre = membreRepository.findByIdOptional(membreId)
|
||||
.orElseThrow(() -> new jakarta.ws.rs.NotFoundException("Membre non trouvé: " + membreId));
|
||||
|
||||
boolean dejaLie = membreOrganisationRepository.findFirstByMembreId(membreId).isPresent();
|
||||
if (dejaLie) {
|
||||
LOG.infof("Membre %s déjà lié à une organisation — opération ignorée", membreId);
|
||||
return membre;
|
||||
}
|
||||
|
||||
lierMembreOrganisationEtIncrementerQuota(membre, organisationId, "EN_ATTENTE_VALIDATION");
|
||||
|
||||
LOG.infof("Membre %s affecté à l'organisation %s", membre.getNumeroMembre(), organisationId);
|
||||
return membre;
|
||||
}
|
||||
|
||||
@@ -141,6 +204,13 @@ public class MembreService {
|
||||
membre.setActif(true);
|
||||
membreRepository.persist(membre);
|
||||
|
||||
// Mettre à jour le rôle BDD vers ORGADMIN
|
||||
membreOrganisationRepository.findFirstByMembreId(membreId).ifPresent(mo -> {
|
||||
membreRoleRepository.findActifsByMembreId(membreId)
|
||||
.forEach(mr -> { mr.setActif(false); entityManager.persist(mr); });
|
||||
assignerRoleDefaut(mo, "ORGADMIN");
|
||||
});
|
||||
|
||||
LOG.infof("Membre promu admin d'organisation: %s (ID: %s)", membre.getNomComplet(), membreId);
|
||||
return membre;
|
||||
}
|
||||
@@ -365,8 +435,34 @@ public class MembreService {
|
||||
dto.setAssociationNom(mo.getOrganisation().getNom());
|
||||
}
|
||||
dto.setDateAdhesion(mo.getDateAdhesion());
|
||||
} else if (membre.getDateCreation() != null) {
|
||||
// Fallback : date de création du compte comme date d'adhésion (membres sans organisation)
|
||||
dto.setDateAdhesion(membre.getDateCreation().toLocalDate());
|
||||
}
|
||||
|
||||
// Nombre d'événements auxquels le membre a participé
|
||||
dto.setNombreEvenementsParticipes(
|
||||
(int) inscriptionEvenementRepository.countByMembre(membre.getId()));
|
||||
|
||||
// Adresse principale (principale=true en priorité, sinon première adresse active)
|
||||
if (membre.getAdresses() != null && !membre.getAdresses().isEmpty()) {
|
||||
dev.lions.unionflow.server.entity.Adresse adressePrincipale = membre.getAdresses().stream()
|
||||
.filter(a -> Boolean.TRUE.equals(a.getPrincipale()) && Boolean.TRUE.equals(a.getActif()))
|
||||
.findFirst()
|
||||
.orElseGet(() -> membre.getAdresses().stream()
|
||||
.filter(a -> Boolean.TRUE.equals(a.getActif()))
|
||||
.findFirst()
|
||||
.orElse(null));
|
||||
if (adressePrincipale != null) {
|
||||
dto.setAdresse(adressePrincipale.getAdresse());
|
||||
dto.setVille(adressePrincipale.getVille());
|
||||
dto.setCodePostal(adressePrincipale.getCodePostal());
|
||||
}
|
||||
}
|
||||
|
||||
// Notes / biographie
|
||||
dto.setNotes(membre.getNotes());
|
||||
|
||||
// Champs de base DTO
|
||||
dto.setDateCreation(membre.getDateCreation());
|
||||
dto.setDateModification(membre.getDateModification());
|
||||
@@ -978,7 +1074,7 @@ public class MembreService {
|
||||
String jpql = "SELECT DISTINCT m FROM Membre m " +
|
||||
"JOIN m.membresOrganisations mo " +
|
||||
"WHERE mo.organisation.id IN :orgIds " +
|
||||
"AND (m.actif IS NULL OR m.actif = true) " +
|
||||
"AND (m.actif IS NULL OR m.actif = true OR m.statutCompte = 'EN_ATTENTE_VALIDATION') " +
|
||||
"ORDER BY m.nom ASC, m.prenom ASC";
|
||||
|
||||
TypedQuery<Membre> query = entityManager.createQuery(jpql, Membre.class);
|
||||
@@ -995,6 +1091,35 @@ public class MembreService {
|
||||
return membres;
|
||||
}
|
||||
|
||||
/** Compte le nombre total de membres pour les organisations données (même filtre que listerMembresParOrganisations). */
|
||||
public long compterMembresParOrganisations(List<UUID> organisationIds) {
|
||||
if (organisationIds == null || organisationIds.isEmpty()) return 0L;
|
||||
String jpql = "SELECT COUNT(DISTINCT m) FROM Membre m " +
|
||||
"JOIN m.membresOrganisations mo " +
|
||||
"WHERE mo.organisation.id IN :orgIds " +
|
||||
"AND (m.actif IS NULL OR m.actif = true OR m.statutCompte = 'EN_ATTENTE_VALIDATION')";
|
||||
TypedQuery<Long> query = entityManager.createQuery(jpql, Long.class);
|
||||
query.setParameter("orgIds", organisationIds);
|
||||
return query.getSingleResult();
|
||||
}
|
||||
|
||||
/**
|
||||
* Vérifie si une organisation possède une souscription active.
|
||||
* Utilisé pour déterminer si un membre créé par un admin doit être auto-activé.
|
||||
*
|
||||
* @param orgId UUID de l'organisation
|
||||
* @return true si une souscription ACTIVE existe pour cette organisation
|
||||
*/
|
||||
public boolean orgHasActiveSubscription(UUID orgId) {
|
||||
if (orgId == null) return false;
|
||||
return entityManager.createQuery(
|
||||
"SELECT COUNT(s) FROM SouscriptionOrganisation s " +
|
||||
"WHERE s.organisation.id = :orgId AND s.statut = 'ACTIVE'",
|
||||
Long.class)
|
||||
.setParameter("orgId", orgId)
|
||||
.getSingleResult() > 0;
|
||||
}
|
||||
|
||||
/**
|
||||
* Lie un membre à une organisation et incrémente le quota de la souscription.
|
||||
* Utilisé lors de la création unitaire ou de l'import massif.
|
||||
@@ -1051,6 +1176,13 @@ public class MembreService {
|
||||
|
||||
LOG.infof("MembreOrganisation créé (statut: %s)", statut);
|
||||
|
||||
// Incrémenter le compteur nombreMembres de l'organisation
|
||||
organisation.ajouterMembre();
|
||||
entityManager.persist(organisation);
|
||||
|
||||
// Assigner le rôle SIMPLEMEMBER par défaut
|
||||
assignerRoleDefaut(membreOrganisation, "SIMPLEMEMBER");
|
||||
|
||||
// Incrémenter quota si souscription existe
|
||||
if (souscriptionOpt.isPresent()) {
|
||||
dev.lions.unionflow.server.entity.SouscriptionOrganisation souscription = souscriptionOpt.get();
|
||||
@@ -1063,4 +1195,18 @@ public class MembreService {
|
||||
LOG.warn("Aucune souscription active trouvée pour organisation " + organisationId);
|
||||
}
|
||||
}
|
||||
|
||||
private void assignerRoleDefaut(dev.lions.unionflow.server.entity.MembreOrganisation mo, String roleCode) {
|
||||
roleRepository.findByCode(roleCode).ifPresent(role -> {
|
||||
dev.lions.unionflow.server.entity.MembreRole membreRole = new dev.lions.unionflow.server.entity.MembreRole();
|
||||
membreRole.setMembreOrganisation(mo);
|
||||
membreRole.setOrganisation(mo.getOrganisation());
|
||||
membreRole.setRole(role);
|
||||
membreRole.setActif(true);
|
||||
membreRole.setDateDebut(LocalDate.now());
|
||||
entityManager.persist(membreRole);
|
||||
LOG.infof("Rôle %s assigné au membre %s dans organisation %s",
|
||||
roleCode, mo.getMembre().getNumeroMembre(), mo.getOrganisation().getId());
|
||||
});
|
||||
}
|
||||
}
|
||||
|
||||
Reference in New Issue
Block a user