256 lines
7.9 KiB
Ruby
256 lines
7.9 KiB
Ruby
class PiecesJustificativesService
|
||
def self.liste_documents(dossiers, for_expert)
|
||
bill_ids = []
|
||
|
||
docs = dossiers.in_batches.flat_map do |batch|
|
||
pjs = pjs_for_champs(batch, for_expert) +
|
||
pjs_for_commentaires(batch) +
|
||
pjs_for_dossier(batch)
|
||
|
||
if !for_expert
|
||
# some bills are shared among operations
|
||
# so first, all the bill_ids are fetched
|
||
operation_logs, some_bill_ids = operation_logs_and_signature_ids(batch)
|
||
|
||
pjs += operation_logs
|
||
bill_ids += some_bill_ids
|
||
end
|
||
|
||
pjs
|
||
end
|
||
|
||
if !for_expert
|
||
# then the bills are retrieved without duplication
|
||
docs += signatures(bill_ids.uniq)
|
||
end
|
||
|
||
docs
|
||
end
|
||
|
||
def self.serialize_types_de_champ_as_type_pj(revision)
|
||
tdcs = revision.types_de_champ_public.filter { |type_champ| type_champ.old_pj.present? }
|
||
tdcs.map.with_index do |type_champ, order_place|
|
||
description = type_champ.description
|
||
if /^(?<original_description>.*?)(?:[\r\n]+)Récupérer le formulaire vierge pour mon dossier : (?<lien_demarche>http.*)$/m =~ description
|
||
description = original_description
|
||
end
|
||
{
|
||
id: type_champ.old_pj[:stable_id],
|
||
libelle: type_champ.libelle,
|
||
description: description,
|
||
order_place: order_place,
|
||
lien_demarche: lien_demarche
|
||
}
|
||
end
|
||
end
|
||
|
||
def self.serialize_champs_as_pjs(dossier)
|
||
dossier.champs_public.filter { |champ| champ.type_de_champ.old_pj }.map do |champ|
|
||
champ.for_api&.map do |content_url|
|
||
{
|
||
created_at: champ.created_at&.in_time_zone('UTC'),
|
||
type_de_piece_justificative_id: champ.type_de_champ.old_pj[:stable_id],
|
||
content_url:,
|
||
user: champ.dossier.user
|
||
}
|
||
end
|
||
end.flatten
|
||
end
|
||
|
||
def self.clone_attachments(original, kopy)
|
||
case original
|
||
when Champs::PieceJustificativeChamp, Champs::TitreIdentiteChamp
|
||
clone_attachment(original.piece_justificative_file, kopy.piece_justificative_file)
|
||
when TypeDeChamp
|
||
clone_attachment(original.piece_justificative_template, kopy.piece_justificative_template)
|
||
when Procedure
|
||
clone_attachment(original.logo, kopy.logo)
|
||
clone_attachment(original.notice, kopy.notice)
|
||
clone_attachment(original.deliberation, kopy.deliberation)
|
||
when AttestationTemplate
|
||
clone_attachment(original.logo, kopy.logo)
|
||
clone_attachment(original.signature, kopy.signature)
|
||
end
|
||
end
|
||
|
||
def self.clone_attachment(original_attachment, copy_attachment)
|
||
if original_attachment.attached?
|
||
copy_attachment.attach(original_attachment.blob)
|
||
end
|
||
end
|
||
|
||
def self.generate_dossier_export(dossiers)
|
||
return [] if dossiers.empty?
|
||
|
||
pdfs = []
|
||
|
||
procedure = dossiers.first.procedure
|
||
dossiers = dossiers.includes(:individual, :traitement, :etablissement, user: :france_connect_information, avis: :expert, commentaires: [:instructeur, :expert])
|
||
dossiers = DossierPreloader.new(dossiers).in_batches
|
||
dossiers.each do |dossier|
|
||
dossier.association(:procedure).target = procedure
|
||
|
||
pdf = ApplicationController
|
||
.render(template: 'dossiers/show', formats: [:pdf],
|
||
assigns: {
|
||
include_infos_administration: true,
|
||
dossier: dossier
|
||
})
|
||
|
||
a = ActiveStorage::FakeAttachment.new(
|
||
file: StringIO.new(pdf),
|
||
filename: "export-#{dossier.id}.pdf",
|
||
name: 'pdf_export_for_instructeur',
|
||
id: dossier.id,
|
||
created_at: dossier.updated_at
|
||
)
|
||
|
||
pdfs << ActiveStorage::DownloadableFile.pj_and_path(dossier.id, a)
|
||
end
|
||
|
||
pdfs
|
||
end
|
||
|
||
private
|
||
|
||
def self.pjs_for_champs(dossiers, for_expert = false)
|
||
champs = Champ
|
||
.joins(:piece_justificative_file_attachments)
|
||
.where(type: "Champs::PieceJustificativeChamp", dossier: dossiers)
|
||
|
||
if for_expert
|
||
champs = champs.where(private: false)
|
||
end
|
||
|
||
champ_id_dossier_id = champs
|
||
.pluck(:id, :dossier_id)
|
||
.to_h
|
||
|
||
ActiveStorage::Attachment
|
||
.includes(:blob)
|
||
.where(record_type: "Champ", record_id: champ_id_dossier_id.keys)
|
||
.filter { |a| safe_attachment(a) }
|
||
.map do |a|
|
||
dossier_id = champ_id_dossier_id[a.record_id]
|
||
ActiveStorage::DownloadableFile.pj_and_path(dossier_id, a)
|
||
end
|
||
end
|
||
|
||
def self.pjs_for_commentaires(dossiers)
|
||
commentaire_id_dossier_id = Commentaire
|
||
.joins(:piece_jointe_attachment)
|
||
.where(dossier: dossiers)
|
||
.pluck(:id, :dossier_id)
|
||
.to_h
|
||
|
||
ActiveStorage::Attachment
|
||
.includes(:blob)
|
||
.where(record_type: "Commentaire", record_id: commentaire_id_dossier_id.keys)
|
||
.filter { |a| safe_attachment(a) }
|
||
.map do |a|
|
||
dossier_id = commentaire_id_dossier_id[a.record_id]
|
||
ActiveStorage::DownloadableFile.pj_and_path(dossier_id, a)
|
||
end
|
||
end
|
||
|
||
def self.pjs_for_dossier(dossiers)
|
||
motivations(dossiers) +
|
||
attestations(dossiers) +
|
||
etablissements(dossiers)
|
||
end
|
||
|
||
def self.etablissements(dossiers)
|
||
etablissement_id_dossier_id = Etablissement
|
||
.where(dossier: dossiers)
|
||
.pluck(:id, :dossier_id)
|
||
.to_h
|
||
|
||
ActiveStorage::Attachment
|
||
.includes(:blob)
|
||
.where(record_type: "Etablissement", record_id: etablissement_id_dossier_id.keys)
|
||
.map do |a|
|
||
dossier_id = etablissement_id_dossier_id[a.record_id]
|
||
ActiveStorage::DownloadableFile.pj_and_path(dossier_id, a)
|
||
end
|
||
end
|
||
|
||
def self.motivations(dossiers)
|
||
ActiveStorage::Attachment
|
||
.includes(:blob)
|
||
.where(record_type: "Dossier", name: "justificatif_motivation", record_id: dossiers)
|
||
.filter { |a| safe_attachment(a) }
|
||
.map do |a|
|
||
dossier_id = a.record_id
|
||
ActiveStorage::DownloadableFile.pj_and_path(dossier_id, a)
|
||
end
|
||
end
|
||
|
||
def self.attestations(dossiers)
|
||
attestation_id_dossier_id = Attestation
|
||
.joins(:pdf_attachment)
|
||
.where(dossier: dossiers)
|
||
.pluck(:id, :dossier_id)
|
||
.to_h
|
||
|
||
ActiveStorage::Attachment
|
||
.includes(:blob)
|
||
.where(record_type: "Attestation", record_id: attestation_id_dossier_id.keys)
|
||
.map do |a|
|
||
dossier_id = attestation_id_dossier_id[a.record_id]
|
||
ActiveStorage::DownloadableFile.pj_and_path(dossier_id, a)
|
||
end
|
||
end
|
||
|
||
def self.operation_logs_and_signature_ids(dossiers)
|
||
dol_id_dossier_id_bill_id = DossierOperationLog
|
||
.where(dossier: dossiers, data: nil)
|
||
.pluck(:bill_signature_id, :id, :dossier_id)
|
||
dol_id_data_bill_id = DossierOperationLog
|
||
.where(dossier: dossiers)
|
||
.with_data
|
||
.pluck(:bill_signature_id, :id, :dossier_id, :data, :digest, :created_at)
|
||
|
||
dol_id_dossier_id = dol_id_dossier_id_bill_id
|
||
.map { |_, dol_id, dossier_id| [dol_id, dossier_id] }
|
||
.to_h
|
||
|
||
bill_ids = (dol_id_dossier_id_bill_id + dol_id_data_bill_id)
|
||
.map(&:first)
|
||
.uniq
|
||
.compact
|
||
|
||
serialized_dols = ActiveStorage::Attachment
|
||
.includes(:blob)
|
||
.where(record_type: "DossierOperationLog", record_id: dol_id_dossier_id.keys)
|
||
.map do |a|
|
||
dossier_id = dol_id_dossier_id[a.record_id]
|
||
ActiveStorage::DownloadableFile.pj_and_path(dossier_id, a)
|
||
end
|
||
serialized_dols += dol_id_data_bill_id.map do |_, id, dossier_id, data, digest, created_at|
|
||
a = ActiveStorage::FakeAttachment.new(
|
||
file: StringIO.new(data.to_json),
|
||
filename: "operation-#{digest}.json",
|
||
name: 'serialized',
|
||
id: id,
|
||
created_at: created_at,
|
||
record_type: 'DossierOperationLog'
|
||
)
|
||
ActiveStorage::DownloadableFile.pj_and_path(dossier_id, a)
|
||
end
|
||
|
||
[serialized_dols, bill_ids]
|
||
end
|
||
|
||
def self.signatures(bill_ids)
|
||
ActiveStorage::Attachment
|
||
.includes(:blob)
|
||
.where(record_type: "BillSignature", record_id: bill_ids)
|
||
.map { |bill| ActiveStorage::DownloadableFile.bill_and_path(bill) }
|
||
end
|
||
|
||
def self.safe_attachment(attachment)
|
||
attachment
|
||
.blob
|
||
.metadata[:virus_scan_result] == ActiveStorage::VirusScanner::SAFE
|
||
end
|
||
end
|