demarches-normaliennes/app/controllers/instructeurs/procedures_controller.rb

431 lines
16 KiB
Ruby
Raw Normal View History

# frozen_string_literal: true
module Instructeurs
class ProceduresController < InstructeurController
before_action :ensure_ownership!, except: [:index]
before_action :ensure_not_super_admin!, only: [:download_export, :exports]
ITEMS_PER_PAGE = 100
2023-01-23 09:42:35 +01:00
BATCH_SELECTION_LIMIT = 500
2017-09-27 15:16:07 +02:00
def index
2023-07-24 10:48:32 +02:00
all_procedures = current_instructeur
.procedures
.kept
all_procedures_for_listing = all_procedures
2019-09-17 11:11:08 +02:00
.with_attached_logo
dossiers = current_instructeur.dossiers
.joins(groupe_instructeur: :procedure)
.where(procedures: { hidden_at: nil })
# .uniq is much more faster than a distinct on a joint column
procedures_dossiers_en_cours = dossiers.joins(:revision).en_cours.pluck(ProcedureRevision.arel_table[:procedure_id]).uniq
2023-07-24 10:48:32 +02:00
@procedures = all_procedures.order(closed_at: :desc, unpublished_at: :desc, published_at: :desc, created_at: :desc)
publiees_or_closes_with_dossiers_en_cours = all_procedures_for_listing.publiees.or(all_procedures.closes.where(id: procedures_dossiers_en_cours))
@procedures_en_cours = publiees_or_closes_with_dossiers_en_cours.order(published_at: :desc).page(params[:page]).per(ITEMS_PER_PAGE)
closes_with_no_dossier_en_cours = all_procedures.closes.excluding(all_procedures.closes.where(id: procedures_dossiers_en_cours))
@procedures_closes = closes_with_no_dossier_en_cours.order(created_at: :desc).page(params[:page]).per(ITEMS_PER_PAGE)
@procedures_draft = all_procedures_for_listing.brouillons.order(created_at: :desc).page(params[:page]).per(ITEMS_PER_PAGE)
@procedures_en_cours_count = publiees_or_closes_with_dossiers_en_cours.count
@procedures_draft_count = all_procedures_for_listing.brouillons.count
@procedures_closes_count = closes_with_no_dossier_en_cours.count
@dossiers_count_per_procedure = dossiers.by_statut('tous').group('groupe_instructeurs.procedure_id').reorder(nil).count
@dossiers_a_suivre_count_per_procedure = dossiers.by_statut('a-suivre').group('groupe_instructeurs.procedure_id').reorder(nil).count
@dossiers_archived_count_per_procedure = dossiers.by_statut('archives').group('groupe_instructeurs.procedure_id').count
@dossiers_termines_count_per_procedure = dossiers.by_statut('traites').group('groupe_instructeurs.procedure_id').reorder(nil).count
@dossiers_expirant_count_per_procedure = dossiers.by_statut('expirant').group('groupe_instructeurs.procedure_id').count
@dossiers_supprimes_count_per_procedure = dossiers.by_statut('supprimes').group('groupe_instructeurs.procedure_id').reorder(nil).count
groupe_ids = current_instructeur.groupe_instructeurs.pluck(:id)
@followed_dossiers_count_per_procedure = current_instructeur
.followed_dossiers
.joins(:groupe_instructeur)
.en_cours
.where(groupe_instructeur_id: groupe_ids)
.visible_by_administration
.group('groupe_instructeurs.procedure_id')
.reorder(nil)
.count
2020-09-10 12:10:38 +02:00
@all_dossiers_counts = {
2022-04-13 15:44:22 +02:00
t('.to_follow') => @dossiers_a_suivre_count_per_procedure.sum { |_, v| v },
t('.followed') => @followed_dossiers_count_per_procedure.sum { |_, v| v },
t('.processed') => @dossiers_termines_count_per_procedure.sum { |_, v| v },
t('.all') => @dossiers_count_per_procedure.sum { |_, v| v },
t('.dossiers_close_to_expiration') => @dossiers_expirant_count_per_procedure.sum { |_, v| v },
t('.archived') => @dossiers_archived_count_per_procedure.sum { |_, v| v },
t('.dossiers_supprimes') => @dossiers_supprimes_count_per_procedure.sum { |_, v| v }
2020-09-10 12:10:38 +02:00
}
@procedure_ids_en_cours_with_notifications = current_instructeur.procedure_ids_with_notifications(:en_cours)
@procedure_ids_termines_with_notifications = current_instructeur.procedure_ids_with_notifications(:termine)
@statut = params[:statut]
@statut.blank? ? @statut = 'en-cours' : @statut = params[:statut]
end
2017-07-11 16:09:03 +02:00
def show
@procedure = procedure
# Technically, procedure_presentation already sets the attribute.
# Setting it here to make clear that it is used by the view
@procedure_presentation = procedure_presentation
2024-09-27 13:09:59 +02:00
@current_filters = procedure_presentation.filters_for(statut)
@counts = current_instructeur
.dossiers_count_summary(groupe_instructeur_ids)
.symbolize_keys
2022-07-28 15:12:24 +02:00
@can_download_dossiers = (@counts[:tous] + @counts[:archives]) > 0 && !instructeur_as_manager?
dossiers = Dossier.where(groupe_instructeur_id: groupe_instructeur_ids)
dossiers_count = @counts[statut.underscore.to_sym]
2017-07-11 16:09:03 +02:00
2022-03-31 16:33:14 +02:00
@followed_dossiers_id = current_instructeur
2017-07-11 16:09:03 +02:00
.followed_dossiers
.en_cours
2022-03-31 16:33:14 +02:00
.merge(dossiers.visible_by_administration)
.pluck(:id)
2021-04-07 18:03:03 +02:00
notifications = current_instructeur.notifications_for_groupe_instructeurs(groupe_instructeur_ids)
@has_en_cours_notifications = notifications[:en_cours].present?
@has_termine_notifications = notifications[:termines].present?
@not_archived_notifications_dossier_ids = notifications[:en_cours] + notifications[:termines]
2020-09-14 13:52:28 +02:00
@has_export_notification = notify_exports?
2023-12-19 14:32:34 +01:00
@last_export = last_export_for(statut)
2023-01-20 11:21:06 +01:00
@filtered_sorted_ids = procedure_presentation.filtered_sorted_ids(dossiers, statut, count: dossiers_count)
2018-03-06 13:44:29 +01:00
page = params[:page].presence || 1
2017-09-27 15:16:07 +02:00
2023-01-20 11:21:06 +01:00
@dossiers_count = @filtered_sorted_ids.size
2021-04-25 21:29:31 +02:00
@filtered_sorted_paginated_ids = Kaminari
2023-01-20 11:21:06 +01:00
.paginate_array(@filtered_sorted_ids)
2017-09-27 15:16:07 +02:00
.page(page)
.per(ITEMS_PER_PAGE)
2021-04-25 21:29:31 +02:00
@projected_dossiers = DossierProjectionService.project(@filtered_sorted_paginated_ids, procedure_presentation.displayed_fields)
@disable_checkbox_all = @projected_dossiers.all? { _1.batch_operation_id.present? }
2019-12-11 17:42:44 +01:00
@batch_operations = BatchOperation.joins(:groupe_instructeurs)
2022-12-02 17:16:29 +01:00
.where(groupe_instructeurs: current_instructeur.groupe_instructeurs.where(procedure_id: @procedure.id))
.where(seen_at: nil)
2022-12-02 17:16:29 +01:00
.distinct
2017-07-11 16:09:03 +02:00
end
def deleted_dossiers
@procedure = procedure
2020-03-24 13:53:15 +01:00
@deleted_dossiers = @procedure
2020-03-31 14:33:30 +02:00
.deleted_dossiers
2020-03-24 13:53:15 +01:00
.order(:dossier_id)
.page params[:page]
2021-09-23 15:03:20 +02:00
@a_suivre_count, @suivis_count, @traites_count, @tous_count, @archives_count, @supprimes_count, @expirant_count = current_instructeur
2021-09-23 15:03:20 +02:00
.dossiers_count_summary(groupe_instructeur_ids)
.fetch_values('a_suivre', 'suivis', 'traites', 'tous', 'archives', 'supprimes', 'expirant')
2022-07-28 15:12:24 +02:00
@can_download_dossiers = (@tous_count + @archives_count) > 0 && !instructeur_as_manager?
2021-09-23 15:03:20 +02:00
notifications = current_instructeur.notifications_for_groupe_instructeurs(groupe_instructeur_ids)
@has_en_cours_notifications = notifications[:en_cours].present?
@has_termine_notifications = notifications[:termines].present?
@statut = 'supprime'
end
def update_displayed_fields
values = (params['values'].presence || []).reject(&:empty?)
2024-05-07 11:31:27 +02:00
procedure_presentation.update_displayed_fields(values)
2017-09-27 15:16:07 +02:00
redirect_back(fallback_location: instructeur_procedure_url(procedure))
2017-09-27 15:16:07 +02:00
end
def update_sort
procedure_presentation.update!(sorted_column_params)
2017-09-27 15:16:07 +02:00
redirect_back(fallback_location: instructeur_procedure_url(procedure))
end
2017-09-28 11:04:18 +02:00
def add_filter
2024-09-27 15:58:34 +02:00
if !procedure_presentation.update(filter_params)
# complicated way to display inner error messages
flash.alert = procedure_presentation.errors
.flat_map { _1.detail[:value].errors.full_messages }
end
redirect_back(fallback_location: instructeur_procedure_url(procedure))
end
def update_filter
@statut = statut
@procedure = procedure
@procedure_presentation = procedure_presentation
2024-09-27 15:58:34 +02:00
current_filter = procedure_presentation.filters_name_for(@statut)
# According to the html, the selected column is the last one
h_id = JSON.parse(params[current_filter].last[:id], symbolize_names: true)
@column = procedure.find_column(h_id:)
2017-09-28 11:04:18 +02:00
end
def remove_filter
2024-08-19 14:34:36 +02:00
procedure_presentation.remove_filter(statut, params[:column], params[:value])
2017-09-28 11:04:18 +02:00
redirect_back(fallback_location: instructeur_procedure_url(procedure))
2017-09-28 11:04:18 +02:00
end
def download_export
2019-12-11 17:43:24 +01:00
groupe_instructeurs = current_instructeur
.groupe_instructeurs
.where(procedure: procedure)
2019-09-30 11:57:21 +02:00
@can_download_dossiers = current_instructeur
.dossiers
.visible_by_administration
2022-07-28 15:12:24 +02:00
.exists?(groupe_instructeur_id: groupe_instructeur_ids) && !instructeur_as_manager?
2021-04-22 08:38:34 +02:00
export = Export.find_or_create_fresh_export(export_format, groupe_instructeurs, current_instructeur, **export_options)
2019-12-11 17:43:24 +01:00
@procedure = procedure
@statut = export_options[:statut]
@dossiers_count = export.count
2021-04-08 12:47:29 +02:00
2023-12-19 14:32:34 +01:00
@last_export = last_export_for(@statut)
2023-12-13 10:26:33 +01:00
if export.available?
2020-01-29 12:16:38 +01:00
respond_to do |format|
format.turbo_stream do
flash.notice = t('instructeurs.procedures.export_available_html', file_format: export.format, file_url: export.file.url)
2020-01-29 12:16:38 +01:00
end
format.html do
redirect_to url_from(export.file.url)
2020-01-29 12:16:38 +01:00
end
end
2019-12-11 17:43:24 +01:00
else
2019-10-16 11:40:19 +02:00
respond_to do |format|
format.turbo_stream do
2020-01-29 12:16:38 +01:00
if !params[:no_progress_notification]
flash.notice = t('instructeurs.procedures.export_pending_html', url: exports_instructeur_procedure_path(procedure))
2020-01-29 12:16:38 +01:00
end
2019-12-11 17:43:24 +01:00
end
format.html do
redirect_to exports_instructeur_procedure_path(procedure), notice: t('instructeurs.procedures.export_pending_html', url: exports_instructeur_procedure_path(procedure))
2019-10-16 11:40:19 +02:00
end
end
end
end
def polling_last_export
2023-12-19 14:32:34 +01:00
@statut = statut
@last_export = last_export_for(@statut)
if @last_export.available?
flash.notice = t('instructeurs.procedures.export_available_html', file_format: @last_export.format, file_url: @last_export.file.url)
else
2023-12-19 14:32:34 +01:00
flash.notice = t('instructeurs.procedures.export_pending_html', url: exports_instructeur_procedure_path(procedure))
end
end
2019-03-13 17:37:07 +01:00
def email_notifications
@procedure = procedure
@assign_to = assign_tos.first
2019-03-13 17:37:07 +01:00
end
def update_email_notifications
assign_tos.each do |assign_to|
assign_to.update!(assign_to_params)
end
2019-03-13 17:37:07 +01:00
flash.notice = 'Vos notifications sont enregistrées.'
redirect_to instructeur_procedure_path(procedure)
2019-03-13 17:37:07 +01:00
end
def stats
@procedure = procedure
@usual_traitement_time = @procedure.stats_usual_traitement_time
@dossiers_funnel = @procedure.stats_dossiers_funnel
@termines_states = @procedure.stats_termines_states
2021-06-02 12:45:47 +02:00
@termines_by_week = @procedure.stats_termines_by_week
@usual_traitement_time_by_month = @procedure.stats_usual_traitement_time_by_month_in_days
end
def exports
@procedure = procedure
@exports = Export.for_groupe_instructeurs(groupe_instructeur_ids).ante_chronological
2024-03-12 16:21:15 +01:00
@export_templates = current_instructeur.export_templates_for(@procedure).includes(:groupe_instructeur)
cookies.encrypted[cookies_export_key] = {
value: DateTime.current,
expires: Export::MAX_DUREE_GENERATION + Export::MAX_DUREE_CONSERVATION_EXPORT,
2024-07-03 11:54:10 +02:00
httponly: true,
secure: Rails.env.production?
}
respond_to do |format|
format.turbo_stream
format.html
end
end
2021-07-19 18:45:49 +02:00
def email_usagers
@procedure = procedure
@bulk_messages = BulkMessage.where(procedure: procedure)
@bulk_message = current_instructeur.bulk_messages.build
@dossiers_without_groupe_count = procedure.dossiers.state_brouillon.for_groupe_instructeur(nil).count
2021-07-19 18:45:49 +02:00
end
def create_multiple_commentaire
@procedure = procedure
errors = []
bulk_message = current_instructeur.bulk_messages.build(bulk_message_params)
dossiers = procedure.dossiers.state_brouillon.for_groupe_instructeur(nil)
dossiers.each do |dossier|
commentaire = CommentaireService.create(current_instructeur, dossier, bulk_message_params.except(:targets))
2021-11-30 18:56:12 +01:00
if commentaire.errors.empty?
2021-07-19 18:45:49 +02:00
commentaire.dossier.update!(last_commentaire_updated_at: Time.zone.now)
else
errors << dossier.id
end
end
valid_dossiers_count = dossiers.count - errors.count
bulk_message.assign_attributes(
dossier_count: valid_dossiers_count,
dossier_state: Dossier.states.fetch(:brouillon),
sent_at: Time.zone.now,
instructeur_id: current_instructeur.id,
procedure_id: @procedure.id
)
bulk_message.save!
2021-07-19 18:45:49 +02:00
if errors.empty?
flash[:notice] = "Tous les messages ont été envoyés avec succès"
else
flash[:alert] = "Envoi terminé. Cependant #{errors.count} messages n'ont pas été envoyés"
end
redirect_to instructeur_procedure_path(@procedure)
end
2022-11-29 09:37:58 +01:00
def administrateurs
@procedure = procedure
@administrateurs = procedure.administrateurs
end
private
def assign_to_params
params.require(:assign_to)
2022-08-30 21:41:04 +02:00
.permit(:instant_expert_avis_email_notifications_enabled, :instant_email_dossier_notifications_enabled, :instant_email_message_notifications_enabled, :daily_email_notifications_enabled, :weekly_email_notifications_enabled)
end
def assign_tos
@assign_tos ||= current_instructeur
.assign_to
.joins(:groupe_instructeur)
.where(groupe_instructeur: { procedure_id: procedure_id })
end
def groupe_instructeur_ids
@groupe_instructeur_ids ||= assign_tos
.map(&:groupe_instructeur_id)
end
2017-09-28 19:07:18 +02:00
def statut
2018-03-06 13:44:29 +01:00
@statut ||= (params[:statut].presence || 'a-suivre')
2017-09-28 19:07:18 +02:00
end
def export_format
2024-03-08 17:14:21 +01:00
@export_format ||= params[:export_format].presence || export_template&.kind
end
def export_template
@export_template ||= ExportTemplate.find(params[:export_template_id]) if params[:export_template_id].present?
end
def export_options
@export_options ||= {
time_span_type: params[:time_span_type],
statut: params[:statut],
2024-03-08 17:14:21 +01:00
export_template:,
procedure_presentation: params[:statut].present? ? procedure_presentation : nil
}.compact
end
2021-04-08 14:35:43 +02:00
def procedure_id
params[:procedure_id]
end
def procedure
2021-04-21 22:02:40 +02:00
Procedure
.with_attached_logo
.find(procedure_id)
.tap { Sentry.set_tags(procedure: _1.id) }
end
def ensure_ownership!
if !current_instructeur.procedures.include?(procedure)
2020-01-29 12:16:38 +01:00
flash[:alert] = "Vous navez pas accès à cette démarche"
redirect_to root_path
end
end
def redirect_to_avis_if_needed
if current_instructeur.procedures.count == 0 && current_instructeur.avis.count > 0
redirect_to instructeur_all_avis_path
end
end
def procedure_presentation
@procedure_presentation ||= begin
procedure_presentation, errors = current_instructeur.procedure_presentation_and_errors_for_procedure_id(procedure_id)
if errors.present?
msg = "Votre affichage a dû être réinitialisé en raison du problème suivant : " + errors.full_messages.join(', ')
if request.get?
flash.now[:alert] = msg
else
flash[:alert] = msg
end
end
procedure_presentation
end
end
2017-09-28 11:04:18 +02:00
def current_filters
@current_filters ||= procedure_presentation.filters.fetch(statut, [])
2017-09-28 11:04:18 +02:00
end
2021-07-19 18:45:49 +02:00
def bulk_message_params
params.require(:bulk_message).permit(:body)
2021-07-19 18:45:49 +02:00
end
def notify_exports?
last_seen_at = begin
DateTime.parse(cookies.encrypted[cookies_export_key])
rescue
nil
end
scope = Export.generated.for_groupe_instructeurs(groupe_instructeur_ids)
scope = scope.where(updated_at: last_seen_at...) if last_seen_at
scope.exists?
end
2023-12-13 10:26:33 +01:00
def last_export_for(statut)
Export.where(user_profile: current_instructeur, statut: statut, updated_at: 1.hour.ago..).last
2023-12-13 10:26:33 +01:00
end
def cookies_export_key
"exports_#{@procedure.id}_seen_at"
end
def sorted_column_params
params.permit(sorted_column: [:order, :id])
end
2024-09-27 15:58:34 +02:00
def filter_params
keys = [:tous_filters, :a_suivre_filters, :suivis_filters, :traites_filters, :expirant_filters, :archives_filters, :supprimes_filters]
h = keys.index_with { [:id, :filter] }
params.permit(h)
end
end
end