demarches-normaliennes/app/services/expired/dossiers_deletion_service.rb
2024-08-22 09:26:48 +02:00

144 lines
5 KiB
Ruby

# frozen_string_literal: true
class Expired::DossiersDeletionService < Expired::MailRateLimiter
def process_expired_dossiers_brouillon
send_brouillon_expiration_notices
delete_expired_brouillons_and_notify
end
def process_expired_dossiers_en_construction
send_en_construction_expiration_notices
delete_expired_en_construction_and_notify
end
def process_expired_dossiers_termine
send_termine_expiration_notices
delete_expired_termine_and_notify
end
def send_brouillon_expiration_notices
dossiers_close_to_expiration = Dossier
.brouillon_close_to_expiration
.without_brouillon_expiration_notice_sent
user_notifications = group_by_user_email(dossiers_close_to_expiration)
dossiers_close_to_expiration.in_batches.update_all(brouillon_close_to_expiration_notice_sent_at: Time.zone.now)
user_notifications.each do |(email, dossiers)|
mail = DossierMailer.notify_brouillon_near_deletion(
dossiers,
email
)
send_with_delay(mail)
end
end
def send_en_construction_expiration_notices
send_expiration_notices(
Dossier.en_construction_close_to_expiration.without_en_construction_expiration_notice_sent,
:en_construction_close_to_expiration_notice_sent_at
)
end
def send_termine_expiration_notices
send_expiration_notices(
Dossier.termine_close_to_expiration.without_termine_expiration_notice_sent,
:termine_close_to_expiration_notice_sent_at
)
end
def delete_expired_brouillons_and_notify
user_notifications = group_by_user_email(Dossier.brouillon_expired)
.map { |(email, dossiers)| [email, dossiers.map(&:hash_for_deletion_mail)] }
Dossier.brouillon_expired.in_batches.destroy_all
user_notifications.each do |(email, dossiers_hash)|
mail = DossierMailer.notify_brouillon_deletion(
dossiers_hash,
email
)
send_with_delay(mail)
end
end
def delete_expired_en_construction_and_notify
delete_expired_and_notify(Dossier.en_construction_expired)
end
def delete_expired_termine_and_notify
delete_expired_and_notify(Dossier.termine_expired, notify_on_closed_procedures_to_user: true)
end
private
def send_expiration_notices(dossiers_close_to_expiration, close_to_expiration_flag)
user_notifications = group_by_user_email(dossiers_close_to_expiration)
administration_notifications = group_by_fonctionnaire_email(dossiers_close_to_expiration)
dossiers_close_to_expiration.in_batches.update_all(close_to_expiration_flag => Time.zone.now)
user_notifications.each do |(email, dossiers)|
mail = DossierMailer.notify_near_deletion_to_user(dossiers, email)
send_with_delay(mail)
end
administration_notifications.each do |(email, dossiers)|
mail = DossierMailer.notify_near_deletion_to_administration(dossiers, email)
send_with_delay(mail)
end
end
def delete_expired_and_notify(dossiers_to_remove, notify_on_closed_procedures_to_user: false)
user_notifications = group_by_user_email(dossiers_to_remove, notify_on_closed_procedures_to_user: notify_on_closed_procedures_to_user)
.map { |(email, dossiers)| [email, dossiers.map(&:id)] }
administration_notifications = group_by_fonctionnaire_email(dossiers_to_remove)
.map { |(email, dossiers)| [email, dossiers.map(&:id)] }
hidden_dossier_ids = []
dossiers_to_remove.find_each do |dossier|
dossier.hide_and_keep_track!(:automatic, :expired)
hidden_dossier_ids << dossier.id
end
user_notifications.each do |(email, dossier_ids)|
dossier_ids = dossier_ids.intersection(hidden_dossier_ids)
if dossier_ids.present?
mail = DossierMailer.notify_automatic_deletion_to_user(
Dossier.where(id: dossier_ids).to_a,
email
)
send_with_delay(mail)
end
end
administration_notifications.each do |(email, dossier_ids)|
dossier_ids = dossier_ids.intersection(hidden_dossier_ids)
if dossier_ids.present?
mail = DossierMailer.notify_automatic_deletion_to_administration(
Dossier.where(id: dossier_ids).to_a,
email
)
send_with_delay(mail)
end
end
end
def group_by_user_email(dossiers, notify_on_closed_procedures_to_user: false)
dossiers
.visible_by_user
.with_notifiable_procedure(notify_on_closed: notify_on_closed_procedures_to_user)
.includes(:user, :procedure)
.group_by(&:user)
.map { |(user, dossiers)| [user.email, dossiers] }
end
def group_by_fonctionnaire_email(dossiers)
dossiers
.visible_by_administration
.with_notifiable_procedure(notify_on_closed: true)
.includes(:followers_instructeurs, procedure: [:administrateurs])
.each_with_object(Hash.new { |h, k| h[k] = Set.new }) do |dossier, h|
(dossier.followers_instructeurs + dossier.procedure.administrateurs).each { |destinataire| h[destinataire.email] << dossier }
end
.map { |(email, dossiers)| [email, dossiers.to_a] }
end
end