describe Dossier, type: :model do let(:user) { create(:user) } describe 'scopes' do describe '.default_scope' do let!(:dossier) { create(:dossier) } subject { Dossier.all } it { is_expected.to match_array([dossier]) } end describe '.without_followers' do let!(:dossier_with_follower) { create(:dossier, :followed, :with_entreprise, user: user) } let!(:dossier_without_follower) { create(:dossier, :with_entreprise, user: user) } it { expect(Dossier.without_followers.to_a).to eq([dossier_without_follower]) } end describe 'brouillons_recently_updated' do let!(:dossier_en_brouillon) { create(:dossier) } let!(:dossier_en_brouillon_2) { create(:dossier) } it { expect(Dossier.brouillons_recently_updated).to eq([dossier_en_brouillon_2, dossier_en_brouillon]) } end describe 'by_statut' do let(:procedure) { create(:procedure) } let(:dossier_en_construction) { create(:dossier, :en_construction, procedure:) } let(:dossier_en_instruction) { create(:dossier, :en_instruction, procedure:) } let(:dossier_accepte) { create(:dossier, :accepte, procedure:) } let(:dossier_refuse) { create(:dossier, :refuse, procedure:) } let(:dossier_accepte_archive) { create(:dossier, :accepte, :archived, procedure:) } let(:dossier_accepte_deleted) { create(:dossier, :accepte, :hidden_by_administration, procedure:) } let(:dossier_accepte_archive_deleted) { create(:dossier, :accepte, :archived, :hidden_by_administration, procedure:) } let!(:dossiers) { [dossier_en_construction, dossier_en_instruction, dossier_accepte, dossier_refuse] } context 'tous' do it do expect(procedure.dossiers.by_statut('tous')).to match_array(dossiers - [dossier_accepte_archive, dossier_accepte_archive_deleted]) end end context 'a-suivre' do it do expect(procedure.dossiers.by_statut('a-suivre')).to match_array([dossier_en_construction, dossier_en_instruction]) end end context 'supprimes_recemment' do it do expect(procedure.dossiers.by_statut('supprimes_recemment')).to match_array([dossier_accepte_deleted, dossier_accepte_archive_deleted]) end end context 'archives' do it do expect(procedure.dossiers.by_statut('archives')).to match_array([dossier_accepte_archive]) end end end end describe 'validations' do let(:procedure) { create(:procedure, :for_individual) } subject(:dossier) { create(:dossier, procedure: procedure) } it { is_expected.to validate_presence_of(:individual) } it { is_expected.to validate_presence_of(:user) } context 'when dossier has deleted_user_email_never_send' do subject(:dossier) { create(:dossier, procedure: procedure, deleted_user_email_never_send: "seb@totoro.org") } it { is_expected.not_to validate_presence_of(:user) } end context 'when dossier is prefilled' do subject(:dossier) { create(:dossier, procedure: procedure, prefilled: true) } it { is_expected.not_to validate_presence_of(:user) } end end describe 'brouillon_close_to_expiration' do let(:procedure) { create(:procedure, :published, duree_conservation_dossiers_dans_ds: 6) } let!(:young_dossier) { create(:dossier, :en_construction, procedure: procedure) } let!(:expiring_dossier) { create(:dossier, created_at: 175.days.ago, procedure: procedure) } let!(:expiring_dossier_with_notification) { create(:dossier, created_at: 175.days.ago, brouillon_close_to_expiration_notice_sent_at: Time.zone.now, procedure: procedure) } let!(:just_expired_dossier) { create(:dossier, created_at: (6.months + 1.hour + 10.seconds).ago, procedure: procedure) } let!(:long_expired_dossier) { create(:dossier, created_at: 1.year.ago, procedure: procedure) } subject { Dossier.brouillon_close_to_expiration } it do is_expected.not_to include(young_dossier) is_expected.to include(expiring_dossier) is_expected.to include(just_expired_dossier) is_expected.to include(long_expired_dossier) end it do expect(expiring_dossier.close_to_expiration?).to be_truthy expect(expiring_dossier_with_notification.close_to_expiration?).to be_truthy end context 'does not include an expiring dossier that has been postponed' do before do expiring_dossier.extend_conservation(1.month) expiring_dossier_with_notification.extend_conservation(1.month) expiring_dossier.reload expiring_dossier_with_notification.reload end it { is_expected.not_to include(expiring_dossier) } it do expect(expiring_dossier.close_to_expiration?).to be_falsey expect(expiring_dossier_with_notification.close_to_expiration?).to be_falsey expect(expiring_dossier.expiration_date).to eq(expiring_dossier.expiration_date_with_extension) expect(expiring_dossier_with_notification.expiration_date).to eq(expiring_dossier_with_notification.expiration_date_with_extension) end end context 'when .close_to_expiration' do subject { Dossier.close_to_expiration } it do is_expected.not_to include(young_dossier) is_expected.to include(expiring_dossier) is_expected.to include(just_expired_dossier) is_expected.to include(long_expired_dossier) end end end describe 'en_construction_close_to_expiration' do let(:procedure) { create(:procedure, :published, duree_conservation_dossiers_dans_ds: 6) } let!(:young_dossier) { create(:dossier, procedure: procedure) } let!(:expiring_dossier) { create(:dossier, :en_construction, en_construction_at: 175.days.ago, procedure: procedure) } let!(:expiring_dossier_with_notification) { create(:dossier, :en_construction, en_construction_at: 175.days.ago, en_construction_close_to_expiration_notice_sent_at: Time.zone.now, procedure: procedure) } let!(:just_expired_dossier) { create(:dossier, :en_construction, en_construction_at: (6.months + 1.hour + 10.seconds).ago, procedure: procedure) } let!(:long_expired_dossier) { create(:dossier, :en_construction, en_construction_at: 1.year.ago, procedure: procedure) } subject { Dossier.en_construction_close_to_expiration } it do is_expected.not_to include(young_dossier) is_expected.to include(expiring_dossier) is_expected.to include(just_expired_dossier) is_expected.to include(long_expired_dossier) end it do expect(expiring_dossier.close_to_expiration?).to be_truthy expect(expiring_dossier_with_notification.close_to_expiration?).to be_truthy end context 'does not include an expiring dossier that has been postponed' do before do expiring_dossier.extend_conservation(1.month) expiring_dossier_with_notification.extend_conservation(1.month) expiring_dossier.reload expiring_dossier_with_notification.reload end it { is_expected.not_to include(expiring_dossier) } it do expect(expiring_dossier.close_to_expiration?).to be_falsey expect(expiring_dossier_with_notification.close_to_expiration?).to be_falsey expect(expiring_dossier.expiration_date).to eq(expiring_dossier.expiration_date_with_extension) expect(expiring_dossier_with_notification.expiration_date).to eq(expiring_dossier_with_notification.expiration_date_with_extension) end end context 'when .close_to_expiration' do subject { Dossier.close_to_expiration } it do is_expected.not_to include(young_dossier) is_expected.to include(expiring_dossier) is_expected.to include(just_expired_dossier) is_expected.to include(long_expired_dossier) end end context 'when .termine_or_en_construction_close_to_expiration' do subject { Dossier.termine_or_en_construction_close_to_expiration } it do is_expected.not_to include(young_dossier) is_expected.to include(expiring_dossier) is_expected.to include(just_expired_dossier) is_expected.to include(long_expired_dossier) end end end describe 'termine_close_to_expiration' do let(:procedure) { create(:procedure, :published, duree_conservation_dossiers_dans_ds: 6, procedure_expires_when_termine_enabled: true) } let!(:young_dossier) { create(:dossier, state: :accepte, procedure: procedure, processed_at: 2.days.ago) } let!(:expiring_dossier) { create(:dossier, state: :accepte, procedure: procedure, processed_at: 175.days.ago) } let!(:expiring_dossier_with_notification) { create(:dossier, state: :accepte, procedure: procedure, processed_at: 175.days.ago, termine_close_to_expiration_notice_sent_at: Time.zone.now) } let!(:just_expired_dossier) { create(:dossier, state: :accepte, procedure: procedure, processed_at: (6.months + 1.hour + 10.seconds).ago) } let!(:long_expired_dossier) { create(:dossier, state: :accepte, procedure: procedure, processed_at: 1.year.ago) } subject { Dossier.termine_close_to_expiration } it do is_expected.not_to include(young_dossier) is_expected.to include(expiring_dossier) is_expected.to include(just_expired_dossier) is_expected.to include(long_expired_dossier) end it do expect(expiring_dossier.close_to_expiration?).to be_truthy expect(expiring_dossier_with_notification.close_to_expiration?).to be_truthy end context 'does not include an expiring dossier that has been postponed' do before do expiring_dossier.extend_conservation(1.month) expiring_dossier_with_notification.extend_conservation(1.month) expiring_dossier.reload expiring_dossier_with_notification.reload end it { is_expected.not_to include(expiring_dossier) } it do expect(expiring_dossier.close_to_expiration?).to be_falsey expect(expiring_dossier_with_notification.close_to_expiration?).to be_falsey expect(expiring_dossier.expiration_date).to eq(expiring_dossier.expiration_date_with_extension) expect(expiring_dossier_with_notification.expiration_date).to eq(expiring_dossier_with_notification.expiration_date_with_extension) end end context 'when .close_to_expiration' do subject { Dossier.close_to_expiration } it do is_expected.not_to include(young_dossier) is_expected.to include(expiring_dossier) is_expected.to include(just_expired_dossier) is_expected.to include(long_expired_dossier) end end context 'when .close_to_expiration' do subject { Dossier.termine_or_en_construction_close_to_expiration } it do is_expected.not_to include(young_dossier) is_expected.to include(expiring_dossier) is_expected.to include(just_expired_dossier) is_expected.to include(long_expired_dossier) end end end describe 'with_notifications' do let(:dossier) { create(:dossier) } let(:instructeur) { create(:instructeur) } before do create(:follow, dossier: dossier, instructeur: instructeur, messagerie_seen_at: 2.hours.ago) end subject { instructeur.followed_dossiers.with_notifications } context('without changes') do it { is_expected.to eq [] } end context('with changes') do context 'when there is a new commentaire' do before { dossier.update!(last_commentaire_updated_at: Time.zone.now) } it { is_expected.to match([dossier]) } end context 'when there is a new avis' do before { dossier.update!(last_avis_updated_at: Time.zone.now) } it { is_expected.to match([dossier]) } end context 'when a public champ is updated' do before { dossier.update!(last_champ_updated_at: Time.zone.now) } it { is_expected.to match([dossier]) } end context 'when a private champ is updated' do before { dossier.update!(last_champ_private_updated_at: Time.zone.now) } it { is_expected.to match([dossier]) } end end end describe 'methods' do let(:dossier) { create(:dossier, :with_entreprise, user: user) } let(:etablissement) { dossier.etablissement } subject { dossier } describe '#create' do let(:procedure) { create(:procedure, :with_type_de_champ, :with_type_de_champ_private) } let(:dossier) { create(:dossier, procedure: procedure, user: user) } it 'builds public and private champs' do expect(dossier.champs_public.count).to eq(1) expect(dossier.champs_private.count).to eq(1) end end describe '#build_default_individual' do let(:dossier) { build(:dossier, procedure: procedure, user: user) } subject do dossier.individual = nil dossier.build_default_individual end context 'when the dossier belongs to a procedure for individuals' do let(:procedure) { create(:procedure, for_individual: true) } it 'creates a default individual' do subject expect(dossier.individual).to be_present expect(dossier.individual.nom).to be_nil expect(dossier.individual.prenom).to be_nil expect(dossier.individual.gender).to be_nil end context 'and the user signs-in using France Connect' do let(:user) { create(:user, france_connect_informations: [build(:france_connect_information)]) } it 'fills the individual with the informations from France Connect' do subject expect(dossier.individual.nom).to eq('DUBOIS') expect(dossier.individual.prenom).to eq('Angela Claire Louise') expect(dossier.individual.gender).to eq(Individual::GENDER_FEMALE) end end context 'and the user signs-in using France Connect many times' do let(:user) { create(:user, france_connect_informations: [build(:france_connect_information), build(:france_connect_information)]) } it 'fills the individual with the informations from France Connect' do subject expect(dossier.individual.nom).to eq(nil) expect(dossier.individual.prenom).to eq(nil) expect(dossier.individual.gender).to eq(nil) end end end context 'when the dossier belongs to a procedure for moral personas' do let(:procedure) { create(:procedure, for_individual: false) } it 'doesn’t create a individual' do subject expect(dossier.individual).to be_nil end end end end context 'when dossier is followed' do let(:procedure) { create(:procedure, :with_type_de_champ, :with_type_de_champ_private) } let(:instructeur) { create(:instructeur) } let(:date1) { 1.day.ago } let(:date2) { 1.hour.ago } let(:date3) { 1.minute.ago } let(:dossier) do d = create(:dossier, :with_entreprise, user: user, procedure: procedure) Timecop.freeze(date1) d.passer_en_construction! Timecop.freeze(date2) d.passer_en_instruction!(instructeur: instructeur) Timecop.freeze(date3) d.accepter!(instructeur: instructeur, motivation: "Motivation") Timecop.return d end describe "followers_instructeurs" do let(:non_following_instructeur) { create(:instructeur) } subject { dossier.followers_instructeurs } it { expect(subject).to eq [instructeur] } it { expect(subject).not_to include(non_following_instructeur) } end end describe "#text_summary" do let(:service) { create(:service, nom: 'nom du service') } let(:procedure) { create(:procedure, libelle: "Démarche", organisation: "Organisme", service: service) } context 'when the dossier has been submitted' do let(:dossier) { create :dossier, procedure: procedure, state: Dossier.states.fetch(:en_construction), depose_at: "31/12/2010".to_date } subject { dossier.text_summary } it { is_expected.to eq("Dossier déposé le 31/12/2010 sur la démarche Démarche gérée par l'organisme nom du service") } end context 'when the dossier has not been submitted' do let(:dossier) { create :dossier, procedure: procedure, state: Dossier.states.fetch(:brouillon) } subject { dossier.text_summary } it { is_expected.to eq("Dossier en brouillon répondant à la démarche Démarche gérée par l'organisme nom du service") } end end describe '#avis_for' do let!(:instructeur) { create(:instructeur) } let!(:procedure) { create(:procedure, :published, instructeurs: [instructeur]) } let!(:dossier) { create(:dossier, procedure: procedure, state: Dossier.states.fetch(:en_construction)) } let!(:experts_procedure) { create(:experts_procedure, expert: expert_1, procedure: procedure) } let!(:experts_procedure_2) { create(:experts_procedure, expert: expert_2, procedure: procedure) } let!(:expert_1) { create(:expert) } let!(:expert_2) { create(:expert) } context 'when there is a public advice asked from the dossiers instructeur' do let!(:avis) { create(:avis, dossier: dossier, claimant: instructeur, experts_procedure: experts_procedure, confidentiel: false) } it { expect(dossier.avis_for_expert(expert_1)).to match([avis]) } it { expect(dossier.avis_for_expert(expert_2)).to match([avis]) } end context 'when there is a private advice asked from the dossiers instructeur' do let!(:avis) { create(:avis, dossier: dossier, claimant: instructeur, experts_procedure: experts_procedure, confidentiel: true) } it { expect(dossier.avis_for_expert(expert_1)).to match([avis]) } it { expect(dossier.avis_for_expert(expert_2)).not_to match([avis]) } end context 'when there is a public advice asked from one instructeur to an expert' do let!(:avis_1) { create(:avis, dossier: dossier, claimant: instructeur, experts_procedure: experts_procedure, confidentiel: false) } let!(:avis_2) { create(:avis, dossier: dossier, claimant: instructeur, experts_procedure: experts_procedure_2, confidentiel: false) } it { expect(dossier.avis_for_expert(expert_1)).to match([avis_1, avis_2]) } it { expect(dossier.avis_for_expert(expert_2)).to match([avis_1, avis_2]) } end context 'when there is a private advice asked from one instructeur to an expert' do let!(:avis_1) { create(:avis, dossier: dossier, claimant: instructeur, experts_procedure: experts_procedure, confidentiel: true) } let!(:avis_2) { create(:avis, dossier: dossier, claimant: instructeur, experts_procedure: experts_procedure_2, confidentiel: true) } it { expect(dossier.avis_for_expert(expert_1)).to match([avis_1]) } it { expect(dossier.avis_for_expert(expert_2)).to match([avis_2]) } end context 'when they are a lot of advice' do let!(:avis_1) { create(:avis, dossier: dossier, claimant: expert_1, experts_procedure: experts_procedure_2, confidentiel: false, created_at: Time.zone.parse('10/01/2010')) } let!(:avis_2) { create(:avis, dossier: dossier, claimant: expert_1, experts_procedure: experts_procedure_2, confidentiel: false, created_at: Time.zone.parse('9/01/2010')) } let!(:avis_3) { create(:avis, dossier: dossier, claimant: expert_1, experts_procedure: experts_procedure_2, confidentiel: false, created_at: Time.zone.parse('11/01/2010')) } it { expect(dossier.avis_for_expert(expert_1)).to match([avis_2, avis_1, avis_3]) } end context 'when they are a advice published on another dossier' do let!(:avis) { create(:avis, dossier: create(:dossier, procedure: procedure), claimant: instructeur, experts_procedure: experts_procedure, confidentiel: false, created_at: Time.zone.parse('9/01/2010')) } it { expect(dossier.avis_for_expert(expert_1)).to match([]) } end end describe '#update_state_dates' do let(:dossier) { create(:dossier, :brouillon, :with_individual) } let(:beginning_of_day) { Time.zone.now.beginning_of_day } let(:instructeur) { create(:instructeur) } before { Timecop.freeze(beginning_of_day) } after { Timecop.return } context 'when dossier is en_construction' do context 'when the procedure.routing_enabled? is false' do subject do dossier.passer_en_construction! dossier.reload end it do subject expect(dossier.state).to eq(Dossier.states.fetch(:en_construction)) expect(dossier.en_construction_at).to eq(beginning_of_day) expect(dossier.depose_at).to eq(beginning_of_day) expect(dossier.traitement.state).to eq(Dossier.states.fetch(:en_construction)) expect(dossier.traitement.processed_at).to eq(beginning_of_day) end it 'should keep first en_construction_at date' do subject Timecop.return dossier.passer_en_instruction!(instructeur: instructeur) dossier.repasser_en_construction!(instructeur: instructeur) expect(dossier.traitements.size).to eq(3) expect(dossier.traitements.first.processed_at).to eq(beginning_of_day) expect(dossier.traitement.processed_at.round).to eq(dossier.en_construction_at.round) expect(dossier.depose_at).to eq(beginning_of_day) expect(dossier.en_construction_at).to be > beginning_of_day end context 'when dossier have piece_justificative or titre_identite' do include Logic let(:procedure) { create(:procedure, types_de_champ_public:) } let(:dossier) { create(:dossier, :brouillon, :with_populated_champs, procedure:) } before { expect(champ).to receive(:visible?).and_return(visible) } context 'when piece_justificative' do let(:types_de_champ_public) { [{ type: :piece_justificative }] } let(:champ) { dossier.champs_for_revision(scope: :public).find(&:piece_justificative?) } context 'when not visible' do let(:visible) { false } it { expect { subject }.to change { champ.reload.piece_justificative_file.attached? } } end context 'when visible' do let(:visible) { true } it { expect { subject }.not_to change { champ.reload.piece_justificative_file.attached? } } end end context 'when titre identite' do let(:types_de_champ_public) { [{ type: :titre_identite }] } let(:champ) { dossier.champs_for_revision(scope: :public).find(&:piece_justificative?) } context 'when not visible' do let(:visible) { false } it { expect { subject }.to change { champ.reload.piece_justificative_file.attached? } } end context 'when visible' do let(:visible) { true } it { expect { subject }.not_to change { champ.reload.piece_justificative_file.attached? } } end end end end context 'when the procedure.routing_enabled? is true' do include Logic let(:gi_libelle) { 'Paris' } let!(:procedure) do create(:procedure, types_de_champ_public: [ { type: :drop_down_list, libelle: 'Votre ville', options: [gi_libelle, 'Lyon', 'Marseille'] }, { type: :text, libelle: 'Un champ texte' } ]) end let!(:drop_down_tdc) { procedure.draft_revision.types_de_champ.first } let(:dossier) { create(:dossier, :brouillon, user:, procedure:, groupe_instructeur: nil) } let(:gi) do create(:groupe_instructeur, routing_rule: ds_eq(champ_value(drop_down_tdc.stable_id), constant(gi_libelle))) end before do procedure.groupe_instructeurs = [gi] procedure.defaut_groupe_instructeur = gi procedure.save! procedure.toggle_routing dossier.champs.first.value = gi_libelle dossier.save! dossier.passer_en_construction! dossier.reload end it 'RoutingEngine.compute' do expect(dossier.groupe_instructeur).not_to be_nil end end end context 'when dossier is en_instruction' do let(:dossier) { create(:dossier, :en_construction, :with_individual) } let(:instructeur) { create(:instructeur) } before do dossier.passer_en_instruction!(instructeur: instructeur) dossier.reload end it { expect(dossier.state).to eq(Dossier.states.fetch(:en_instruction)) } it { expect(dossier.en_instruction_at).to eq(beginning_of_day) } it { expect(dossier.traitement.state).to eq(Dossier.states.fetch(:en_instruction)) } it { expect(dossier.traitement.processed_at).to eq(beginning_of_day) } it 'should keep first en_instruction_at date if dossier is set to en_construction again' do Timecop.return dossier.repasser_en_construction!(instructeur: instructeur) dossier.passer_en_instruction!(instructeur: instructeur) expect(dossier.traitements.size).to eq(4) expect(dossier.traitements.en_construction.first.processed_at).to eq(dossier.depose_at) expect(dossier.traitements.en_instruction.first.processed_at).to eq(beginning_of_day) expect(dossier.traitement.processed_at.round).to eq(dossier.en_instruction_at.round) expect(dossier.en_instruction_at).to be > beginning_of_day end end context 'when dossier is accepte' do let(:dossier) { create(:dossier, :en_instruction, :with_individual) } before do dossier.accepter!(instructeur: instructeur) dossier.reload end it { expect(dossier.state).to eq(Dossier.states.fetch(:accepte)) } it { expect(dossier.processed_at).to eq(beginning_of_day) } it { expect(dossier.traitement.state).to eq(Dossier.states.fetch(:accepte)) } it { expect(dossier.traitement.processed_at).to eq(beginning_of_day) } end context 'when dossier is refuse' do let(:dossier) { create(:dossier, :en_instruction, :with_individual) } before do dossier.refuser!(instructeur: instructeur) dossier.reload end it { expect(dossier.state).to eq(Dossier.states.fetch(:refuse)) } it { expect(dossier.processed_at).to eq(beginning_of_day) } it { expect(dossier.traitement.state).to eq(Dossier.states.fetch(:refuse)) } it { expect(dossier.traitement.processed_at).to eq(beginning_of_day) } end context 'when dossier is sans_suite' do let(:dossier) { create(:dossier, :en_instruction, :with_individual) } before do dossier.classer_sans_suite!(instructeur: instructeur) dossier.reload end it { expect(dossier.state).to eq(Dossier.states.fetch(:sans_suite)) } it { expect(dossier.processed_at).to eq(beginning_of_day) } it { expect(dossier.traitement.state).to eq(Dossier.states.fetch(:sans_suite)) } it { expect(dossier.traitement.processed_at).to eq(beginning_of_day) } end end describe '.ordered_for_export' do let(:procedure) { create(:procedure) } let!(:dossier2) { create(:dossier, :with_entreprise, procedure: procedure, state: Dossier.states.fetch(:en_construction), depose_at: Time.zone.parse('03/01/2010')) } let!(:dossier3) { create(:dossier, :with_entreprise, procedure: procedure, state: Dossier.states.fetch(:en_instruction), depose_at: Time.zone.parse('01/01/2010')) } let!(:dossier4) { create(:dossier, :with_entreprise, procedure: procedure, state: Dossier.states.fetch(:en_instruction), archived: true, depose_at: Time.zone.parse('02/01/2010')) } subject { procedure.dossiers.ordered_for_export } it { is_expected.to match([dossier3, dossier4, dossier2]) } end describe "#assign_to_groupe_instructeur" do let(:procedure) { create(:procedure) } let(:new_groupe_instructeur_new_procedure) { create(:groupe_instructeur) } let(:new_groupe_instructeur) { create(:groupe_instructeur, procedure: procedure) } let(:dossier) { create(:dossier, :en_construction, procedure: procedure) } it "can change groupe instructeur" do dossier.assign_to_groupe_instructeur(new_groupe_instructeur_new_procedure, DossierAssignment.modes.fetch(:auto)) expect(dossier.groupe_instructeur).not_to eq(new_groupe_instructeur_new_procedure) end it "can not change groupe instructeur if new groupe is from another procedure" do dossier.assign_to_groupe_instructeur(new_groupe_instructeur, DossierAssignment.modes.fetch(:auto)) expect(dossier.groupe_instructeur).to eq(new_groupe_instructeur) end end describe "#unfollow_stale_instructeurs" do let(:procedure) { create(:procedure, :published, :for_individual) } let(:instructeur) { create(:instructeur) } let(:new_groupe_instructeur) { create(:groupe_instructeur, procedure: procedure) } let(:instructeur2) { create(:instructeur, groupe_instructeurs: [procedure.defaut_groupe_instructeur, new_groupe_instructeur]) } let(:dossier) { create(:dossier, :en_construction, :with_individual, procedure: procedure) } let(:last_operation) { DossierOperationLog.last } before do allow(DossierMailer).to receive(:notify_groupe_instructeur_changed).and_return(double(deliver_later: nil)) end it "unfollows stale instructeurs when groupe instructeur change" do instructeur.follow(dossier) instructeur2.follow(dossier) dossier.reload.assign_to_groupe_instructeur(new_groupe_instructeur, DossierAssignment.modes.fetch(:auto), procedure.administrateurs.first) expect(dossier.reload.followers_instructeurs).not_to include(instructeur) expect(dossier.reload.followers_instructeurs).to include(instructeur2) expect(DossierMailer).to have_received(:notify_groupe_instructeur_changed).with(instructeur, dossier) expect(DossierMailer).not_to have_received(:notify_groupe_instructeur_changed).with(instructeur2, dossier) expect(last_operation.operation).to eq("changer_groupe_instructeur") expect(last_operation.dossier).to eq(dossier) expect(last_operation.automatic_operation?).to be_falsey end end describe "#send_dossier_received" do let(:procedure) { create(:procedure) } let(:dossier) { create(:dossier, procedure: procedure, state: Dossier.states.fetch(:en_construction)) } let(:instructeur) { create(:instructeur) } before do allow(NotificationMailer).to receive(:send_en_instruction_notification).and_return(double(deliver_later: nil)) end it "sends an email when the dossier becomes en_instruction" do dossier.passer_en_instruction!(instructeur: instructeur) expect(NotificationMailer).to have_received(:send_en_instruction_notification).with(dossier) end it "does not an email when the dossier becomes accepte" do dossier.accepte! expect(NotificationMailer).to_not have_received(:send_en_instruction_notification) end end describe "#unspecified_attestation_champs" do let(:procedure) { create(:procedure, attestation_template: attestation_template, types_de_champ_public: types_de_champ, types_de_champ_private: types_de_champ_private) } let(:dossier) { create(:dossier, :en_instruction, procedure: procedure) } let(:types_de_champ) { [tdc_1, tdc_2, tdc_3, tdc_4] } let(:types_de_champ_private) { [tdc_5, tdc_6, tdc_7, tdc_8] } let(:tdc_1) { { libelle: "specified champ-in-title" } } let(:tdc_2) { { libelle: "unspecified champ-in-title" } } let(:tdc_3) { { libelle: "specified champ-in-body" } } let(:tdc_4) { { libelle: "unspecified champ-in-body" } } let(:tdc_5) { { libelle: "specified annotation privée-in-title" } } let(:tdc_6) { { libelle: "unspecified annotation privée-in-title" } } let(:tdc_7) { { libelle: "specified annotation privée-in-body" } } let(:tdc_8) { { libelle: "unspecified annotation privée-in-body" } } before do (dossier.champs_public + dossier.champs_private) .filter { |c| c.libelle.match?(/^specified/) } .each { |c| c.update_attribute(:value, "specified") } end subject { dossier.unspecified_attestation_champs.map(&:libelle) } context "without attestation template" do let(:attestation_template) { nil } it { is_expected.to eq([]) } end context "with attestation template v1" do # Test all combinations: # - with tag specified and unspecified # - with tag in body and tag in title # - with tag correponding to a champ and an annotation privée # - with a dash in the champ libelle / tag let(:title) { "voici --specified champ-in-title-- un --unspecified champ-in-title-- beau --specified annotation privée-in-title-- titre --unspecified annotation privée-in-title-- non --numéro du dossier--" } let(:body) { "voici --specified champ-in-body-- un --unspecified champ-in-body-- beau --specified annotation privée-in-body-- body --unspecified annotation privée-in-body-- non ?" } let(:attestation_template) { build(:attestation_template, title: title, body: body, activated: activated) } context "which is disabled" do let(:activated) { false } it { is_expected.to eq([]) } end context "which is enabled" do let(:activated) { true } it do is_expected.to eq([ "unspecified champ-in-title", "unspecified annotation privée-in-title", "unspecified champ-in-body", "unspecified annotation privée-in-body" ]) end end end context "with attestation template v2" do # Test all combinations: # - with tag specified and unspecified # - with tag correponding to a champ and an annotation privée let(:body) { [ { "type" => "mention", "attrs" => { "id" => "tdc#{procedure.types_de_champ_for_tags.find {  _1.libelle == "unspecified champ-in-body" }.stable_id}", "label" => "unspecified champ-in-body" } } ] } let(:attestation_template) { build(:attestation_template, :v2) } before do tdc_content = (types_de_champ + types_de_champ_private).filter_map do |tdc_config| next if tdc_config[:libelle].include?("in-title") { "type" => "mention", "attrs" => { "id" => "tdc#{procedure.types_de_champ_for_tags.find { _1.libelle == tdc_config[:libelle] }.stable_id}", "label" => tdc_config[:libelle] } } end json_body = attestation_template.json_body["content"] attestation_template.json_body["content"][-1]["content"].concat(tdc_content) attestation_template.save! end it do is_expected.to eq([ "unspecified champ-in-body", "unspecified annotation privée-in-body" ]) end end end describe '#build_attestation' do let(:attestation_template) { nil } let(:procedure) { create(:procedure, attestation_template: attestation_template) } before :each do dossier.attestation = dossier.build_attestation dossier.reload end context 'when the dossier is in en_instruction state ' do let!(:dossier) { create(:dossier, :en_instruction, procedure: procedure) } context 'when the procedure has no attestation' do it { expect(dossier.attestation).to be_nil } end context 'when the procedure has an unactivated attestation' do let(:attestation_template) { build(:attestation_template, activated: false) } it { expect(dossier.attestation).to be_nil } end context 'when the procedure attached has an activated attestation' do let(:attestation_template) { build(:attestation_template, activated: true) } it { expect(dossier.attestation).not_to be_nil } end end end describe 'updated_at' do let!(:dossier) { create(:dossier) } let(:modif_date) { Time.zone.parse('01/01/2100') } before { Timecop.freeze(modif_date) } after { Timecop.return } subject do dossier.reload dossier.updated_at end it { is_expected.not_to eq(modif_date) } context 'when a champ is modified' do before { dossier.champs_public.first.update_attribute('value', 'yop') } it { is_expected.to eq(modif_date) } end context 'when a commentaire is modified' do before { dossier.commentaires << create(:commentaire) } it { is_expected.to eq(modif_date) } end context 'when an avis is modified' do before { dossier.avis << create(:avis) } it { is_expected.to eq(modif_date) } end end describe '#owner_name' do let(:procedure) { create(:procedure) } subject { dossier.owner_name } context 'when there is no entreprise or individual' do let(:dossier) { create(:dossier, individual: nil, procedure: procedure) } it { is_expected.to be_nil } end context 'when there is entreprise' do let(:dossier) { create(:dossier, :with_entreprise, procedure: procedure) } it { is_expected.to eq(dossier.etablissement.entreprise_raison_sociale) } end context 'when there is an individual' do let(:procedure) { create(:procedure, :for_individual) } let(:dossier) { create(:dossier, :with_individual, procedure: procedure) } it { is_expected.to eq("#{dossier.individual.nom} #{dossier.individual.prenom}") } end end describe "#hide_and_keep_track!" do let(:dossier) { create(:dossier, :en_construction) } let(:user) { dossier.user } let(:last_operation) { dossier.dossier_operation_logs.last } let(:reason) { :user_request } before do allow(DossierMailer).to receive(:notify_deletion_to_administration).and_return(double(deliver_later: nil)) end subject! { dossier.hide_and_keep_track!(user, reason) } context 'brouillon' do let(:dossier) { create(:dossier) } it 'hide the dossier' do expect(dossier.reload.hidden_by_user_at).to be_present end it 'does not records operation in the log' do expect(dossier.reload.dossier_operation_logs.last).to eq(nil) end end context 'en_construction' do it 'hide the dossier but does not discard' do expect(dossier.hidden_at).to be_nil expect(dossier.hidden_by_user_at).to be_present end it 'records the operation in the log' do expect(last_operation.operation).to eq("supprimer") expect(last_operation.automatic_operation?).to be_falsey end context 'where instructeurs are following the dossier' do let(:dossier) { create(:dossier, :en_construction, :followed) } let!(:non_following_instructeur) do non_following_instructeur = create(:instructeur) non_following_instructeur.groupe_instructeurs << dossier.procedure.defaut_groupe_instructeur non_following_instructeur end end context 'when dossier is brouillon' do let(:dossier) { create(:dossier) } it 'do not notifies the procedure administrateur' do expect(DossierMailer).not_to have_received(:notify_deletion_to_administration) end end context 'with reason: user_removed' do let(:reason) { :user_removed } it 'hide the dossier' do expect(dossier.hidden_by_user_at).to be_present end it 'write the good reason to hidden_by_reason' do expect(dossier.hidden_by_reason).to eq("user_removed") end end end context 'termine' do let(:dossier) { create(:dossier, state: "accepte", hidden_by_administration_at: 1.hour.ago) } before { subject } it 'affect the right deletion reason to the dossier' do expect(dossier.hidden_by_reason).to eq("user_request") end end end describe 'webhook' do let(:dossier) { create(:dossier) } let(:instructeur) { create(:instructeur) } it 'should not call webhook' do expect { dossier.accepte! }.to_not have_enqueued_job(WebHookJob) end it 'should not call webhook with empty value' do dossier.procedure.update_column(:web_hook_url, '') expect { dossier.accepte! }.to_not have_enqueued_job(WebHookJob) end it 'should not call webhook with blank value' do dossier.procedure.update_column(:web_hook_url, ' ') expect { dossier.accepte! }.to_not have_enqueued_job(WebHookJob) end it 'should call webhook' do dossier.procedure.update_column(:web_hook_url, '/webhook.json') expect { dossier.update_column(:conservation_extension, 'P1W') }.to_not have_enqueued_job(WebHookJob) expect { dossier.passer_en_construction! }.to have_enqueued_job(WebHookJob).with(dossier.procedure.id, dossier.id, 'en_construction', anything) expect { dossier.update_column(:conservation_extension, 'P2W') }.to_not have_enqueued_job(WebHookJob) expect { dossier.passer_en_instruction!(instructeur: instructeur) }.to have_enqueued_job(WebHookJob).with(dossier.procedure.id, dossier.id, 'en_instruction', anything) end end describe "#can_transition_to_en_construction?" do let(:procedure) { create(:procedure, :published) } let(:dossier) { create(:dossier, state: state, procedure: procedure) } subject { dossier.can_transition_to_en_construction? } context "dossier state is brouillon" do let(:state) { Dossier.states.fetch(:brouillon) } it { is_expected.to be true } context "procedure is closed" do before { procedure.close! } it { is_expected.to be false } end end context "dossier state is en_construction" do let(:state) { Dossier.states.fetch(:en_construction) } it { is_expected.to be false } end context "dossier state is en_instruction" do let(:state) { Dossier.states.fetch(:en_instruction) } it { is_expected.to be false } end context "dossier state is en_instruction" do let(:state) { Dossier.states.fetch(:accepte) } it { is_expected.to be false } end context "dossier state is en_instruction" do let(:state) { Dossier.states.fetch(:refuse) } it { is_expected.to be false } end context "dossier state is en_instruction" do let(:state) { Dossier.states.fetch(:sans_suite) } it { is_expected.to be false } end end describe "#messagerie_available?" do let(:procedure) { create(:procedure) } let(:dossier) { create(:dossier, procedure: procedure) } subject { dossier.messagerie_available? } context "dossier is brouillon" do before { dossier.state = Dossier.states.fetch(:brouillon) } it { is_expected.to be false } end context "dossier is submitted" do before { dossier.state = Dossier.states.fetch(:en_instruction) } it { is_expected.to be true } end context "dossier is archived" do before { dossier.archived = true } it { is_expected.to be false } end end describe '#accepter!' do let(:dossier) { create(:dossier, :en_instruction, :with_individual) } let(:last_operation) { dossier.dossier_operation_logs.last } let(:operation_serialized) { last_operation.data } let!(:instructeur) { create(:instructeur) } let!(:now) { Time.zone.parse('01/01/2100') } let(:attestation) { Attestation.new } before do allow(NotificationMailer).to receive(:send_accepte_notification).and_return(double(deliver_later: true)) allow(dossier).to receive(:build_attestation).and_return(attestation) travel_to now dossier.accepter!(instructeur: instructeur, motivation: 'motivation') dossier.reload end it "update attributes" do expect(dossier.traitements.last.motivation).to eq('motivation') expect(dossier.motivation).to eq('motivation') expect(dossier.traitements.last.instructeur_email).to eq(instructeur.email) expect(dossier.en_instruction_at).to eq(dossier.en_instruction_at) expect(dossier.traitements.last.processed_at).to eq(now) expect(dossier.processed_at).to eq(now) expect(dossier.state).to eq('accepte') expect(last_operation.operation).to eq('accepter') expect(last_operation.automatic_operation?).to be_falsey expect(operation_serialized['operation']).to eq('accepter') expect(operation_serialized['dossier_id']).to eq(dossier.id) expect(operation_serialized['executed_at']).to eq(last_operation.executed_at.iso8601) expect(NotificationMailer).to have_received(:send_accepte_notification).with(dossier) expect(dossier.attestation).to eq(attestation) expect(dossier.commentaires.count).to eq(1) end end describe '#accepter_automatiquement!' do let(:last_operation) { dossier.dossier_operation_logs.last } let!(:now) { Time.zone.parse('01/01/2100') } let(:attestation) { Attestation.new } before do allow(NotificationMailer).to receive(:send_accepte_notification).and_return(double(deliver_later: true)) allow(dossier).to receive(:build_attestation).and_return(attestation) Timecop.freeze(now) end after { Timecop.return } subject { dossier.accepter_automatiquement! dossier.reload } context 'as declarative procedure' do let(:dossier) { create(:dossier, :en_construction, :with_individual, :with_declarative_accepte) } it 'accepts dossier automatiquement' do expect(subject.motivation).to eq(nil) expect(subject.en_instruction_at).to eq(now) expect(subject.processed_at).to eq(now) expect(subject.declarative_triggered_at).to eq(now) expect(subject.sva_svr_decision_triggered_at).to be_nil expect(subject).to be_accepte expect(last_operation.operation).to eq('accepter') expect(last_operation.automatic_operation?).to be_truthy expect(NotificationMailer).to have_received(:send_accepte_notification).with(dossier) expect(subject.attestation).to eq(attestation) end end context 'as sva procedure' do let(:procedure) { create(:procedure, :for_individual, :published, :sva) } let(:dossier) { create(:dossier, :en_instruction, :with_individual, procedure:, sva_svr_decision_on: Date.current, en_instruction_at: DateTime.new(2021, 5, 1, 12)) } it 'accepts dossier automatiquement' do expect(subject.motivation).to eq(nil) expect(subject.en_instruction_at).to eq(DateTime.new(2021, 5, 1, 12)) expect(subject.processed_at).to eq(now) expect(subject.declarative_triggered_at).to be_nil expect(subject.sva_svr_decision_triggered_at).to eq(now) expect(subject).to be_accepte expect(last_operation.operation).to eq('accepter') expect(last_operation.automatic_operation?).to be_truthy expect(NotificationMailer).to have_received(:send_accepte_notification).with(dossier) expect(subject.attestation).to eq(attestation) expect(dossier.commentaires.count).to eq(1) end end end describe '#refuser_automatiquement' do context 'as svr procedure' do let(:last_operation) { dossier.dossier_operation_logs.last } let(:procedure) { create(:procedure, :for_individual, :published, :svr) } let(:dossier) { create(:dossier, :en_instruction, :with_individual, procedure:, sva_svr_decision_on: Date.current, en_instruction_at: DateTime.new(2021, 5, 1, 12)) } before { freeze_time allow(NotificationMailer).to receive(:send_refuse_notification).and_return(double(deliver_later: true)) } subject { dossier.refuser_automatiquement! dossier.reload } it 'refuses dossier automatiquement' do expect(subject.en_instruction_at).to eq(DateTime.new(2021, 5, 1, 12)) expect(subject.processed_at).to eq(Time.current) expect(subject.declarative_triggered_at).to be_nil expect(subject.sva_svr_decision_triggered_at).to eq(Time.current) expect(subject.motivation).to include("dans le délai imparti") expect(subject).to be_refuse expect(last_operation.operation).to eq('refuser') expect(last_operation.automatic_operation?).to be_truthy expect(NotificationMailer).to have_received(:send_refuse_notification).with(dossier) expect(subject.attestation).to be_nil expect(dossier.commentaires.count).to eq(1) end context 'for an user having english locale' do before { dossier.user.update!(locale: 'en') } it 'translates the motivation' do expect(subject.motivation).to include('within the time limit') end end end end describe '#passer_en_instruction!' do let(:dossier) { create(:dossier, :en_construction, en_construction_close_to_expiration_notice_sent_at: Time.zone.now) } let(:last_operation) { dossier.dossier_operation_logs.last } let(:operation_serialized) { last_operation.data } let(:instructeur) { create(:instructeur) } let!(:correction) { create(:dossier_correction, dossier:) } # correction has a commentaire subject(:passer_en_instruction) { dossier.passer_en_instruction!(instructeur: instructeur) } it do passer_en_instruction expect(dossier.state).to eq('en_instruction') expect(dossier.followers_instructeurs).to include(instructeur) expect(dossier.en_construction_close_to_expiration_notice_sent_at).to be_nil expect(last_operation.operation).to eq('passer_en_instruction') expect(last_operation.automatic_operation?).to be_falsey expect(operation_serialized['operation']).to eq('passer_en_instruction') expect(operation_serialized['dossier_id']).to eq(dossier.id) expect(operation_serialized['executed_at']).to eq(last_operation.executed_at.iso8601) end it { expect { passer_en_instruction }.to change { dossier.commentaires.count }.by(1) } it "resolve pending correction" do passer_en_instruction expect(dossier.pending_correction?).to be_falsey expect(correction.reload.resolved_at).to be_present end it 'creates a commentaire in the messagerie with expected wording' do passer_en_instruction email_template = dossier.procedure.email_template_for(dossier.state) commentaire = dossier.commentaires.last expect(commentaire.body).to include(email_template.subject_for_dossier(dossier), email_template.body_for_dossier(dossier)) expect(commentaire.dossier).to eq(dossier) end end describe '#passer_automatiquement_en_instruction!' do let(:last_operation) { dossier.dossier_operation_logs.last } let(:operation_serialized) { last_operation.data } let(:instructeur) { create(:instructeur) } context "via procedure declarative en instruction" do let(:dossier) { create(:dossier, :en_construction, :with_declarative_en_instruction, en_construction_close_to_expiration_notice_sent_at: Time.zone.now) } subject do dossier.process_declarative! dossier.reload end it 'passes dossier en instruction' do expect(subject.followers_instructeurs).not_to include(instructeur) expect(subject.en_construction_close_to_expiration_notice_sent_at).to be_nil expect(subject.declarative_triggered_at).to be_within(1.second).of(Time.current) expect(last_operation.operation).to eq('passer_en_instruction') expect(last_operation.automatic_operation?).to be_truthy expect(operation_serialized['operation']).to eq('passer_en_instruction') expect(operation_serialized['dossier_id']).to eq(dossier.id) expect(operation_serialized['executed_at']).to eq(last_operation.executed_at.iso8601) expect(dossier.commentaires.count).to eq(1) end end context "via procedure sva" do let(:procedure) { create(:procedure, :sva, :published, :for_individual) } let(:dossier) { create(:dossier, :en_construction, :with_individual, procedure:, sva_svr_decision_on: 10.days.from_now) } let(:sva_svr_decision_on) { SVASVRDecisionDateCalculatorService.new(dossier, procedure).decision_date } subject do dossier.process_sva_svr! dossier.reload end it 'passes dossier en instruction' do expect(subject.state).to eq('en_instruction') expect(subject.followers_instructeurs).not_to include(instructeur) expect(subject.sva_svr_decision_on).to eq(sva_svr_decision_on) expect(last_operation.operation).to eq('passer_en_instruction') expect(last_operation.automatic_operation?).to be_truthy expect(operation_serialized['operation']).to eq('passer_en_instruction') expect(operation_serialized['dossier_id']).to eq(dossier.id) expect(operation_serialized['executed_at']).to eq(last_operation.executed_at.iso8601) end context 'when dossier was submitted with sva not yet enabled' do let(:dossier) { create(:dossier, :en_construction, :with_individual, procedure:, depose_at: 10.days.ago) } it 'leaves dossier en construction' do expect(subject.sva_svr_decision_on).to be_nil expect(subject.state).to eq('en_construction') end end end end describe '#can_repasser_en_construction?' do let(:dossier) { create(:dossier, :en_instruction) } it { expect(dossier.can_repasser_en_construction?).to be_truthy } context 'when procedure is sva' do let(:dossier) { create(:dossier, :en_instruction, procedure: create(:procedure, :sva)) } it { expect(dossier.can_repasser_en_construction?).to be_falsey } end end describe '#can_passer_en_instruction?' do let(:dossier) { create(:dossier, :en_construction) } it { expect(dossier.can_passer_en_instruction?).to be_truthy } context 'when there is a pending correction' do before { create(:dossier_correction, dossier:) } it { expect(dossier.can_passer_en_instruction?).to be_truthy } end context 'when there is a pending correction with procedure blocking_pending_correction feature' do let(:resolved_at) { nil } before do Flipper.enable(:blocking_pending_correction, dossier.procedure) create(:dossier_correction, dossier:, resolved_at: resolved_at) end it { expect(dossier.can_passer_en_instruction?).to be_falsey } context 'when there is a resolved correction' do let(:resolved_at) { Time.current } it { expect(dossier.can_passer_en_instruction?).to be_truthy } end end end describe '#can_passer_automatiquement_en_instruction?' do let(:dossier) { create(:dossier, :en_construction, declarative_triggered_at: declarative_triggered_at) } let(:declarative_triggered_at) { nil } it { expect(dossier.can_passer_automatiquement_en_instruction?).to be_falsey } context 'when dossier is declarative' do before { dossier.procedure.update(declarative_with_state: :en_instruction) } context 'when dossier never transitioned' do it { expect(dossier.can_passer_automatiquement_en_instruction?).to be_truthy } end context 'when dossier transitioned before' do let(:declarative_triggered_at) { 1.day.ago } it { expect(dossier.can_passer_automatiquement_en_instruction?).to be_falsey } end end context 'when procedure has auto archive set' do before { dossier.procedure.update(auto_archive_on: 1.day.ago) } it { expect(dossier.can_passer_automatiquement_en_instruction?).to be_truthy } context 'when auto_archive_on is in the future' do before { dossier.procedure.update(auto_archive_on: 1.day.from_now) } it { expect(dossier.can_passer_automatiquement_en_instruction?).to be_falsey } end context 'when dossier transitioned before' do let(:declarative_triggered_at) { 1.day.ago } it { expect(dossier.can_passer_automatiquement_en_instruction?).to be_truthy } end context 'when there are pending correction' do before { create(:dossier_correction, dossier:) } it "passes en instruction and keep the correction request" do expect(dossier.can_passer_automatiquement_en_instruction?).to be_truthy expect(dossier.pending_correction?).to be_truthy end end end context 'when procedure has sva or svr enabled' do let(:procedure) { create(:procedure, :published, :sva) } let(:dossier) { create(:dossier, :en_construction, procedure:) } it { expect(dossier.can_passer_automatiquement_en_instruction?).to be_truthy } context 'when dossier was already processed by sva' do let(:dossier) { create(:dossier, :en_construction, procedure:, sva_svr_decision_triggered_at: 1.hour.ago) } it { expect(dossier.can_passer_automatiquement_en_instruction?).to be_falsey } end end end describe '#can_accepter_automatiquement?' do let(:dossier) { create(:dossier, state: initial_state, declarative_triggered_at: declarative_triggered_at) } let(:initial_state) { :en_construction } let(:declarative_triggered_at) { nil } it { expect(dossier.can_accepter_automatiquement?).to be_falsey } context 'when dossier is declarative' do before { dossier.procedure.update(declarative_with_state: :accepte) } context 'when dossier never transitioned' do it { expect(dossier.can_accepter_automatiquement?).to be_truthy } end context 'when dossier transitioned before' do let(:declarative_triggered_at) { 1.day.ago } it { expect(dossier.can_accepter_automatiquement?).to be_falsey } end end context 'when procedure is sva/svr' do let(:decision) { :sva } let(:initial_state) { :en_instruction } before do dossier.procedure.update!(sva_svr: SVASVRConfiguration.new(decision:).attributes) dossier.update!(sva_svr_decision_on: Date.current) end it { expect(dossier.can_accepter_automatiquement?).to be_truthy } context 'when sva_svr_decision_on is in the future' do before { dossier.update!(sva_svr_decision_on: 1.day.from_now) } it { expect(dossier.can_accepter_automatiquement?).to be_falsey } end context 'when dossier has pending correction' do let(:dossier) { create(:dossier, :en_construction) } let!(:dossier_correction) { create(:dossier_correction, dossier:) } it { expect(dossier.can_accepter_automatiquement?).to be_falsey } end context 'when decision is svr' do let(:decision) { :svr } it { expect(dossier.can_accepter_automatiquement?).to be_falsey } end context 'when dossier was already processed by sva' do before { dossier.update!(sva_svr_decision_triggered_at: 1.hour.ago) } it { expect(dossier.can_accepter_automatiquement?).to be_falsey } end end end describe '#can_refuser_automatiquement?' do let(:dossier) { create(:dossier, state: initial_state) } let(:initial_state) { :en_instruction } it { expect(dossier.can_refuser_automatiquement?).to be_falsey } context 'when procedure is sva/svr' do let(:decision) { :svr } before do dossier.procedure.update!(sva_svr: SVASVRConfiguration.new(decision:).attributes) dossier.update!(sva_svr_decision_on: Date.current) end it { expect(dossier.can_refuser_automatiquement?).to be_truthy } context 'when procedure is svr' do let(:decision) { :svr } before do dossier.procedure.update!(sva_svr: SVASVRConfiguration.new(decision:).attributes) dossier.update!(sva_svr_decision_on: Date.current) end it { expect(dossier.can_refuser_automatiquement?).to be_truthy } context 'when sva_svr_decision_on is in the future' do before { dossier.update!(sva_svr_decision_on: 1.day.from_now) } it { expect(dossier.can_refuser_automatiquement?).to be_falsey } end context 'when dossier has pending correction' do let(:dossier) { create(:dossier, :en_construction) } let!(:dossier_correction) { create(:dossier_correction, dossier:) } it { expect(dossier.can_refuser_automatiquement?).to be_falsey } end context 'when decision is sva' do let(:decision) { :sva } it { expect(dossier.can_refuser_automatiquement?).to be_falsey } end context 'when dossier was already processed by svr' do before { dossier.update!(sva_svr_decision_triggered_at: 1.hour.ago) } it { expect(dossier.can_refuser_automatiquement?).to be_falsey } end end end end describe "can't transition to terminer when etablissement is in degraded mode" do let(:instructeur) { create(:instructeur) } let(:motivation) { 'motivation' } context "when dossier is en_instruction" do let(:dossier_incomplete) { create(:dossier, :en_instruction, :with_entreprise, as_degraded_mode: true) } let(:dossier_ok) { create(:dossier, :en_instruction, :with_entreprise, as_degraded_mode: false) } it "can't accepter" do expect(dossier_incomplete.may_accepter?(instructeur:, motivation:)).to be_falsey expect(dossier_ok.accepter(instructeur:, motivation:)).to be_truthy end it "can't refuser" do expect(dossier_incomplete.may_refuser?(instructeur:, motivation:)).to be_falsey expect(dossier_ok.may_refuser?(instructeur:, motivation:)).to be_truthy end it "can't classer_sans_suite" do expect(dossier_incomplete.may_classer_sans_suite?(instructeur:, motivation:)).to be_falsey expect(dossier_ok.may_classer_sans_suite?(instructeur:, motivation:)).to be_truthy end end context "when dossier is en_construction" do let(:dossier_incomplete) { create(:dossier, :en_construction, :with_entreprise, :with_declarative_accepte, as_degraded_mode: true) } let(:dossier_ok) { create(:dossier, :en_construction, :with_entreprise, :with_declarative_accepte, as_degraded_mode: false) } it "can't accepter_automatiquement" do expect(dossier_incomplete.may_accepter_automatiquement?(instructeur:, motivation:)).to be_falsey expect(dossier_ok.accepter_automatiquement(instructeur:, motivation:)).to be_truthy end end context "when a SIRET champ has etablissement in degraded mode" do let(:procedure) { create(:procedure, types_de_champ_public: [{ type: :siret }]) } let(:dossier_incomplete) { create(:dossier, :en_instruction, :with_populated_champs, procedure:) } let(:dossier_ok) { create(:dossier, :en_instruction, :with_populated_champs, procedure:) } before do dossier_incomplete.champs.first.update(etablissement: Etablissement.new(siret: build(:etablissement).siret)) end it "can't accepter" do expect(dossier_incomplete.may_accepter?(instructeur:, motivation:)).to be_falsey expect(dossier_ok.may_accepter?(instructeur:, motivation:)).to be_truthy end end end describe "#check_mandatory_and_visible_champs" do include Logic let(:procedure) { create(:procedure, types_de_champ_public: types_de_champ) } let(:dossier) { create(:dossier, procedure: procedure) } let(:types_de_champ) { [type_de_champ].compact } let(:type_de_champ) { nil } let(:errors) { dossier.check_mandatory_and_visible_champs } it 'no mandatory champs' do expect(errors).to be_empty end context "with mandatory champs" do let(:type_de_champ) { { mandatory: true } } let(:champ_with_error) { dossier.champs_public.first } before do champ_with_error.value = nil champ_with_error.save end it 'should have errors' do expect(errors).not_to be_empty expect(errors.first.full_message).to eq("doit être rempli") end context "conditionaly visible" do let(:types_de_champ) { [{ type: :yes_no, stable_id: 99, mandatory: false }, type_de_champ] } let(:type_de_champ) { { mandatory: true, condition: ds_eq(champ_value(99), constant(true)) } } it 'should not have errors' do expect(errors).to be_empty end end end context "with mandatory SIRET champ" do let(:type_de_champ) { { type: :siret, mandatory: true } } let(:champ_siret) { dossier.champs.first } before do champ_siret.update(value: '44011762001530') end it 'should not have errors' do expect(errors).to be_empty end context "and invalid SIRET" do before do champ_siret.update(value: "1234") dossier.reload end it 'should have errors' do expect(errors).not_to be_empty expect(errors.first.full_message).to eq("doit être rempli") end end end context "with champ repetition" do let(:type_de_champ) { { type: :repetition, mandatory: true, children: [{ mandatory: true }] } } let(:revision) { procedure.active_revision } let(:type_de_champ_repetition) { revision.types_de_champ.first } context "when no champs" do let(:champ_with_error) { dossier.champs_public.first } it 'should have errors' do dossier.champs_public.first.champs.destroy_all expect(dossier.champs_public.first.rows).to be_empty expect(errors).not_to be_empty expect(errors.first.full_message).to eq("doit être rempli") end end context "when mandatory champ inside repetition" do let(:champ_with_error) { dossier.champs_public.first.champs.first } it 'should have errors' do expect(dossier.champs_public.first.rows).not_to be_empty expect(errors.first.full_message).to eq("doit être rempli") end context "conditionaly visible" do let(:champ_with_error) { dossier.champs_public.second.champs.first } let(:types_de_champ) { [{ type: :yes_no, stable_id: 99, mandatory: false }, type_de_champ] } let(:type_de_champ) { { type: :repetition, mandatory: true, children: [{ mandatory: true }], condition: ds_eq(champ_value(99), constant(true)) } } it 'should not have errors' do expect(dossier.champs_public.second.rows).not_to be_empty expect(errors).to be_empty end it 'should have errors' do dossier.champs_public.first.update(value: 'true') expect(dossier.champs_public.second.rows).not_to be_empty expect(errors).not_to be_empty expect(errors.first.full_message).to eq("doit être rempli") end end end end end describe "#check_expressions_regulieres_champs" do let(:procedure) { create(:procedure, types_de_champ_public: types_de_champ) } let(:dossier) { create(:dossier, procedure: procedure) } let(:types_de_champ) { [type_de_champ] } let(:type_de_champ) { { type: :expression_reguliere, expression_reguliere:, expression_reguliere_exemple_text:, expression_reguliere_error_message: } } context "with bad example" do let(:expression_reguliere_exemple_text) { "01234567" } let(:expression_reguliere) { "[A-Z]+" } let(:expression_reguliere_error_message) { "Le champ doit être composé de lettres majuscules" } before do champ = dossier.champs_public.first champ.value = expression_reguliere_exemple_text dossier.save(context: :champs_public_value) end it 'should have errors' do expect(dossier.errors).not_to be_empty expect(dossier.errors.full_messages.join(',')).to include(dossier.champs_public.first.expression_reguliere_error_message) end end context "with good example" do let(:expression_reguliere_exemple_text) { "AZERTY" } let(:expression_reguliere) { "[A-Z]+" } let(:expression_reguliere_error_message) { "Le champ doit être composé de lettres majuscules" } before do champ = dossier.champs_public.first champ.value = expression_reguliere_exemple_text dossier.save end it 'should not have errors' do expect(dossier.errors).to be_empty end end end describe 'index_for_section_header' do let(:types_de_champ_public) { [{ type: :repetition, mandatory: true, children: [{ type: :header_section }] }] } let(:procedure) { create(:procedure, types_de_champ_public:) } let(:dossier) { create(:dossier, procedure:) } let(:header_in_repetition) { dossier.revision.types_de_champ.find(&:header_section?) } it 'index classly' do expect(dossier.index_for_section_header(header_in_repetition)).to eq("1.1") end end describe '#repasser_en_instruction!' do let(:dossier) { create(:dossier, :refuse, :with_attestation, :with_justificatif, archived: true, termine_close_to_expiration_notice_sent_at: Time.zone.now, sva_svr_decision_on: 1.day.ago) } let!(:instructeur) { create(:instructeur) } let(:last_operation) { dossier.dossier_operation_logs.last } before do freeze_time allow(NotificationMailer).to receive(:send_repasser_en_instruction_notification).and_return(double(deliver_later: true)) dossier.repasser_en_instruction!(instructeur: instructeur) dossier.reload end it "update attributes" do expect(dossier.state).to eq('en_instruction') expect(dossier.archived).to be_falsey expect(dossier.motivation).to be_nil expect(dossier.justificatif_motivation.attached?).to be_falsey expect(dossier.attestation).to be_nil expect(dossier.sva_svr_decision_on).to be_nil expect(dossier.termine_close_to_expiration_notice_sent_at).to be_nil expect(last_operation.operation).to eq('repasser_en_instruction') expect(last_operation.data['author']['email']).to eq(instructeur.email) expect(NotificationMailer).to have_received(:send_repasser_en_instruction_notification).with(dossier) end end describe '#notify_draft_not_submitted' do let!(:user1) { create(:user) } let!(:user2) { create(:user) } let!(:procedure_near_closing) { create(:procedure, :published, auto_archive_on: Time.zone.today + Dossier::REMAINING_DAYS_BEFORE_CLOSING.days) } let!(:procedure_closed_later) { create(:procedure, :published, auto_archive_on: Time.zone.today + Dossier::REMAINING_DAYS_BEFORE_CLOSING.days + 1.day) } let!(:procedure_closed_before) { create(:procedure, :published, auto_archive_on: Time.zone.today + Dossier::REMAINING_DAYS_BEFORE_CLOSING.days - 1.day) } # user 1 has three draft dossiers where one is for procedure that closes in two days ==> should trigger one mail let!(:draft_near_closing) { create(:dossier, user: user1, procedure: procedure_near_closing) } let!(:draft_before) { create(:dossier, user: user1, procedure: procedure_closed_before) } let!(:draft_later) { create(:dossier, user: user1, procedure: procedure_closed_later) } # user 2 submitted a draft and en_construction dossier for the same procedure ==> should not trigger the mail let!(:draft_near_closing_2) { create(:dossier, :en_construction, user: user2, procedure: procedure_near_closing) } let!(:submitted_near_closing_2) { create(:dossier, user: user2, procedure: procedure_near_closing) } before do allow(DossierMailer).to receive(:notify_brouillon_not_submitted).and_return(double(deliver_later: nil)) Dossier.notify_draft_not_submitted end it 'notifies draft is not submitted' do expect(DossierMailer).to have_received(:notify_brouillon_not_submitted).once expect(DossierMailer).to have_received(:notify_brouillon_not_submitted).with(draft_near_closing) end end describe '#geo_position' do let(:lat) { "46.538192" } let(:lon) { "2.428462" } let(:zoom) { "13" } let(:etablissement_geo_adresse_lat) { "40.7143528" } let(:etablissement_geo_adresse_lon) { "-74.0059731" } let(:result) { { lat: lat, lon: lon, zoom: zoom } } let(:dossier) { create(:dossier) } it 'should geolocate' do expect(dossier.geo_position).to eq(result) end context 'with etablissement' do before do Geocoder::Lookup::Test.add_stub( dossier.etablissement.geo_adresse, [ { 'coordinates' => [etablissement_geo_adresse_lat.to_f, etablissement_geo_adresse_lon.to_f] } ] ) end let(:dossier) { create(:dossier, :with_entreprise) } let(:result) { { lat: etablissement_geo_adresse_lat, lon: etablissement_geo_adresse_lon, zoom: zoom } } it 'should geolocate' do expect(dossier.geo_position).to eq(result) end end end describe '#geo_data' do let(:procedure) { create(:procedure, types_de_champ_public:, types_de_champ_private:) } let(:dossier) { create(:dossier, :with_populated_champs, :with_populated_annotations, procedure:) } let(:types_de_champ_public) { [] } let(:types_de_champ_private) { [] } context "without data" do it { expect(dossier.geo_data?).to be_falsey } end context "with geo data in public champ" do let(:types_de_champ_public) { [{ type: :carte }] } it { expect(dossier.geo_data?).to be_truthy } end context "with geo data in private champ" do let(:types_de_champ_private) { [{ type: :carte }] } it { expect(dossier.geo_data?).to be_truthy } end context "should solve N+1 problem" do let(:types_de_champ_public) { [{ type: :carte }, { type: :carte }, { type: :carte }] } it do dossier.champs_for_revision count = 0 callback = lambda { |*_args| count += 1 } ActiveSupport::Notifications.subscribed(callback, "sql.active_record") do dossier.geo_data? end expect(count).to eq(1) end end end describe 'dossier_operation_log after dossier deletion' do let(:dossier) { create(:dossier) } let(:dossier_operation_log) { create(:dossier_operation_log, dossier: dossier) } it 'should nullify dossier link' do expect(dossier_operation_log.dossier).to eq(dossier) expect(DossierOperationLog.count).to eq(1) dossier.destroy expect(dossier_operation_log.reload.dossier).to be_nil expect(DossierOperationLog.count).to eq(1) end end describe 'brouillon_expired and en_construction_expired' do let(:administrateur) { administrateurs(:default_admin) } let(:user) { administrateur.user } let(:reason) { DeletedDossier.reasons.fetch(:user_request) } before do create(:dossier, user: user) create(:dossier, :en_construction, user: user) create(:dossier, user: user).hide_and_keep_track!(user, reason) create(:dossier, :en_construction, user: user).hide_and_keep_track!(user, reason) Timecop.travel(2.months.ago) do create(:dossier, user: user).hide_and_keep_track!(user, reason) create(:dossier, :en_construction, user: user).hide_and_keep_track!(user, reason) create(:dossier, user: user).procedure.discard_and_keep_track!(administrateur) create(:dossier, :en_construction, user: user).procedure.discard_and_keep_track!(administrateur) end Timecop.travel(1.week.ago) do create(:dossier, user: user).hide_and_keep_track!(user, reason) create(:dossier, :en_construction, user: user).hide_and_keep_track!(user, reason) end end it { expect(Dossier.en_brouillon_expired_to_delete.count).to eq(2) } it { expect(Dossier.en_construction_expired_to_delete.count).to eq(2) } end describe "discarded procedure dossier should be able to access it's procedure" do let(:dossier) { create(:dossier) } let(:procedure) { dossier.reload.procedure } before { dossier.procedure.discard! } it { expect(procedure).not_to be_nil } it { expect(procedure.discarded?).to be_truthy } end describe "to_feature_collection" do let(:procedure) { create(:procedure, types_de_champ_public: [{ type: :carte }]) } let(:dossier) { create(:dossier, :with_populated_champs, procedure:) } let(:champ_carte) { dossier.champs.first } let(:geo_area) { build(:geo_area, :selection_utilisateur, :polygon) } before do champ_carte.update(geo_areas: [geo_area]) end it 'should have all champs carto' do expect(dossier.to_feature_collection).to eq({ type: 'FeatureCollection', id: dossier.id, bbox: [2.428439855575562, 46.538491597754714, 2.42824137210846, 46.53841410755813], features: [ { type: 'Feature', geometry: { coordinates: [[[2.428439855575562, 46.538476837725796], [2.4284291267395024, 46.53842148758162], [2.4282521009445195, 46.53841410755813], [2.42824137210846, 46.53847314771794], [2.428284287452698, 46.53847314771794], [2.428364753723145, 46.538487907747864], [2.4284291267395024, 46.538491597754714], [2.428439855575562, 46.538476837725796]]], type: 'Polygon' }, properties: { area: 103.6, champ_label: champ_carte.libelle, champ_id: champ_carte.stable_id, champ_private: false, dossier_id: dossier.id, id: geo_area.id, source: 'selection_utilisateur' } } ] }) end end describe "with_notifiable_procedure" do let(:test_procedure) { create(:procedure) } let(:published_procedure) { create(:procedure, :published) } let(:closed_procedure) { create(:procedure, :closed) } let(:unpublished_procedure) { create(:procedure, :unpublished) } let!(:dossier_on_test_procedure) { create(:dossier, procedure: test_procedure) } let!(:dossier_on_published_procedure) { create(:dossier, procedure: published_procedure) } let!(:dossier_on_closed_procedure) { create(:dossier, procedure: closed_procedure) } let!(:dossier_on_unpublished_procedure) { create(:dossier, procedure: unpublished_procedure) } let(:notify_on_closed) { false } let(:dossiers) { Dossier.with_notifiable_procedure(notify_on_closed: notify_on_closed) } it 'should find dossiers with notifiable procedure' do expect(dossiers).to match_array([dossier_on_published_procedure, dossier_on_unpublished_procedure]) end context 'when notify on closed is true' do let(:notify_on_closed) { true } it 'should find dossiers with notifiable procedure' do expect(dossiers).to match_array([dossier_on_published_procedure, dossier_on_closed_procedure, dossier_on_unpublished_procedure]) end end end describe "champs_for_export" do context 'with a unconditionnal procedure' do let(:procedure) { create(:procedure, types_de_champ_public:, zones: [create(:zone)]) } let(:types_de_champ_public) do [ { type: :text }, { type: :datetime }, { type: :yes_no }, { type: :explication }, { type: :communes }, { type: :repetition, children: [{ type: :text }] } ] end let(:text_type_de_champ) { procedure.active_revision.types_de_champ_public.find { |type_de_champ| type_de_champ.type_champ == TypeDeChamp.type_champs.fetch(:text) } } let(:yes_no_type_de_champ) { procedure.active_revision.types_de_champ_public.find { |type_de_champ| type_de_champ.type_champ == TypeDeChamp.type_champs.fetch(:yes_no) } } let(:datetime_type_de_champ) { procedure.active_revision.types_de_champ_public.find { |type_de_champ| type_de_champ.type_champ == TypeDeChamp.type_champs.fetch(:datetime) } } let(:explication_type_de_champ) { procedure.active_revision.types_de_champ_public.find { |type_de_champ| type_de_champ.type_champ == TypeDeChamp.type_champs.fetch(:explication) } } let(:commune_type_de_champ) { procedure.active_revision.types_de_champ_public.find { |type_de_champ| type_de_champ.type_champ == TypeDeChamp.type_champs.fetch(:communes) } } let(:repetition_type_de_champ) { procedure.active_revision.types_de_champ_public.find { |type_de_champ| type_de_champ.type_champ == TypeDeChamp.type_champs.fetch(:repetition) } } let(:repetition_champ) { dossier.champs_public.find(&:repetition?) } let(:repetition_second_revision_champ) { dossier_second_revision.champs_public.find(&:repetition?) } let(:dossier) { create(:dossier, procedure: procedure) } let(:dossier_second_revision) { create(:dossier, procedure: procedure) } let(:dossier_champs_for_export) { dossier.champs_for_export(procedure.types_de_champ_for_procedure_presentation.not_repetition) } let(:dossier_second_revision_champs_for_export) { dossier_second_revision.champs_for_export(procedure.types_de_champ_for_procedure_presentation.not_repetition) } context "when procedure published" do before do procedure.publish! dossier procedure.draft_revision.remove_type_de_champ(text_type_de_champ.stable_id) coordinate = procedure.draft_revision.add_type_de_champ(type_champ: TypeDeChamp.type_champs.fetch(:text), libelle: 'New text field', after_stable_id: repetition_champ.stable_id) procedure.draft_revision.find_and_ensure_exclusive_use(yes_no_type_de_champ.stable_id).update(libelle: 'Updated yes/no') procedure.draft_revision.find_and_ensure_exclusive_use(commune_type_de_champ.stable_id).update(libelle: 'Commune de naissance') procedure.draft_revision.find_and_ensure_exclusive_use(repetition_type_de_champ.stable_id).update(libelle: 'Repetition') procedure.update(published_revision: procedure.draft_revision, draft_revision: procedure.create_new_revision) dossier.reload procedure.reload end it "should have champs from all revisions" do expect(dossier.types_de_champ.map(&:libelle)).to eq([text_type_de_champ.libelle, datetime_type_de_champ.libelle, "Yes/no", explication_type_de_champ.libelle, commune_type_de_champ.libelle, repetition_type_de_champ.libelle]) expect(dossier_second_revision.types_de_champ.map(&:libelle)).to eq([datetime_type_de_champ.libelle, "Updated yes/no", explication_type_de_champ.libelle, 'Commune de naissance', "Repetition", "New text field"]) expect(dossier_champs_for_export.map { |(libelle)| libelle }).to eq([datetime_type_de_champ.libelle, text_type_de_champ.libelle, "Updated yes/no", "Commune de naissance", "Commune de naissance (Code INSEE)", "Commune de naissance (Département)", "New text field"]) expect(dossier_champs_for_export).to eq(dossier_second_revision_champs_for_export) end context 'within a repetition having a type de champs commune (multiple values for export)' do it 'works' do proc_test = create(:procedure) draft = proc_test.draft_revision tdc_repetition = draft.add_type_de_champ(type_champ: :repetition, libelle: "repetition") draft.add_type_de_champ(type_champ: :communes, libelle: "communes", parent_stable_id: tdc_repetition.stable_id) dossier_test = create(:dossier, procedure: proc_test) repetition = proc_test.types_de_champ_for_procedure_presentation.repetition.first type_champs = proc_test.types_de_champ_for_procedure_presentation(repetition).to_a expect(type_champs.size).to eq(1) expect(dossier.champs_for_export(type_champs).size).to eq(3) end end end context "when procedure brouillon" do let(:procedure) { create(:procedure, types_de_champ_public: [{ type: :text }, { type: :explication }]) } it "should not contain non-exportable types de champ" do expect(dossier_champs_for_export.map { |(libelle)| libelle }).to eq([text_type_de_champ.libelle]) end end end context 'with a procedure with a condition' do include Logic let(:types_de_champ) { [{ type: :yes_no }, { type: :text }] } let(:procedure) { create(:procedure, types_de_champ_public: types_de_champ) } let(:dossier) { create(:dossier, procedure:) } let(:yes_no_tdc) { procedure.active_revision.types_de_champ_public.first } let(:text_tdc) { procedure.active_revision.types_de_champ_public.second } let(:tdcs) { dossier.champs_public.map(&:type_de_champ) } subject { dossier.champs_for_export(tdcs) } before do text_tdc.update(condition: ds_eq(champ_value(yes_no_tdc.stable_id), constant(true))) yes_no, text = dossier.champs_public yes_no.update(value: yes_no_value) text.update(value: 'text') end context 'with a champ visible' do let(:yes_no_value) { 'true' } it { is_expected.to eq([[yes_no_tdc.libelle, "Oui"], [text_tdc.libelle, "text"]]) } end context 'with a champ invisible' do let(:yes_no_value) { 'false' } it { is_expected.to eq([[yes_no_tdc.libelle, "Non"], [text_tdc.libelle, nil]]) } end context 'with another revision' do let(:tdc_from_another_revision) { create(:type_de_champ_communes, libelle: 'commune', condition: ds_eq(constant(true), constant(true))) } let(:tdcs) { dossier.champs_public.map(&:type_de_champ) << tdc_from_another_revision } let(:yes_no_value) { 'true' } let(:expected) do [ [yes_no_tdc.libelle, "Oui"], [text_tdc.libelle, "text"], ["commune", nil], ["commune (Code INSEE)", nil], ["commune (Département)", nil] ] end it { is_expected.to eq(expected) } end end end describe "remove_titres_identite!" do let(:declarative_with_state) { nil } let(:procedure) { create(:procedure, declarative_with_state:, types_de_champ_public: [{ type: :titre_identite }, { type: :titre_identite }]) } let(:dossier) { create(:dossier, :en_instruction, :followed, :with_populated_champs, procedure:) } let(:champ_titre_identite) { dossier.champs.first } let(:champ_titre_identite_vide) { dossier.champs.second } before do champ_titre_identite_vide.piece_justificative_file.purge end it "clean up titres identite on accepter" do expect(champ_titre_identite.piece_justificative_file.attached?).to be_truthy expect(champ_titre_identite_vide.piece_justificative_file.attached?).to be_falsey dossier.accepter!(instructeur: dossier.followers_instructeurs.first, motivation: "yolo!") expect(champ_titre_identite.piece_justificative_file.attached?).to be_falsey end it "clean up titres identite on refuser" do expect(champ_titre_identite.piece_justificative_file.attached?).to be_truthy expect(champ_titre_identite_vide.piece_justificative_file.attached?).to be_falsey dossier.refuser!(instructeur: dossier.followers_instructeurs.first, motivation: "yolo!") expect(champ_titre_identite.piece_justificative_file.attached?).to be_falsey end it "clean up titres identite on classer_sans_suite" do expect(champ_titre_identite.piece_justificative_file.attached?).to be_truthy expect(champ_titre_identite_vide.piece_justificative_file.attached?).to be_falsey dossier.classer_sans_suite!(instructeur: dossier.followers_instructeurs.first, motivation: "yolo!") expect(champ_titre_identite.piece_justificative_file.attached?).to be_falsey end context 'en_construction' do let(:declarative_with_state) { 'accepte' } let(:dossier) { create(:dossier, :en_construction, :followed, :with_populated_champs, procedure:) } it "clean up titres identite on accepter_automatiquement" do expect(champ_titre_identite.piece_justificative_file.attached?).to be_truthy expect(champ_titre_identite_vide.piece_justificative_file.attached?).to be_falsey dossier.accepter_automatiquement! expect(champ_titre_identite.piece_justificative_file.attached?).to be_falsey end end end describe '#log_api_entreprise_job_exception' do let(:dossier) { create(:dossier) } context "add execption to the log" do before do dossier.log_api_entreprise_job_exception(StandardError.new('My special exception!')) end it { expect(dossier.api_entreprise_job_exceptions).to eq(['#']) } end end describe "#destroy" do let(:procedure) { create(:procedure, :with_all_champs, :with_all_annotations) } let(:transfer) { create(:dossier_transfer) } let(:dossier) { create(:dossier, :with_populated_champs, :with_populated_annotations, transfer: transfer, procedure: procedure) } before do create(:dossier, transfer: transfer) create(:attestation, dossier: dossier) end it "can destroy dossier, reset demarche, logg context" do json_message = nil allow(Rails.logger).to receive(:info) { json_message ||= _1 } expect(dossier.destroy).to be_truthy expect { dossier.reload }.to raise_error(ActiveRecord::RecordNotFound) expect(JSON.parse(json_message)).to a_hash_including( { message: "Dossier destroyed", dossier_id: dossier.id, procedure_id: procedure.id }.stringify_keys ) expect { dossier.procedure.reset! }.not_to raise_error expect { dossier.reload }.to raise_error(ActiveRecord::RecordNotFound) end end describe "#spreadsheet_columns" do let(:dossier) { create(:dossier) } context 'user france connected' do let(:dossier) { build(:dossier, user: build(:user, france_connect_informations: [build(:france_connect_information)])) } it { expect(dossier.spreadsheet_columns(types_de_champ: [])).to include(["FranceConnect ?", true]) } end context 'user not france connected' do let(:dossier) { build(:dossier) } it { expect(dossier.spreadsheet_columns(types_de_champ: [])).to include(["FranceConnect ?", false]) } end context 'for_individual' do let(:dossier) { create(:dossier, procedure: create(:procedure, :for_individual)) } it { expect(dossier.spreadsheet_columns(types_de_champ: [])).to include(["Dépôt pour un tiers", :for_tiers]) } it { expect(dossier.spreadsheet_columns(types_de_champ: [])).to include(['Nom du mandataire', :mandataire_last_name]) } it { expect(dossier.spreadsheet_columns(types_de_champ: [])).to include(['Prénom du mandataire', :mandataire_first_name]) } end it { expect(dossier.spreadsheet_columns(types_de_champ: [])).to include(["État du dossier", "Brouillon"]) } context 'procedure sva' do let(:dossier) { build(:dossier, :en_instruction, procedure: create(:procedure, :sva)) } it { expect(dossier.spreadsheet_columns(types_de_champ: [])).to include(["Date décision SVA", :sva_svr_decision_on]) } end context 'procedure svr' do let(:dossier) { build(:dossier, :en_instruction, procedure: create(:procedure, :svr)) } it { expect(dossier.spreadsheet_columns(types_de_champ: [])).to include(["Date décision SVR", :sva_svr_decision_on]) } end end describe '#processed_in_month' do let(:dossier_accepte_at) { DateTime.new(2022, 3, 31, 12, 0) } before do travel_to(dossier_accepte_at) do dossier = create(:dossier, :accepte) end end context 'given a date' do let(:archive_date) { Date.new(2022, 3, 1) } it 'includes a dossier processed_at at last day of month' do expect(Dossier.processed_in_month(archive_date).count).to eq(1) end end context 'given a datetime' do let(:archive_date) { DateTime.new(2022, 3, 1, 12, 0) } it 'includes a dossier processed_at at last day of month' do expect(Dossier.processed_in_month(archive_date).count).to eq(1) end end end describe '#processed_by_month' do let(:procedure) { create(:procedure, :published, groupe_instructeurs: [groupe_instructeurs]) } let(:groupe_instructeurs) { create(:groupe_instructeur) } before do create_dossier_for_month(procedure, 2021, 3) create_dossier_for_month(procedure, 2021, 3) create_archived_dossier_for_month(procedure, 2021, 3) create_dossier_for_month(procedure, 2021, 2) end subject do Timecop.freeze(Time.zone.local(2021, 3, 5)) do Dossier.processed_by_month(groupe_instructeurs).count end end it 'count dossiers_termines by month' do expect(count_for_month(subject, 3)).to eq 3 expect(count_for_month(subject, 2)).to eq 1 end it 'returns descending order by month' do expect(subject.keys.first.month).to eq 3 expect(subject.keys.last.month).to eq 2 end end describe 'BatchOperation' do subject { build(:dossier) } it { is_expected.to belong_to(:batch_operation).optional } end describe '#orphan?' do subject(:orphan) { dossier.orphan? } context 'when the dossier is prefilled' do context 'when the dossier has a user' do let(:dossier) { build(:dossier, :prefilled) } it { expect(orphan).to be_falsey } end context 'when the dossier does not have a user' do let(:dossier) { build(:dossier, :prefilled, user: nil) } it { expect(orphan).to be_truthy } end end context 'when the dossier is not prefilled' do context 'when the dossier has a user' do let(:dossier) { build(:dossier) } it { expect(orphan).to be_falsey } end context 'when the dossier does not have a user' do let(:dossier) { build(:dossier, user: nil) } it { expect(orphan).to be_falsey } end end end describe '#owned_by?' do subject(:owned_by) { dossier.owned_by?(user) } context 'when the dossier is orphan' do let(:dossier) { build(:dossier, user: nil) } let(:user) { build(:user) } it { expect(owned_by).to be_falsey } end context 'when the given user is nil' do let(:dossier) { build(:dossier) } let(:user) { nil } it { expect(owned_by).to be_falsey } end context 'when the dossier has a user and it is not the given user' do let(:dossier) { build(:dossier) } let(:user) { build(:user) } it { expect(owned_by).to be_falsey } end context 'when the dossier has a user and it is the given user' do let(:dossier) { build(:dossier, user: user) } let(:user) { build(:user) } it { expect(owned_by).to be_truthy } end end describe 'update procedure dossiers count' do let(:dossier) { create(:dossier, :brouillon, :with_individual) } it 'update procedure dossiers count when passing to construction' do expect(dossier.procedure).to receive(:compute_dossiers_count) dossier.passer_en_construction! end end describe '#sva_svr_decision_in_days' do let(:dossier) { create(:dossier, :en_instruction, sva_svr_decision_on: 10.days.from_now) } it { expect(dossier.sva_svr_decision_in_days).to eq 10 } end private def count_for_month(processed_by_month, month) processed_by_month.find { |date, _count| date.month == month }[1] end def create_dossier_for_month(procedure, year, month) Timecop.freeze(Time.zone.local(year, month, 5)) do create(:dossier, :accepte, :with_attestation, procedure: procedure) end end def create_archived_dossier_for_month(procedure, year, month) Timecop.freeze(Time.zone.local(year, month, 5)) do create(:dossier, :accepte, :archived, :with_attestation, procedure: procedure) end end end