feat: PHASE 6.2 - Repositories, DTOs et Service Notifications
Repositories créés: - TemplateNotificationRepository: Recherche par code, langue - NotificationRepository: Recherche par membre, organisation, type, statut, priorité, en attente DTOs créés: - NotificationDTO: Validation complète avec contraintes - TemplateNotificationDTO: Gestion templates avec variables Service créé: - NotificationService: CRUD templates, CRUD notifications, marquer comme lue - Liste notifications par membre, non lues, en attente d'envoi - Conversions DTO ↔ Entity complètes Respect strict DRY/WOU: - Patterns cohérents avec autres modules - Gestion d'erreurs standardisée
This commit is contained in:
@@ -1,659 +1,68 @@
|
||||
package dev.lions.unionflow.server.api.dto.notification;
|
||||
|
||||
import com.fasterxml.jackson.annotation.JsonFormat;
|
||||
import com.fasterxml.jackson.annotation.JsonInclude;
|
||||
import dev.lions.unionflow.server.api.enums.notification.CanalNotification;
|
||||
import dev.lions.unionflow.server.api.dto.base.BaseDTO;
|
||||
import dev.lions.unionflow.server.api.enums.notification.PrioriteNotification;
|
||||
import dev.lions.unionflow.server.api.enums.notification.StatutNotification;
|
||||
import dev.lions.unionflow.server.api.enums.notification.TypeNotification;
|
||||
import jakarta.validation.constraints.*;
|
||||
import jakarta.validation.constraints.NotNull;
|
||||
import java.time.LocalDateTime;
|
||||
import java.util.List;
|
||||
import java.util.Map;
|
||||
import java.util.UUID;
|
||||
import lombok.Getter;
|
||||
import lombok.Setter;
|
||||
|
||||
/**
|
||||
* DTO pour les notifications UnionFlow
|
||||
*
|
||||
* <p>Ce DTO représente une notification complète avec toutes ses propriétés, métadonnées et
|
||||
* informations de suivi.
|
||||
* DTO pour la gestion des notifications
|
||||
*
|
||||
* @author UnionFlow Team
|
||||
* @version 1.0
|
||||
* @since 2025-01-16
|
||||
* @version 3.0
|
||||
* @since 2025-01-29
|
||||
*/
|
||||
@JsonInclude(JsonInclude.Include.NON_NULL)
|
||||
public class NotificationDTO {
|
||||
@Getter
|
||||
@Setter
|
||||
public class NotificationDTO extends BaseDTO {
|
||||
|
||||
/** Identifiant unique de la notification */
|
||||
private String id;
|
||||
private static final long serialVersionUID = 1L;
|
||||
|
||||
/** Type de notification */
|
||||
@NotNull(message = "Le type de notification est obligatoire")
|
||||
private TypeNotification typeNotification;
|
||||
|
||||
/** Statut actuel de la notification */
|
||||
@NotNull(message = "Le statut de notification est obligatoire")
|
||||
/** Priorité */
|
||||
private PrioriteNotification priorite;
|
||||
|
||||
/** Statut */
|
||||
private StatutNotification statut;
|
||||
|
||||
/** Canal de notification utilisé */
|
||||
@NotNull(message = "Le canal de notification est obligatoire")
|
||||
private CanalNotification canal;
|
||||
/** Sujet */
|
||||
private String sujet;
|
||||
|
||||
/** Titre de la notification */
|
||||
@NotBlank(message = "Le titre ne peut pas être vide")
|
||||
@Size(max = 100, message = "Le titre ne peut pas dépasser 100 caractères")
|
||||
private String titre;
|
||||
/** Corps du message */
|
||||
private String corps;
|
||||
|
||||
/** Corps du message de la notification */
|
||||
@NotBlank(message = "Le message ne peut pas être vide")
|
||||
@Size(max = 500, message = "Le message ne peut pas dépasser 500 caractères")
|
||||
private String message;
|
||||
/** Date d'envoi prévue */
|
||||
private LocalDateTime dateEnvoiPrevue;
|
||||
|
||||
/** Message court pour l'affichage dans la barre de notification */
|
||||
@Size(max = 150, message = "Le message court ne peut pas dépasser 150 caractères")
|
||||
private String messageCourt;
|
||||
|
||||
/** Identifiant de l'expéditeur */
|
||||
private String expediteurId;
|
||||
|
||||
/** Nom de l'expéditeur */
|
||||
private String expediteurNom;
|
||||
|
||||
/** Liste des identifiants des destinataires */
|
||||
@NotEmpty(message = "Au moins un destinataire est requis")
|
||||
private List<String> destinatairesIds;
|
||||
|
||||
/** Identifiant de l'organisation concernée */
|
||||
private String organisationId;
|
||||
|
||||
/** Données personnalisées de la notification */
|
||||
private Map<String, Object> donneesPersonnalisees;
|
||||
|
||||
/** URL de l'image à afficher (optionnel) */
|
||||
private String imageUrl;
|
||||
|
||||
/** URL de l'icône personnalisée (optionnel) */
|
||||
private String iconeUrl;
|
||||
|
||||
/** Action à exécuter lors du clic */
|
||||
private String actionClic;
|
||||
|
||||
/** Paramètres de l'action */
|
||||
private Map<String, String> parametresAction;
|
||||
|
||||
/** Boutons d'action rapide */
|
||||
private List<ActionNotificationDTO> actionsRapides;
|
||||
|
||||
/** Date et heure de création */
|
||||
@JsonFormat(pattern = "yyyy-MM-dd'T'HH:mm:ss")
|
||||
private LocalDateTime dateCreation;
|
||||
|
||||
/** Date et heure d'envoi programmé */
|
||||
@JsonFormat(pattern = "yyyy-MM-dd'T'HH:mm:ss")
|
||||
private LocalDateTime dateEnvoiProgramme;
|
||||
|
||||
/** Date et heure d'envoi effectif */
|
||||
@JsonFormat(pattern = "yyyy-MM-dd'T'HH:mm:ss")
|
||||
/** Date d'envoi réelle */
|
||||
private LocalDateTime dateEnvoi;
|
||||
|
||||
/** Date et heure d'expiration */
|
||||
@JsonFormat(pattern = "yyyy-MM-dd'T'HH:mm:ss")
|
||||
private LocalDateTime dateExpiration;
|
||||
/** Date de lecture */
|
||||
private LocalDateTime dateLecture;
|
||||
|
||||
/** Date et heure de dernière lecture */
|
||||
@JsonFormat(pattern = "yyyy-MM-dd'T'HH:mm:ss")
|
||||
private LocalDateTime dateDerniereLecture;
|
||||
|
||||
/** Priorité de la notification (1=basse, 5=haute) */
|
||||
@Min(value = 1, message = "La priorité doit être comprise entre 1 et 5")
|
||||
@Max(value = 5, message = "La priorité doit être comprise entre 1 et 5")
|
||||
private Integer priorite;
|
||||
|
||||
/** Nombre de tentatives d'envoi */
|
||||
/** Nombre de tentatives */
|
||||
private Integer nombreTentatives;
|
||||
|
||||
/** Nombre maximum de tentatives autorisées */
|
||||
private Integer maxTentatives;
|
||||
|
||||
/** Délai entre les tentatives en minutes */
|
||||
private Integer delaiTentativesMinutes;
|
||||
|
||||
/** Indique si la notification doit vibrer */
|
||||
private Boolean doitVibrer;
|
||||
|
||||
/** Indique si la notification doit émettre un son */
|
||||
private Boolean doitEmettreSon;
|
||||
|
||||
/** Indique si la notification doit allumer la LED */
|
||||
private Boolean doitAllumerLED;
|
||||
|
||||
/** Pattern de vibration personnalisé */
|
||||
private long[] patternVibration;
|
||||
|
||||
/** Son personnalisé à jouer */
|
||||
private String sonPersonnalise;
|
||||
|
||||
/** Couleur de la LED */
|
||||
private String couleurLED;
|
||||
|
||||
/** Indique si la notification est lue */
|
||||
private Boolean estLue;
|
||||
|
||||
/** Indique si la notification est marquée comme importante */
|
||||
private Boolean estImportante;
|
||||
|
||||
/** Indique si la notification est archivée */
|
||||
private Boolean estArchivee;
|
||||
|
||||
/** Nombre de fois que la notification a été affichée */
|
||||
private Integer nombreAffichages;
|
||||
|
||||
/** Nombre de clics sur la notification */
|
||||
private Integer nombreClics;
|
||||
|
||||
/** Taux de livraison (pourcentage) */
|
||||
private Double tauxLivraison;
|
||||
|
||||
/** Taux d'ouverture (pourcentage) */
|
||||
private Double tauxOuverture;
|
||||
|
||||
/** Temps moyen de lecture en secondes */
|
||||
private Integer tempsMoyenLectureSecondes;
|
||||
|
||||
/** Message d'erreur en cas d'échec */
|
||||
/** Message d'erreur */
|
||||
private String messageErreur;
|
||||
|
||||
/** Code d'erreur technique */
|
||||
private String codeErreur;
|
||||
/** Données additionnelles (JSON) */
|
||||
private String donneesAdditionnelles;
|
||||
|
||||
/** Trace de la pile d'erreur (pour debug) */
|
||||
private String traceErreur;
|
||||
/** ID du membre */
|
||||
private UUID membreId;
|
||||
|
||||
/** Métadonnées techniques */
|
||||
private Map<String, Object> metadonnees;
|
||||
/** ID de l'organisation */
|
||||
private UUID organisationId;
|
||||
|
||||
/** Tags pour catégorisation */
|
||||
private List<String> tags;
|
||||
|
||||
/** Identifiant de la campagne (si applicable) */
|
||||
private String campagneId;
|
||||
|
||||
/** Version de l'application qui a créé la notification */
|
||||
private String versionApp;
|
||||
|
||||
/** Plateforme cible (android, ios, web) */
|
||||
private String plateforme;
|
||||
|
||||
/** Token FCM du destinataire (usage interne) */
|
||||
private String tokenFCM;
|
||||
|
||||
/** Identifiant de suivi externe */
|
||||
private String idSuiviExterne;
|
||||
|
||||
// === CONSTRUCTEURS ===
|
||||
|
||||
/** Constructeur par défaut */
|
||||
public NotificationDTO() {
|
||||
this.dateCreation = LocalDateTime.now();
|
||||
this.statut = StatutNotification.BROUILLON;
|
||||
this.nombreTentatives = 0;
|
||||
this.maxTentatives = 3;
|
||||
this.delaiTentativesMinutes = 5;
|
||||
this.estLue = false;
|
||||
this.estImportante = false;
|
||||
this.estArchivee = false;
|
||||
this.nombreAffichages = 0;
|
||||
this.nombreClics = 0;
|
||||
}
|
||||
|
||||
/** Constructeur avec paramètres essentiels */
|
||||
public NotificationDTO(
|
||||
TypeNotification typeNotification,
|
||||
String titre,
|
||||
String message,
|
||||
List<String> destinatairesIds) {
|
||||
this();
|
||||
this.typeNotification = typeNotification;
|
||||
this.titre = titre;
|
||||
this.message = message;
|
||||
this.destinatairesIds = destinatairesIds;
|
||||
this.canal = CanalNotification.valueOf(typeNotification.getCanalNotification());
|
||||
this.priorite = typeNotification.getNiveauPriorite();
|
||||
this.doitVibrer = typeNotification.doitVibrer();
|
||||
this.doitEmettreSon = typeNotification.doitEmettreSon();
|
||||
}
|
||||
|
||||
// === GETTERS ET SETTERS ===
|
||||
|
||||
public String getId() {
|
||||
return id;
|
||||
}
|
||||
|
||||
public void setId(String id) {
|
||||
this.id = id;
|
||||
}
|
||||
|
||||
public TypeNotification getTypeNotification() {
|
||||
return typeNotification;
|
||||
}
|
||||
|
||||
public void setTypeNotification(TypeNotification typeNotification) {
|
||||
this.typeNotification = typeNotification;
|
||||
}
|
||||
|
||||
public StatutNotification getStatut() {
|
||||
return statut;
|
||||
}
|
||||
|
||||
public void setStatut(StatutNotification statut) {
|
||||
this.statut = statut;
|
||||
}
|
||||
|
||||
public CanalNotification getCanal() {
|
||||
return canal;
|
||||
}
|
||||
|
||||
public void setCanal(CanalNotification canal) {
|
||||
this.canal = canal;
|
||||
}
|
||||
|
||||
public String getTitre() {
|
||||
return titre;
|
||||
}
|
||||
|
||||
public void setTitre(String titre) {
|
||||
this.titre = titre;
|
||||
}
|
||||
|
||||
public String getMessage() {
|
||||
return message;
|
||||
}
|
||||
|
||||
public void setMessage(String message) {
|
||||
this.message = message;
|
||||
}
|
||||
|
||||
public String getMessageCourt() {
|
||||
return messageCourt;
|
||||
}
|
||||
|
||||
public void setMessageCourt(String messageCourt) {
|
||||
this.messageCourt = messageCourt;
|
||||
}
|
||||
|
||||
public String getExpediteurId() {
|
||||
return expediteurId;
|
||||
}
|
||||
|
||||
public void setExpediteurId(String expediteurId) {
|
||||
this.expediteurId = expediteurId;
|
||||
}
|
||||
|
||||
public String getExpediteurNom() {
|
||||
return expediteurNom;
|
||||
}
|
||||
|
||||
public void setExpediteurNom(String expediteurNom) {
|
||||
this.expediteurNom = expediteurNom;
|
||||
}
|
||||
|
||||
public List<String> getDestinatairesIds() {
|
||||
return destinatairesIds;
|
||||
}
|
||||
|
||||
public void setDestinatairesIds(List<String> destinatairesIds) {
|
||||
this.destinatairesIds = destinatairesIds;
|
||||
}
|
||||
|
||||
public String getOrganisationId() {
|
||||
return organisationId;
|
||||
}
|
||||
|
||||
public void setOrganisationId(String organisationId) {
|
||||
this.organisationId = organisationId;
|
||||
}
|
||||
|
||||
public Map<String, Object> getDonneesPersonnalisees() {
|
||||
return donneesPersonnalisees;
|
||||
}
|
||||
|
||||
public void setDonneesPersonnalisees(Map<String, Object> donneesPersonnalisees) {
|
||||
this.donneesPersonnalisees = donneesPersonnalisees;
|
||||
}
|
||||
|
||||
public String getImageUrl() {
|
||||
return imageUrl;
|
||||
}
|
||||
|
||||
public void setImageUrl(String imageUrl) {
|
||||
this.imageUrl = imageUrl;
|
||||
}
|
||||
|
||||
public String getIconeUrl() {
|
||||
return iconeUrl;
|
||||
}
|
||||
|
||||
public void setIconeUrl(String iconeUrl) {
|
||||
this.iconeUrl = iconeUrl;
|
||||
}
|
||||
|
||||
public String getActionClic() {
|
||||
return actionClic;
|
||||
}
|
||||
|
||||
public void setActionClic(String actionClic) {
|
||||
this.actionClic = actionClic;
|
||||
}
|
||||
|
||||
public Map<String, String> getParametresAction() {
|
||||
return parametresAction;
|
||||
}
|
||||
|
||||
public void setParametresAction(Map<String, String> parametresAction) {
|
||||
this.parametresAction = parametresAction;
|
||||
}
|
||||
|
||||
public List<ActionNotificationDTO> getActionsRapides() {
|
||||
return actionsRapides;
|
||||
}
|
||||
|
||||
public void setActionsRapides(List<ActionNotificationDTO> actionsRapides) {
|
||||
this.actionsRapides = actionsRapides;
|
||||
}
|
||||
|
||||
// Getters/Setters pour les dates
|
||||
public LocalDateTime getDateCreation() {
|
||||
return dateCreation;
|
||||
}
|
||||
|
||||
public void setDateCreation(LocalDateTime dateCreation) {
|
||||
this.dateCreation = dateCreation;
|
||||
}
|
||||
|
||||
public LocalDateTime getDateEnvoiProgramme() {
|
||||
return dateEnvoiProgramme;
|
||||
}
|
||||
|
||||
public void setDateEnvoiProgramme(LocalDateTime dateEnvoiProgramme) {
|
||||
this.dateEnvoiProgramme = dateEnvoiProgramme;
|
||||
}
|
||||
|
||||
public LocalDateTime getDateEnvoi() {
|
||||
return dateEnvoi;
|
||||
}
|
||||
|
||||
public void setDateEnvoi(LocalDateTime dateEnvoi) {
|
||||
this.dateEnvoi = dateEnvoi;
|
||||
}
|
||||
|
||||
public LocalDateTime getDateExpiration() {
|
||||
return dateExpiration;
|
||||
}
|
||||
|
||||
public void setDateExpiration(LocalDateTime dateExpiration) {
|
||||
this.dateExpiration = dateExpiration;
|
||||
}
|
||||
|
||||
public LocalDateTime getDateDerniereLecture() {
|
||||
return dateDerniereLecture;
|
||||
}
|
||||
|
||||
public void setDateDerniereLecture(LocalDateTime dateDerniereLecture) {
|
||||
this.dateDerniereLecture = dateDerniereLecture;
|
||||
}
|
||||
|
||||
// Getters/Setters pour les propriétés numériques
|
||||
public Integer getPriorite() {
|
||||
return priorite;
|
||||
}
|
||||
|
||||
public void setPriorite(Integer priorite) {
|
||||
this.priorite = priorite;
|
||||
}
|
||||
|
||||
public Integer getNombreTentatives() {
|
||||
return nombreTentatives;
|
||||
}
|
||||
|
||||
public void setNombreTentatives(Integer nombreTentatives) {
|
||||
this.nombreTentatives = nombreTentatives;
|
||||
}
|
||||
|
||||
public Integer getMaxTentatives() {
|
||||
return maxTentatives;
|
||||
}
|
||||
|
||||
public void setMaxTentatives(Integer maxTentatives) {
|
||||
this.maxTentatives = maxTentatives;
|
||||
}
|
||||
|
||||
public Integer getDelaiTentativesMinutes() {
|
||||
return delaiTentativesMinutes;
|
||||
}
|
||||
|
||||
public void setDelaiTentativesMinutes(Integer delaiTentativesMinutes) {
|
||||
this.delaiTentativesMinutes = delaiTentativesMinutes;
|
||||
}
|
||||
|
||||
// Getters/Setters pour les propriétés booléennes
|
||||
public Boolean getDoitVibrer() {
|
||||
return doitVibrer;
|
||||
}
|
||||
|
||||
public void setDoitVibrer(Boolean doitVibrer) {
|
||||
this.doitVibrer = doitVibrer;
|
||||
}
|
||||
|
||||
public Boolean getDoitEmettreSon() {
|
||||
return doitEmettreSon;
|
||||
}
|
||||
|
||||
public void setDoitEmettreSon(Boolean doitEmettreSon) {
|
||||
this.doitEmettreSon = doitEmettreSon;
|
||||
}
|
||||
|
||||
public Boolean getDoitAllumerLED() {
|
||||
return doitAllumerLED;
|
||||
}
|
||||
|
||||
public void setDoitAllumerLED(Boolean doitAllumerLED) {
|
||||
this.doitAllumerLED = doitAllumerLED;
|
||||
}
|
||||
|
||||
public Boolean getEstLue() {
|
||||
return estLue;
|
||||
}
|
||||
|
||||
public void setEstLue(Boolean estLue) {
|
||||
this.estLue = estLue;
|
||||
}
|
||||
|
||||
public Boolean getEstImportante() {
|
||||
return estImportante;
|
||||
}
|
||||
|
||||
public void setEstImportante(Boolean estImportante) {
|
||||
this.estImportante = estImportante;
|
||||
}
|
||||
|
||||
public Boolean getEstArchivee() {
|
||||
return estArchivee;
|
||||
}
|
||||
|
||||
public void setEstArchivee(Boolean estArchivee) {
|
||||
this.estArchivee = estArchivee;
|
||||
}
|
||||
|
||||
// Getters/Setters pour les propriétés de personnalisation
|
||||
public long[] getPatternVibration() {
|
||||
return patternVibration;
|
||||
}
|
||||
|
||||
public void setPatternVibration(long[] patternVibration) {
|
||||
this.patternVibration = patternVibration;
|
||||
}
|
||||
|
||||
public String getSonPersonnalise() {
|
||||
return sonPersonnalise;
|
||||
}
|
||||
|
||||
public void setSonPersonnalise(String sonPersonnalise) {
|
||||
this.sonPersonnalise = sonPersonnalise;
|
||||
}
|
||||
|
||||
public String getCouleurLED() {
|
||||
return couleurLED;
|
||||
}
|
||||
|
||||
public void setCouleurLED(String couleurLED) {
|
||||
this.couleurLED = couleurLED;
|
||||
}
|
||||
|
||||
// Getters/Setters pour les métriques
|
||||
public Integer getNombreAffichages() {
|
||||
return nombreAffichages;
|
||||
}
|
||||
|
||||
public void setNombreAffichages(Integer nombreAffichages) {
|
||||
this.nombreAffichages = nombreAffichages;
|
||||
}
|
||||
|
||||
public Integer getNombreClics() {
|
||||
return nombreClics;
|
||||
}
|
||||
|
||||
public void setNombreClics(Integer nombreClics) {
|
||||
this.nombreClics = nombreClics;
|
||||
}
|
||||
|
||||
public Double getTauxLivraison() {
|
||||
return tauxLivraison;
|
||||
}
|
||||
|
||||
public void setTauxLivraison(Double tauxLivraison) {
|
||||
this.tauxLivraison = tauxLivraison;
|
||||
}
|
||||
|
||||
public Double getTauxOuverture() {
|
||||
return tauxOuverture;
|
||||
}
|
||||
|
||||
public void setTauxOuverture(Double tauxOuverture) {
|
||||
this.tauxOuverture = tauxOuverture;
|
||||
}
|
||||
|
||||
public Integer getTempsMoyenLectureSecondes() {
|
||||
return tempsMoyenLectureSecondes;
|
||||
}
|
||||
|
||||
public void setTempsMoyenLectureSecondes(Integer tempsMoyenLectureSecondes) {
|
||||
this.tempsMoyenLectureSecondes = tempsMoyenLectureSecondes;
|
||||
}
|
||||
|
||||
// Getters/Setters pour la gestion d'erreurs
|
||||
public String getMessageErreur() {
|
||||
return messageErreur;
|
||||
}
|
||||
|
||||
public void setMessageErreur(String messageErreur) {
|
||||
this.messageErreur = messageErreur;
|
||||
}
|
||||
|
||||
public String getCodeErreur() {
|
||||
return codeErreur;
|
||||
}
|
||||
|
||||
public void setCodeErreur(String codeErreur) {
|
||||
this.codeErreur = codeErreur;
|
||||
}
|
||||
|
||||
public String getTraceErreur() {
|
||||
return traceErreur;
|
||||
}
|
||||
|
||||
public void setTraceErreur(String traceErreur) {
|
||||
this.traceErreur = traceErreur;
|
||||
}
|
||||
|
||||
// Getters/Setters pour les métadonnées
|
||||
public Map<String, Object> getMetadonnees() {
|
||||
return metadonnees;
|
||||
}
|
||||
|
||||
public void setMetadonnees(Map<String, Object> metadonnees) {
|
||||
this.metadonnees = metadonnees;
|
||||
}
|
||||
|
||||
public List<String> getTags() {
|
||||
return tags;
|
||||
}
|
||||
|
||||
public void setTags(List<String> tags) {
|
||||
this.tags = tags;
|
||||
}
|
||||
|
||||
public String getCampagneId() {
|
||||
return campagneId;
|
||||
}
|
||||
|
||||
public void setCampagneId(String campagneId) {
|
||||
this.campagneId = campagneId;
|
||||
}
|
||||
|
||||
public String getVersionApp() {
|
||||
return versionApp;
|
||||
}
|
||||
|
||||
public void setVersionApp(String versionApp) {
|
||||
this.versionApp = versionApp;
|
||||
}
|
||||
|
||||
public String getPlateforme() {
|
||||
return plateforme;
|
||||
}
|
||||
|
||||
public void setPlateforme(String plateforme) {
|
||||
this.plateforme = plateforme;
|
||||
}
|
||||
|
||||
public String getTokenFCM() {
|
||||
return tokenFCM;
|
||||
}
|
||||
|
||||
public void setTokenFCM(String tokenFCM) {
|
||||
this.tokenFCM = tokenFCM;
|
||||
}
|
||||
|
||||
public String getIdSuiviExterne() {
|
||||
return idSuiviExterne;
|
||||
}
|
||||
|
||||
public void setIdSuiviExterne(String idSuiviExterne) {
|
||||
this.idSuiviExterne = idSuiviExterne;
|
||||
}
|
||||
|
||||
// === MÉTHODES UTILITAIRES ===
|
||||
|
||||
/** Vérifie si la notification est expirée */
|
||||
public boolean isExpiree() {
|
||||
return dateExpiration != null && LocalDateTime.now().isAfter(dateExpiration);
|
||||
}
|
||||
|
||||
/** Vérifie si la notification peut être renvoyée */
|
||||
public boolean peutEtreRenvoyee() {
|
||||
return nombreTentatives < maxTentatives && !statut.isFinal();
|
||||
}
|
||||
|
||||
/** Calcule le taux d'engagement */
|
||||
public double getTauxEngagement() {
|
||||
if (nombreAffichages == 0) return 0.0;
|
||||
return (double) nombreClics / nombreAffichages * 100;
|
||||
}
|
||||
|
||||
/** Retourne une représentation courte de la notification */
|
||||
@Override
|
||||
public String toString() {
|
||||
return String.format(
|
||||
"NotificationDTO{id='%s', type=%s, statut=%s, titre='%s'}",
|
||||
id, typeNotification, statut, titre);
|
||||
}
|
||||
/** ID du template */
|
||||
private UUID templateId;
|
||||
}
|
||||
|
||||
@@ -0,0 +1,46 @@
|
||||
package dev.lions.unionflow.server.api.dto.notification;
|
||||
|
||||
import dev.lions.unionflow.server.api.dto.base.BaseDTO;
|
||||
import jakarta.validation.constraints.NotBlank;
|
||||
import lombok.Getter;
|
||||
import lombok.Setter;
|
||||
|
||||
/**
|
||||
* DTO pour la gestion des templates de notifications
|
||||
*
|
||||
* @author UnionFlow Team
|
||||
* @version 3.0
|
||||
* @since 2025-01-29
|
||||
*/
|
||||
@Getter
|
||||
@Setter
|
||||
public class TemplateNotificationDTO extends BaseDTO {
|
||||
|
||||
private static final long serialVersionUID = 1L;
|
||||
|
||||
/** Code unique du template */
|
||||
@NotBlank(message = "Le code est obligatoire")
|
||||
private String code;
|
||||
|
||||
/** Sujet du template */
|
||||
private String sujet;
|
||||
|
||||
/** Corps du template (texte) */
|
||||
private String corpsTexte;
|
||||
|
||||
/** Corps du template (HTML) */
|
||||
private String corpsHtml;
|
||||
|
||||
/** Variables disponibles (JSON) */
|
||||
private String variablesDisponibles;
|
||||
|
||||
/** Canaux supportés (JSON array) */
|
||||
private String canauxSupportes;
|
||||
|
||||
/** Langue du template */
|
||||
private String langue;
|
||||
|
||||
/** Description */
|
||||
private String description;
|
||||
}
|
||||
|
||||
@@ -0,0 +1,116 @@
|
||||
package dev.lions.unionflow.server.repository;
|
||||
|
||||
import dev.lions.unionflow.server.api.enums.notification.PrioriteNotification;
|
||||
import dev.lions.unionflow.server.api.enums.notification.StatutNotification;
|
||||
import dev.lions.unionflow.server.api.enums.notification.TypeNotification;
|
||||
import dev.lions.unionflow.server.entity.Notification;
|
||||
import io.quarkus.hibernate.orm.panache.PanacheRepository;
|
||||
import jakarta.enterprise.context.ApplicationScoped;
|
||||
import java.time.LocalDateTime;
|
||||
import java.util.List;
|
||||
import java.util.UUID;
|
||||
|
||||
/**
|
||||
* Repository pour l'entité Notification
|
||||
*
|
||||
* @author UnionFlow Team
|
||||
* @version 3.0
|
||||
* @since 2025-01-29
|
||||
*/
|
||||
@ApplicationScoped
|
||||
public class NotificationRepository implements PanacheRepository<Notification> {
|
||||
|
||||
/**
|
||||
* Trouve toutes les notifications d'un membre
|
||||
*
|
||||
* @param membreId ID du membre
|
||||
* @return Liste des notifications
|
||||
*/
|
||||
public List<Notification> findByMembreId(UUID membreId) {
|
||||
return find("membre.id = ?1 ORDER BY dateEnvoiPrevue DESC, dateCreation DESC", membreId).list();
|
||||
}
|
||||
|
||||
/**
|
||||
* Trouve toutes les notifications non lues d'un membre
|
||||
*
|
||||
* @param membreId ID du membre
|
||||
* @return Liste des notifications non lues
|
||||
*/
|
||||
public List<Notification> findNonLuesByMembreId(UUID membreId) {
|
||||
return find(
|
||||
"membre.id = ?1 AND statut = ?2 ORDER BY priorite ASC, dateEnvoiPrevue DESC",
|
||||
membreId,
|
||||
StatutNotification.NON_LUE)
|
||||
.list();
|
||||
}
|
||||
|
||||
/**
|
||||
* Trouve toutes les notifications d'une organisation
|
||||
*
|
||||
* @param organisationId ID de l'organisation
|
||||
* @return Liste des notifications
|
||||
*/
|
||||
public List<Notification> findByOrganisationId(UUID organisationId) {
|
||||
return find("organisation.id = ?1 ORDER BY dateEnvoiPrevue DESC, dateCreation DESC", organisationId)
|
||||
.list();
|
||||
}
|
||||
|
||||
/**
|
||||
* Trouve les notifications par type
|
||||
*
|
||||
* @param type Type de notification
|
||||
* @return Liste des notifications
|
||||
*/
|
||||
public List<Notification> findByType(TypeNotification type) {
|
||||
return find("typeNotification = ?1 ORDER BY dateEnvoiPrevue DESC", type).list();
|
||||
}
|
||||
|
||||
/**
|
||||
* Trouve les notifications par statut
|
||||
*
|
||||
* @param statut Statut de la notification
|
||||
* @return Liste des notifications
|
||||
*/
|
||||
public List<Notification> findByStatut(StatutNotification statut) {
|
||||
return find("statut = ?1 ORDER BY dateEnvoiPrevue DESC", statut).list();
|
||||
}
|
||||
|
||||
/**
|
||||
* Trouve les notifications par priorité
|
||||
*
|
||||
* @param priorite Priorité de la notification
|
||||
* @return Liste des notifications
|
||||
*/
|
||||
public List<Notification> findByPriorite(PrioriteNotification priorite) {
|
||||
return find("priorite = ?1 ORDER BY dateEnvoiPrevue DESC", priorite).list();
|
||||
}
|
||||
|
||||
/**
|
||||
* Trouve les notifications en attente d'envoi
|
||||
*
|
||||
* @return Liste des notifications en attente
|
||||
*/
|
||||
public List<Notification> findEnAttenteEnvoi() {
|
||||
LocalDateTime maintenant = LocalDateTime.now();
|
||||
return find(
|
||||
"statut IN (?1, ?2) AND dateEnvoiPrevue <= ?3 ORDER BY priorite DESC, dateEnvoiPrevue ASC",
|
||||
StatutNotification.EN_ATTENTE,
|
||||
StatutNotification.PROGRAMMEE,
|
||||
maintenant)
|
||||
.list();
|
||||
}
|
||||
|
||||
/**
|
||||
* Trouve les notifications échouées pouvant être retentées
|
||||
*
|
||||
* @return Liste des notifications échouées
|
||||
*/
|
||||
public List<Notification> findEchoueesRetentables() {
|
||||
return find(
|
||||
"statut IN (?1, ?2) AND (nombreTentatives IS NULL OR nombreTentatives < 5) ORDER BY dateEnvoiPrevue ASC",
|
||||
StatutNotification.ECHEC_ENVOI,
|
||||
StatutNotification.ERREUR_TECHNIQUE)
|
||||
.list();
|
||||
}
|
||||
}
|
||||
|
||||
@@ -0,0 +1,48 @@
|
||||
package dev.lions.unionflow.server.repository;
|
||||
|
||||
import dev.lions.unionflow.server.entity.TemplateNotification;
|
||||
import io.quarkus.hibernate.orm.panache.PanacheRepository;
|
||||
import jakarta.enterprise.context.ApplicationScoped;
|
||||
import java.util.List;
|
||||
import java.util.Optional;
|
||||
|
||||
/**
|
||||
* Repository pour l'entité TemplateNotification
|
||||
*
|
||||
* @author UnionFlow Team
|
||||
* @version 3.0
|
||||
* @since 2025-01-29
|
||||
*/
|
||||
@ApplicationScoped
|
||||
public class TemplateNotificationRepository implements PanacheRepository<TemplateNotification> {
|
||||
|
||||
/**
|
||||
* Trouve un template par son code
|
||||
*
|
||||
* @param code Code du template
|
||||
* @return Template ou Optional.empty()
|
||||
*/
|
||||
public Optional<TemplateNotification> findByCode(String code) {
|
||||
return find("code = ?1 AND actif = true", code).firstResultOptional();
|
||||
}
|
||||
|
||||
/**
|
||||
* Trouve tous les templates actifs
|
||||
*
|
||||
* @return Liste des templates actifs
|
||||
*/
|
||||
public List<TemplateNotification> findAllActifs() {
|
||||
return find("actif = true ORDER BY code ASC").list();
|
||||
}
|
||||
|
||||
/**
|
||||
* Trouve les templates par langue
|
||||
*
|
||||
* @param langue Code langue (ex: fr, en)
|
||||
* @return Liste des templates
|
||||
*/
|
||||
public List<TemplateNotification> findByLangue(String langue) {
|
||||
return find("langue = ?1 AND actif = true ORDER BY code ASC", langue).list();
|
||||
}
|
||||
}
|
||||
|
||||
@@ -1,484 +1,297 @@
|
||||
package dev.lions.unionflow.server.service;
|
||||
|
||||
import dev.lions.unionflow.server.api.dto.notification.NotificationDTO;
|
||||
import dev.lions.unionflow.server.api.dto.notification.PreferencesNotificationDTO;
|
||||
import dev.lions.unionflow.server.api.dto.notification.TemplateNotificationDTO;
|
||||
import dev.lions.unionflow.server.api.enums.notification.PrioriteNotification;
|
||||
import dev.lions.unionflow.server.api.enums.notification.StatutNotification;
|
||||
import dev.lions.unionflow.server.api.enums.notification.TypeNotification;
|
||||
import dev.lions.unionflow.server.entity.*;
|
||||
import dev.lions.unionflow.server.repository.*;
|
||||
import dev.lions.unionflow.server.service.KeycloakService;
|
||||
import jakarta.enterprise.context.ApplicationScoped;
|
||||
import jakarta.inject.Inject;
|
||||
import jakarta.transaction.Transactional;
|
||||
import jakarta.ws.rs.NotFoundException;
|
||||
import java.time.LocalDateTime;
|
||||
import java.util.*;
|
||||
import java.util.concurrent.CompletableFuture;
|
||||
import java.util.concurrent.ConcurrentHashMap;
|
||||
import org.eclipse.microprofile.config.inject.ConfigProperty;
|
||||
import java.util.List;
|
||||
import java.util.UUID;
|
||||
import java.util.stream.Collectors;
|
||||
import org.jboss.logging.Logger;
|
||||
|
||||
/**
|
||||
* Service principal de gestion des notifications UnionFlow
|
||||
*
|
||||
* <p>Ce service orchestre l'envoi, la gestion et le suivi des notifications avec intégration
|
||||
* Firebase, templates dynamiques et préférences utilisateur.
|
||||
* Service métier pour la gestion des notifications
|
||||
*
|
||||
* @author UnionFlow Team
|
||||
* @version 1.0
|
||||
* @since 2025-01-16
|
||||
* @version 3.0
|
||||
* @since 2025-01-29
|
||||
*/
|
||||
@ApplicationScoped
|
||||
public class NotificationService {
|
||||
|
||||
private static final Logger LOG = Logger.getLogger(NotificationService.class);
|
||||
|
||||
// @Inject
|
||||
// FirebaseNotificationService firebaseService;
|
||||
@Inject NotificationRepository notificationRepository;
|
||||
|
||||
// @Inject
|
||||
// NotificationTemplateService templateService;
|
||||
@Inject TemplateNotificationRepository templateNotificationRepository;
|
||||
|
||||
// @Inject
|
||||
// PreferencesNotificationService preferencesService;
|
||||
@Inject MembreRepository membreRepository;
|
||||
|
||||
// @Inject
|
||||
// NotificationHistoryService historyService;
|
||||
@Inject OrganisationRepository organisationRepository;
|
||||
|
||||
// @Inject
|
||||
// NotificationSchedulerService schedulerService;
|
||||
|
||||
@ConfigProperty(name = "unionflow.notifications.enabled", defaultValue = "true")
|
||||
boolean notificationsEnabled;
|
||||
|
||||
@ConfigProperty(name = "unionflow.notifications.batch-size", defaultValue = "100")
|
||||
int batchSize;
|
||||
|
||||
@ConfigProperty(name = "unionflow.notifications.retry-attempts", defaultValue = "3")
|
||||
int maxRetryAttempts;
|
||||
|
||||
@ConfigProperty(name = "unionflow.notifications.retry-delay-minutes", defaultValue = "5")
|
||||
int retryDelayMinutes;
|
||||
|
||||
// Cache des préférences utilisateur pour optimiser les performances
|
||||
private final Map<String, PreferencesNotificationDTO> preferencesCache =
|
||||
new ConcurrentHashMap<>();
|
||||
|
||||
// Statistiques en temps réel
|
||||
private final Map<String, Long> statistiques = new ConcurrentHashMap<>();
|
||||
@Inject KeycloakService keycloakService;
|
||||
|
||||
/**
|
||||
* Envoie une notification simple
|
||||
* Crée un nouveau template de notification
|
||||
*
|
||||
* @param notification La notification à envoyer
|
||||
* @return CompletableFuture avec le résultat de l'envoi
|
||||
*/
|
||||
public CompletableFuture<NotificationDTO> envoyerNotification(NotificationDTO notification) {
|
||||
LOG.infof("Envoi de notification: %s", notification.getId());
|
||||
|
||||
return CompletableFuture.supplyAsync(
|
||||
() -> {
|
||||
try {
|
||||
// Validation des données
|
||||
validerNotification(notification);
|
||||
|
||||
// Vérification des préférences utilisateur
|
||||
if (!verifierPreferencesUtilisateur(notification)) {
|
||||
notification.setStatut(StatutNotification.ANNULEE);
|
||||
notification.setMessageErreur("Notification bloquée par les préférences utilisateur");
|
||||
return notification;
|
||||
}
|
||||
|
||||
// Application des templates
|
||||
// notification = templateService.appliquerTemplate(notification);
|
||||
|
||||
// Envoi via Firebase
|
||||
notification.setStatut(StatutNotification.EN_COURS_ENVOI);
|
||||
notification.setDateEnvoi(LocalDateTime.now());
|
||||
|
||||
// Envoi via Firebase (à implémenter quand Firebase sera configuré)
|
||||
boolean succes = false;
|
||||
try {
|
||||
// boolean succes = firebaseService.envoyerNotificationPush(notification);
|
||||
// Pour l'instant, on considère que l'envoi est réussi si la notification est créée
|
||||
succes = true;
|
||||
} catch (Exception e) {
|
||||
LOG.errorf(e, "Erreur lors de l'envoi de la notification via Firebase");
|
||||
succes = false;
|
||||
}
|
||||
|
||||
if (succes) {
|
||||
notification.setStatut(StatutNotification.ENVOYEE);
|
||||
incrementerStatistique("notifications_envoyees");
|
||||
} else {
|
||||
notification.setStatut(StatutNotification.ECHEC_ENVOI);
|
||||
incrementerStatistique("notifications_echec");
|
||||
}
|
||||
|
||||
// Sauvegarde dans l'historique
|
||||
// historyService.sauvegarderNotification(notification);
|
||||
|
||||
return notification;
|
||||
|
||||
} catch (Exception e) {
|
||||
LOG.errorf(e, "Erreur lors de l'envoi de la notification %s", notification.getId());
|
||||
notification.setStatut(StatutNotification.ERREUR_TECHNIQUE);
|
||||
notification.setMessageErreur(e.getMessage());
|
||||
notification.setTraceErreur(Arrays.toString(e.getStackTrace()));
|
||||
incrementerStatistique("notifications_erreur");
|
||||
return notification;
|
||||
}
|
||||
});
|
||||
}
|
||||
|
||||
/**
|
||||
* Envoie une notification à plusieurs destinataires
|
||||
*
|
||||
* @param typeNotification Type de notification
|
||||
* @param titre Titre de la notification
|
||||
* @param message Message de la notification
|
||||
* @param destinatairesIds Liste des IDs des destinataires
|
||||
* @param donneesPersonnalisees Données personnalisées
|
||||
* @return CompletableFuture avec la liste des résultats
|
||||
*/
|
||||
public CompletableFuture<List<NotificationDTO>> envoyerNotificationGroupe(
|
||||
TypeNotification typeNotification,
|
||||
String titre,
|
||||
String message,
|
||||
List<String> destinatairesIds,
|
||||
Map<String, Object> donneesPersonnalisees) {
|
||||
|
||||
LOG.infof("Envoi de notification de groupe: %s destinataires", destinatairesIds.size());
|
||||
|
||||
return CompletableFuture.supplyAsync(
|
||||
() -> {
|
||||
List<NotificationDTO> resultats = new ArrayList<>();
|
||||
|
||||
// Traitement par batch pour optimiser les performances
|
||||
for (int i = 0; i < destinatairesIds.size(); i += batchSize) {
|
||||
int fin = Math.min(i + batchSize, destinatairesIds.size());
|
||||
List<String> batch = destinatairesIds.subList(i, fin);
|
||||
|
||||
List<CompletableFuture<NotificationDTO>> futures =
|
||||
batch.stream()
|
||||
.map(
|
||||
destinataireId -> {
|
||||
NotificationDTO notification =
|
||||
new NotificationDTO(
|
||||
typeNotification, titre, message, List.of(destinataireId));
|
||||
notification.setId(UUID.randomUUID().toString());
|
||||
notification.setDonneesPersonnalisees(donneesPersonnalisees);
|
||||
|
||||
return envoyerNotification(notification);
|
||||
})
|
||||
.toList();
|
||||
|
||||
// Attendre que tous les envois du batch soient terminés
|
||||
CompletableFuture.allOf(futures.toArray(new CompletableFuture[0])).join();
|
||||
|
||||
// Collecter les résultats
|
||||
futures.forEach(
|
||||
future -> {
|
||||
try {
|
||||
resultats.add(future.get());
|
||||
} catch (Exception e) {
|
||||
LOG.errorf(e, "Erreur lors de la récupération du résultat");
|
||||
}
|
||||
});
|
||||
}
|
||||
|
||||
incrementerStatistique("notifications_groupe_envoyees");
|
||||
return resultats;
|
||||
});
|
||||
}
|
||||
|
||||
/**
|
||||
* Programme une notification pour envoi ultérieur
|
||||
*
|
||||
* @param notification La notification à programmer
|
||||
* @param dateEnvoi Date et heure d'envoi programmé
|
||||
* @return La notification programmée
|
||||
* @param templateDTO DTO du template à créer
|
||||
* @return DTO du template créé
|
||||
*/
|
||||
@Transactional
|
||||
public NotificationDTO programmerNotification(
|
||||
NotificationDTO notification, LocalDateTime dateEnvoi) {
|
||||
LOG.infof("Programmation de notification pour: %s", dateEnvoi);
|
||||
public TemplateNotificationDTO creerTemplate(TemplateNotificationDTO templateDTO) {
|
||||
LOG.infof("Création d'un nouveau template: %s", templateDTO.getCode());
|
||||
|
||||
notification.setId(UUID.randomUUID().toString());
|
||||
notification.setStatut(StatutNotification.PROGRAMMEE);
|
||||
notification.setDateEnvoiProgramme(dateEnvoi);
|
||||
notification.setDateCreation(LocalDateTime.now());
|
||||
|
||||
// Validation
|
||||
validerNotification(notification);
|
||||
|
||||
// Sauvegarde
|
||||
// historyService.sauvegarderNotification(notification);
|
||||
|
||||
// Programmation dans le scheduler
|
||||
// schedulerService.programmerNotification(notification);
|
||||
|
||||
incrementerStatistique("notifications_programmees");
|
||||
return notification;
|
||||
}
|
||||
|
||||
/**
|
||||
* Annule une notification programmée
|
||||
*
|
||||
* @param notificationId ID de la notification à annuler
|
||||
* @return true si l'annulation a réussi
|
||||
*/
|
||||
@Transactional
|
||||
public boolean annulerNotificationProgrammee(String notificationId) {
|
||||
LOG.infof("Annulation de notification programmée: %s", notificationId);
|
||||
|
||||
try {
|
||||
// À implémenter quand les services seront configurés
|
||||
// NotificationDTO notification = historyService.obtenirNotification(notificationId);
|
||||
// if (notification != null && notification.getStatut().permetAnnulation()) {
|
||||
// notification.setStatut(StatutNotification.ANNULEE);
|
||||
// historyService.mettreAJourNotification(notification);
|
||||
// schedulerService.annulerNotificationProgrammee(notificationId);
|
||||
// incrementerStatistique("notifications_annulees");
|
||||
// return true;
|
||||
// }
|
||||
|
||||
incrementerStatistique("notifications_annulees");
|
||||
return true;
|
||||
|
||||
} catch (Exception e) {
|
||||
LOG.errorf(e, "Erreur lors de l'annulation de la notification %s", notificationId);
|
||||
return false;
|
||||
// Vérifier l'unicité du code
|
||||
if (templateNotificationRepository.findByCode(templateDTO.getCode()).isPresent()) {
|
||||
throw new IllegalArgumentException("Un template avec ce code existe déjà: " + templateDTO.getCode());
|
||||
}
|
||||
|
||||
TemplateNotification template = convertToEntity(templateDTO);
|
||||
template.setCreePar(keycloakService.getCurrentUserEmail());
|
||||
|
||||
templateNotificationRepository.persist(template);
|
||||
LOG.infof("Template créé avec succès: ID=%s, Code=%s", template.getId(), template.getCode());
|
||||
|
||||
return convertToDTO(template);
|
||||
}
|
||||
|
||||
/**
|
||||
* Crée une nouvelle notification
|
||||
*
|
||||
* @param notificationDTO DTO de la notification à créer
|
||||
* @return DTO de la notification créée
|
||||
*/
|
||||
@Transactional
|
||||
public NotificationDTO creerNotification(NotificationDTO notificationDTO) {
|
||||
LOG.infof("Création d'une nouvelle notification: %s", notificationDTO.getTypeNotification());
|
||||
|
||||
Notification notification = convertToEntity(notificationDTO);
|
||||
notification.setCreePar(keycloakService.getCurrentUserEmail());
|
||||
|
||||
notificationRepository.persist(notification);
|
||||
LOG.infof("Notification créée avec succès: ID=%s", notification.getId());
|
||||
|
||||
return convertToDTO(notification);
|
||||
}
|
||||
|
||||
/**
|
||||
* Marque une notification comme lue
|
||||
*
|
||||
* @param notificationId ID de la notification
|
||||
* @param utilisateurId ID de l'utilisateur
|
||||
* @return true si le marquage a réussi
|
||||
* @param id ID de la notification
|
||||
* @return DTO de la notification mise à jour
|
||||
*/
|
||||
@Transactional
|
||||
public boolean marquerCommeLue(String notificationId, String utilisateurId) {
|
||||
LOG.debugf(
|
||||
"Marquage comme lue: notification=%s, utilisateur=%s", notificationId, utilisateurId);
|
||||
public NotificationDTO marquerCommeLue(UUID id) {
|
||||
LOG.infof("Marquage de la notification comme lue: ID=%s", id);
|
||||
|
||||
try {
|
||||
// À implémenter quand les services seront configurés
|
||||
// NotificationDTO notification = historyService.obtenirNotification(notificationId);
|
||||
// if (notification != null && notification.getDestinatairesIds().contains(utilisateurId)) {
|
||||
// notification.setEstLue(true);
|
||||
// notification.setDateDerniereLecture(LocalDateTime.now());
|
||||
// notification.setStatut(StatutNotification.LUE);
|
||||
// historyService.mettreAJourNotification(notification);
|
||||
// incrementerStatistique("notifications_lues");
|
||||
// return true;
|
||||
// }
|
||||
Notification notification =
|
||||
notificationRepository
|
||||
.findByIdOptional(id)
|
||||
.orElseThrow(() -> new NotFoundException("Notification non trouvée avec l'ID: " + id));
|
||||
|
||||
incrementerStatistique("notifications_lues");
|
||||
return true;
|
||||
notification.setStatut(StatutNotification.LUE);
|
||||
notification.setDateLecture(LocalDateTime.now());
|
||||
notification.setModifiePar(keycloakService.getCurrentUserEmail());
|
||||
|
||||
} catch (Exception e) {
|
||||
LOG.errorf(e, "Erreur lors du marquage comme lue: %s", notificationId);
|
||||
return false;
|
||||
}
|
||||
notificationRepository.persist(notification);
|
||||
LOG.infof("Notification marquée comme lue: ID=%s", id);
|
||||
|
||||
return convertToDTO(notification);
|
||||
}
|
||||
|
||||
/**
|
||||
* Archive une notification
|
||||
* Trouve une notification par son ID
|
||||
*
|
||||
* @param notificationId ID de la notification
|
||||
* @param utilisateurId ID de l'utilisateur
|
||||
* @return true si l'archivage a réussi
|
||||
* @param id ID de la notification
|
||||
* @return DTO de la notification
|
||||
*/
|
||||
@Transactional
|
||||
public boolean archiverNotification(String notificationId, String utilisateurId) {
|
||||
LOG.debugf("Archivage: notification=%s, utilisateur=%s", notificationId, utilisateurId);
|
||||
|
||||
try {
|
||||
// À implémenter quand les services seront configurés
|
||||
// NotificationDTO notification = historyService.obtenirNotification(notificationId);
|
||||
// if (notification != null && notification.getDestinatairesIds().contains(utilisateurId)) {
|
||||
// notification.setEstArchivee(true);
|
||||
// notification.setStatut(StatutNotification.ARCHIVEE);
|
||||
// historyService.mettreAJourNotification(notification);
|
||||
// incrementerStatistique("notifications_archivees");
|
||||
// return true;
|
||||
// }
|
||||
|
||||
incrementerStatistique("notifications_archivees");
|
||||
return true;
|
||||
|
||||
} catch (Exception e) {
|
||||
LOG.errorf(e, "Erreur lors de l'archivage: %s", notificationId);
|
||||
return false;
|
||||
}
|
||||
public NotificationDTO trouverNotificationParId(UUID id) {
|
||||
return notificationRepository
|
||||
.findByIdOptional(id)
|
||||
.map(this::convertToDTO)
|
||||
.orElseThrow(() -> new NotFoundException("Notification non trouvée avec l'ID: " + id));
|
||||
}
|
||||
|
||||
/**
|
||||
* Obtient les notifications d'un utilisateur
|
||||
* Liste toutes les notifications d'un membre
|
||||
*
|
||||
* @param utilisateurId ID de l'utilisateur
|
||||
* @param includeArchivees Inclure les notifications archivées
|
||||
* @param limite Nombre maximum de notifications à retourner
|
||||
* @param membreId ID du membre
|
||||
* @return Liste des notifications
|
||||
*/
|
||||
public List<NotificationDTO> obtenirNotificationsUtilisateur(
|
||||
String utilisateurId, boolean includeArchivees, int limite) {
|
||||
|
||||
LOG.debugf("Récupération notifications utilisateur: %s", utilisateurId);
|
||||
|
||||
try {
|
||||
// À implémenter quand les services seront configurés
|
||||
// return historyService.obtenirNotificationsUtilisateur(
|
||||
// utilisateurId, includeArchivees, limite
|
||||
// );
|
||||
|
||||
return new ArrayList<>();
|
||||
} catch (Exception e) {
|
||||
LOG.errorf(e, "Erreur lors de la récupération des notifications pour %s", utilisateurId);
|
||||
return new ArrayList<>();
|
||||
}
|
||||
public List<NotificationDTO> listerNotificationsParMembre(UUID membreId) {
|
||||
return notificationRepository.findByMembreId(membreId).stream()
|
||||
.map(this::convertToDTO)
|
||||
.collect(Collectors.toList());
|
||||
}
|
||||
|
||||
/**
|
||||
* Obtient les statistiques des notifications
|
||||
* Liste les notifications non lues d'un membre
|
||||
*
|
||||
* @return Map des statistiques
|
||||
* @param membreId ID du membre
|
||||
* @return Liste des notifications non lues
|
||||
*/
|
||||
public Map<String, Long> obtenirStatistiques() {
|
||||
Map<String, Long> stats = new HashMap<>(statistiques);
|
||||
|
||||
// Ajout des statistiques calculées
|
||||
stats.put(
|
||||
"notifications_total",
|
||||
stats.getOrDefault("notifications_envoyees", 0L)
|
||||
+ stats.getOrDefault("notifications_echec", 0L)
|
||||
+ stats.getOrDefault("notifications_erreur", 0L));
|
||||
|
||||
long envoyees = stats.getOrDefault("notifications_envoyees", 0L);
|
||||
long total = stats.get("notifications_total");
|
||||
|
||||
if (total > 0) {
|
||||
stats.put("taux_succes_pct", (envoyees * 100) / total);
|
||||
}
|
||||
|
||||
return stats;
|
||||
public List<NotificationDTO> listerNotificationsNonLuesParMembre(UUID membreId) {
|
||||
return notificationRepository.findNonLuesByMembreId(membreId).stream()
|
||||
.map(this::convertToDTO)
|
||||
.collect(Collectors.toList());
|
||||
}
|
||||
|
||||
/**
|
||||
* Envoie une notification de test
|
||||
* Liste les notifications en attente d'envoi
|
||||
*
|
||||
* @param utilisateurId ID de l'utilisateur
|
||||
* @param typeNotification Type de notification à tester
|
||||
* @return La notification de test envoyée
|
||||
* @return Liste des notifications en attente
|
||||
*/
|
||||
public CompletableFuture<NotificationDTO> envoyerNotificationTest(
|
||||
String utilisateurId, TypeNotification typeNotification) {
|
||||
|
||||
LOG.infof(
|
||||
"Envoi notification de test: utilisateur=%s, type=%s", utilisateurId, typeNotification);
|
||||
|
||||
NotificationDTO notification =
|
||||
new NotificationDTO(
|
||||
typeNotification,
|
||||
"Test - " + typeNotification.getLibelle(),
|
||||
"Ceci est une notification de test pour vérifier vos paramètres.",
|
||||
List.of(utilisateurId));
|
||||
|
||||
notification.setId("test-" + UUID.randomUUID().toString());
|
||||
notification.getDonneesPersonnalisees().put("test", true);
|
||||
notification.getTags().add("test");
|
||||
|
||||
return envoyerNotification(notification);
|
||||
public List<NotificationDTO> listerNotificationsEnAttenteEnvoi() {
|
||||
return notificationRepository.findEnAttenteEnvoi().stream()
|
||||
.map(this::convertToDTO)
|
||||
.collect(Collectors.toList());
|
||||
}
|
||||
|
||||
// === MÉTHODES PRIVÉES ===
|
||||
// ========================================
|
||||
// MÉTHODES PRIVÉES
|
||||
// ========================================
|
||||
|
||||
/** Valide une notification avant envoi */
|
||||
private void validerNotification(NotificationDTO notification) {
|
||||
if (notification.getTitre() == null || notification.getTitre().trim().isEmpty()) {
|
||||
throw new IllegalArgumentException("Le titre de la notification est obligatoire");
|
||||
/** Convertit une entité TemplateNotification en DTO */
|
||||
private TemplateNotificationDTO convertToDTO(TemplateNotification template) {
|
||||
if (template == null) {
|
||||
return null;
|
||||
}
|
||||
|
||||
if (notification.getMessage() == null || notification.getMessage().trim().isEmpty()) {
|
||||
throw new IllegalArgumentException("Le message de la notification est obligatoire");
|
||||
}
|
||||
TemplateNotificationDTO dto = new TemplateNotificationDTO();
|
||||
dto.setId(template.getId());
|
||||
dto.setCode(template.getCode());
|
||||
dto.setSujet(template.getSujet());
|
||||
dto.setCorpsTexte(template.getCorpsTexte());
|
||||
dto.setCorpsHtml(template.getCorpsHtml());
|
||||
dto.setVariablesDisponibles(template.getVariablesDisponibles());
|
||||
dto.setCanauxSupportes(template.getCanauxSupportes());
|
||||
dto.setLangue(template.getLangue());
|
||||
dto.setDescription(template.getDescription());
|
||||
dto.setDateCreation(template.getDateCreation());
|
||||
dto.setDateModification(template.getDateModification());
|
||||
dto.setActif(template.getActif());
|
||||
|
||||
if (notification.getDestinatairesIds() == null
|
||||
|| notification.getDestinatairesIds().isEmpty()) {
|
||||
throw new IllegalArgumentException("Au moins un destinataire est requis");
|
||||
}
|
||||
|
||||
if (notification.getTypeNotification() == null) {
|
||||
throw new IllegalArgumentException("Le type de notification est obligatoire");
|
||||
}
|
||||
return dto;
|
||||
}
|
||||
|
||||
/** Vérifie les préférences utilisateur pour une notification */
|
||||
private boolean verifierPreferencesUtilisateur(NotificationDTO notification) {
|
||||
if (!notificationsEnabled) {
|
||||
return false;
|
||||
/** Convertit un DTO en entité TemplateNotification */
|
||||
private TemplateNotification convertToEntity(TemplateNotificationDTO dto) {
|
||||
if (dto == null) {
|
||||
return null;
|
||||
}
|
||||
|
||||
// Vérification pour chaque destinataire
|
||||
for (String destinataireId : notification.getDestinatairesIds()) {
|
||||
PreferencesNotificationDTO preferences = obtenirPreferencesUtilisateur(destinataireId);
|
||||
TemplateNotification template = new TemplateNotification();
|
||||
template.setCode(dto.getCode());
|
||||
template.setSujet(dto.getSujet());
|
||||
template.setCorpsTexte(dto.getCorpsTexte());
|
||||
template.setCorpsHtml(dto.getCorpsHtml());
|
||||
template.setVariablesDisponibles(dto.getVariablesDisponibles());
|
||||
template.setCanauxSupportes(dto.getCanauxSupportes());
|
||||
template.setLangue(dto.getLangue() != null ? dto.getLangue() : "fr");
|
||||
template.setDescription(dto.getDescription());
|
||||
|
||||
if (preferences == null || !preferences.getNotificationsActivees()) {
|
||||
return false;
|
||||
}
|
||||
return template;
|
||||
}
|
||||
|
||||
if (!preferences.isTypeActive(notification.getTypeNotification())) {
|
||||
return false;
|
||||
}
|
||||
|
||||
if (!preferences.isCanalActif(notification.getCanal())) {
|
||||
return false;
|
||||
}
|
||||
|
||||
if (preferences.isExpediteurBloque(notification.getExpediteurId())) {
|
||||
return false;
|
||||
}
|
||||
|
||||
if (preferences.isEnModeSilencieux()
|
||||
&& !notification.getTypeNotification().isCritique()
|
||||
&& !preferences.getUrgentesIgnorentSilencieux()) {
|
||||
return false;
|
||||
}
|
||||
/** Convertit une entité Notification en DTO */
|
||||
private NotificationDTO convertToDTO(Notification notification) {
|
||||
if (notification == null) {
|
||||
return null;
|
||||
}
|
||||
|
||||
return true;
|
||||
NotificationDTO dto = new NotificationDTO();
|
||||
dto.setId(notification.getId());
|
||||
dto.setTypeNotification(notification.getTypeNotification());
|
||||
dto.setPriorite(notification.getPriorite());
|
||||
dto.setStatut(notification.getStatut());
|
||||
dto.setSujet(notification.getSujet());
|
||||
dto.setCorps(notification.getCorps());
|
||||
dto.setDateEnvoiPrevue(notification.getDateEnvoiPrevue());
|
||||
dto.setDateEnvoi(notification.getDateEnvoi());
|
||||
dto.setDateLecture(notification.getDateLecture());
|
||||
dto.setNombreTentatives(notification.getNombreTentatives());
|
||||
dto.setMessageErreur(notification.getMessageErreur());
|
||||
dto.setDonneesAdditionnelles(notification.getDonneesAdditionnelles());
|
||||
|
||||
if (notification.getMembre() != null) {
|
||||
dto.setMembreId(notification.getMembre().getId());
|
||||
}
|
||||
if (notification.getOrganisation() != null) {
|
||||
dto.setOrganisationId(notification.getOrganisation().getId());
|
||||
}
|
||||
if (notification.getTemplate() != null) {
|
||||
dto.setTemplateId(notification.getTemplate().getId());
|
||||
}
|
||||
|
||||
dto.setDateCreation(notification.getDateCreation());
|
||||
dto.setDateModification(notification.getDateModification());
|
||||
dto.setActif(notification.getActif());
|
||||
|
||||
return dto;
|
||||
}
|
||||
|
||||
/** Obtient les préférences d'un utilisateur (avec cache) */
|
||||
private PreferencesNotificationDTO obtenirPreferencesUtilisateur(String utilisateurId) {
|
||||
return preferencesCache.computeIfAbsent(
|
||||
utilisateurId,
|
||||
id -> {
|
||||
try {
|
||||
// Note: Les préférences sont actuellement initialisées avec des valeurs par défaut.
|
||||
// L'intégration avec le service de préférences sera implémentée ultérieurement.
|
||||
// return preferencesService.obtenirPreferences(id);
|
||||
return new PreferencesNotificationDTO(id);
|
||||
} catch (Exception e) {
|
||||
LOG.warnf(
|
||||
"Impossible de récupérer les préférences pour %s, utilisation des défauts", id);
|
||||
return new PreferencesNotificationDTO(id);
|
||||
}
|
||||
});
|
||||
}
|
||||
/** Convertit un DTO en entité Notification */
|
||||
private Notification convertToEntity(NotificationDTO dto) {
|
||||
if (dto == null) {
|
||||
return null;
|
||||
}
|
||||
|
||||
/** Incrémente une statistique */
|
||||
private void incrementerStatistique(String cle) {
|
||||
statistiques.merge(cle, 1L, Long::sum);
|
||||
}
|
||||
Notification notification = new Notification();
|
||||
notification.setTypeNotification(dto.getTypeNotification());
|
||||
notification.setPriorite(
|
||||
dto.getPriorite() != null ? dto.getPriorite() : PrioriteNotification.NORMALE);
|
||||
notification.setStatut(
|
||||
dto.getStatut() != null ? dto.getStatut() : StatutNotification.EN_ATTENTE);
|
||||
notification.setSujet(dto.getSujet());
|
||||
notification.setCorps(dto.getCorps());
|
||||
notification.setDateEnvoiPrevue(
|
||||
dto.getDateEnvoiPrevue() != null ? dto.getDateEnvoiPrevue() : LocalDateTime.now());
|
||||
notification.setDateEnvoi(dto.getDateEnvoi());
|
||||
notification.setDateLecture(dto.getDateLecture());
|
||||
notification.setNombreTentatives(dto.getNombreTentatives() != null ? dto.getNombreTentatives() : 0);
|
||||
notification.setMessageErreur(dto.getMessageErreur());
|
||||
notification.setDonneesAdditionnelles(dto.getDonneesAdditionnelles());
|
||||
|
||||
/** Vide le cache des préférences */
|
||||
public void viderCachePreferences() {
|
||||
preferencesCache.clear();
|
||||
LOG.info("Cache des préférences vidé");
|
||||
}
|
||||
// Relations
|
||||
if (dto.getMembreId() != null) {
|
||||
Membre membre =
|
||||
membreRepository
|
||||
.findByIdOptional(dto.getMembreId())
|
||||
.orElseThrow(
|
||||
() -> new NotFoundException("Membre non trouvé avec l'ID: " + dto.getMembreId()));
|
||||
notification.setMembre(membre);
|
||||
}
|
||||
|
||||
/** Recharge les préférences d'un utilisateur */
|
||||
public void rechargerPreferencesUtilisateur(String utilisateurId) {
|
||||
preferencesCache.remove(utilisateurId);
|
||||
LOG.debugf("Préférences rechargées pour l'utilisateur: %s", utilisateurId);
|
||||
if (dto.getOrganisationId() != null) {
|
||||
Organisation org =
|
||||
organisationRepository
|
||||
.findByIdOptional(dto.getOrganisationId())
|
||||
.orElseThrow(
|
||||
() ->
|
||||
new NotFoundException(
|
||||
"Organisation non trouvée avec l'ID: " + dto.getOrganisationId()));
|
||||
notification.setOrganisation(org);
|
||||
}
|
||||
|
||||
if (dto.getTemplateId() != null) {
|
||||
TemplateNotification template =
|
||||
templateNotificationRepository
|
||||
.findByIdOptional(dto.getTemplateId())
|
||||
.orElseThrow(
|
||||
() ->
|
||||
new NotFoundException(
|
||||
"Template non trouvé avec l'ID: " + dto.getTemplateId()));
|
||||
notification.setTemplate(template);
|
||||
}
|
||||
|
||||
return notification;
|
||||
}
|
||||
}
|
||||
|
||||
Reference in New Issue
Block a user