demarches-normaliennes/spec/controllers/experts/avis_controller_spec.rb

531 lines
20 KiB
Ruby
Raw Normal View History

2021-02-28 22:20:24 +01:00
describe Experts::AvisController, type: :controller do
context 'with an expert signed in' do
render_views
let(:now) { Time.zone.parse('01/02/2345') }
let(:instructeur) { create(:instructeur) }
2022-08-31 16:23:46 +02:00
let!(:instructeur_with_instant_avis_notification) { create(:instructeur) }
let(:another_instructeur) { create(:instructeur) }
2021-02-28 22:20:24 +01:00
let(:claimant) { create(:expert) }
let(:expert) { create(:expert) }
2022-08-31 16:23:46 +02:00
let(:procedure) { create(:procedure, :published, instructeurs: [instructeur, another_instructeur, instructeur_with_instant_avis_notification]) }
2021-02-28 22:20:24 +01:00
let(:another_procedure) { create(:procedure, :published, instructeurs: [instructeur]) }
2022-11-28 14:35:53 +01:00
let(:dossier) { create(:dossier, :en_construction, procedure:) }
let(:experts_procedure) { create(:experts_procedure, expert:, procedure:) }
let!(:avis_without_answer) { create(:avis, dossier:, claimant:, experts_procedure:) }
let!(:avis_with_answer) { create(:avis, dossier:, claimant:, experts_procedure:, answer: 'yop') }
2021-02-28 22:20:24 +01:00
let!(:revoked_procedure) { create(:procedure, :published, instructeurs: [instructeur]) }
let!(:revoked_dossier) { create(:dossier, :en_construction, procedure: revoked_procedure) }
let!(:revoked_experts_procedure) { create(:experts_procedure, expert: expert, procedure: revoked_procedure) }
let!(:revoked_avis) do
create(:avis, dossier: revoked_dossier, claimant:, experts_procedure: revoked_experts_procedure, introduction: 'revoked', revoked_at: Time.zone.now)
end
2021-02-28 22:20:24 +01:00
before do
sign_in(expert.user)
end
describe '#index' do
before { get :index }
2022-11-28 14:24:56 +01:00
it do
expect(response).to have_http_status(:success)
expect(assigns(:avis_by_procedure).keys).to match_array(procedure)
expect(assigns(:avis_by_procedure).values.flatten).to match_array([avis_without_answer, avis_with_answer])
2022-11-28 14:24:56 +01:00
end
2021-02-28 22:20:24 +01:00
end
describe '#procedure' do
2022-02-28 14:24:41 +01:00
context 'without filter' do
let!(:oldest_avis_without_answer) { create(:avis, dossier: dossier, claimant: claimant, experts_procedure: experts_procedure, created_at: 2.years.ago) }
before { get :procedure, params: { procedure_id: procedure_to_show.id } }
2021-02-28 22:20:24 +01:00
context 'with legitimates avis' do
let(:procedure_to_show) { procedure }
it do
expect(response).to have_http_status(:success)
expect(assigns(:avis_a_donner)).to match([avis_without_answer, oldest_avis_without_answer])
expect(assigns(:avis_donnes)).to match([avis_with_answer])
expect(assigns(:statut)).to eq('a-donner')
end
end
context 'with a revoked avis' do
let(:procedure_to_show) { revoked_procedure }
it { expect(response).to redirect_to expert_all_avis_path }
2022-11-28 14:24:56 +01:00
end
2022-02-28 14:24:41 +01:00
end
2021-02-28 22:20:24 +01:00
context 'with a statut equal to donnes' do
before { get :procedure, params: { statut: 'donnes', procedure_id: procedure.id } }
it { expect(assigns(:statut)).to eq('donnes') }
end
2022-02-28 14:24:41 +01:00
context 'with different procedure' do
subject { get :procedure, params: { statut: 'donnes', procedure_id: procedure.id } }
it 'fails' do
sign_in(create(:expert).user)
subject
expect(response).to redirect_to(expert_all_avis_path)
expect(flash.alert).to eq("Vous navez pas accès à cette démarche.")
end
end
2021-02-28 22:20:24 +01:00
end
describe '#bilans_bdf' do
before { get :bilans_bdf, params: { id: avis_without_answer.id, procedure_id: procedure.id } }
it { expect(response).to redirect_to(instructeur_avis_path(avis_without_answer)) }
end
describe '#show' do
subject { get :show, params: { id: avis_with_answer.id, procedure_id: procedure.id } }
context 'with a valid avis' do
before { subject }
2022-11-28 14:24:56 +01:00
it do
expect(response).to have_http_status(:success)
expect(assigns(:avis)).to eq(avis_with_answer)
expect(assigns(:dossier)).to eq(dossier)
end
2021-02-28 22:20:24 +01:00
end
context 'with a revoked avis' do
it "refuse l'accès au dossier" do
avis_with_answer.update!(revoked_at: Time.zone.now)
subject
2021-05-26 15:16:30 +02:00
expect(flash.alert).to eq("Vous navez plus accès à ce dossier.")
2021-02-28 22:20:24 +01:00
expect(response).to redirect_to(root_path)
end
end
2022-02-28 14:24:41 +01:00
context 'with an avis that does not belongs to current_expert' do
it "refuse l'accès au dossier" do
sign_in(create(:expert).user)
subject
expect(response).to redirect_to(expert_all_avis_path)
expect(flash.alert).to eq("Vous navez pas accès à cet avis.")
end
end
2021-02-28 22:20:24 +01:00
end
describe '#instruction' do
subject { get :instruction, params: { id: avis_to_instruct.id, procedure_id: procedure.id } }
2022-02-28 14:24:41 +01:00
context 'with valid avis' do
let(:avis_to_instruct) { avis_without_answer }
2022-02-28 14:24:41 +01:00
before { subject }
2022-11-28 14:24:56 +01:00
it do
expect(response).to have_http_status(:success)
expect(assigns(:avis)).to eq(avis_without_answer)
expect(assigns(:dossier)).to eq(dossier)
end
2022-02-28 14:24:41 +01:00
end
2022-02-28 14:24:41 +01:00
context 'with an avis that does not belongs to current_expert' do
let(:avis_to_instruct) { avis_without_answer }
2022-02-28 14:24:41 +01:00
it "refuse l'accès au dossier" do
sign_in(create(:expert).user)
subject
expect(response).to redirect_to(expert_all_avis_path)
expect(flash.alert).to eq("Vous navez pas accès à cet avis.")
end
end
context 'with a revoked avis' do
let(:avis_to_instruct) { revoked_avis }
before { subject }
it { expect(response).to redirect_to root_path }
end
2021-02-28 22:20:24 +01:00
end
context 'with destroyed claimant' do
render_views
it 'does not raise' do
avis_with_merged_instructeur = create(:avis, dossier: dossier, claimant: another_instructeur, experts_procedure: experts_procedure)
another_instructeur.user.destroy
sign_in(expert.user)
get :instruction, params: { id: avis_with_merged_instructeur.id, procedure_id: procedure.id }
expect(response).to have_http_status(200)
end
end
2021-02-28 22:20:24 +01:00
describe '#messagerie' do
2022-02-28 14:24:41 +01:00
subject { get :messagerie, params: { id: avis_without_answer.id, procedure_id: procedure.id } }
context 'with valid avis' do
before { subject }
2021-02-28 22:20:24 +01:00
2022-11-28 14:24:56 +01:00
it do
expect(response).to have_http_status(:success)
expect(assigns(:avis)).to eq(avis_without_answer)
expect(assigns(:dossier)).to eq(dossier)
end
2022-02-28 14:24:41 +01:00
end
context 'with an avis that does not belongs to current_expert' do
it "refuse l'accès au dossier" do
sign_in(create(:expert).user)
subject
expect(response).to redirect_to(expert_all_avis_path)
expect(flash.alert).to eq("Vous navez pas accès à cet avis.")
end
end
2021-02-28 22:20:24 +01:00
end
describe '#update' do
2022-11-29 10:30:35 +01:00
before { Timecop.freeze(now) }
after { Timecop.return }
2021-02-28 22:20:24 +01:00
2022-11-29 10:30:35 +01:00
let(:avis) { avis_without_answer }
subject do
post :update, params: { id: avis.id, procedure_id: procedure.id, avis: { answer: 'answer' } }
avis.reload
end
context 'on a revoked avis' do
let(:avis) { revoked_avis }
it { expect(subject).to redirect_to(root_path) }
end
2022-11-29 10:30:35 +01:00
context 'without attachment' do
2021-02-28 22:20:24 +01:00
it 'should be ok' do
2022-11-29 10:30:35 +01:00
expect(subject).to redirect_to(instruction_expert_avis_path(avis.procedure, avis))
expect(avis.answer).to eq('answer')
expect(avis.piece_justificative_file).to_not be_attached
2021-02-28 22:20:24 +01:00
expect(dossier.reload.last_avis_updated_at).to eq(now)
expect(flash.notice).to eq('Votre réponse est enregistrée.')
end
end
2022-08-31 16:23:46 +02:00
context 'without attachment with an instructeur wants to be notified' do
before do
allow(DossierMailer).to receive(:notify_new_avis_to_instructeur).and_return(double(deliver_later: nil))
AssignTo.find_by(instructeur: instructeur_with_instant_avis_notification).update!(instant_expert_avis_email_notifications_enabled: true)
instructeur_with_instant_avis_notification.follow(avis_without_answer.dossier)
2022-11-29 10:30:35 +01:00
subject
2022-08-31 16:23:46 +02:00
end
it 'The instructeur should be notified of the new avis' do
expect(DossierMailer).to have_received(:notify_new_avis_to_instructeur).once.with(avis_without_answer, instructeur_with_instant_avis_notification.email)
end
end
2021-02-28 22:20:24 +01:00
context 'with attachment' do
let(:file) { fixture_file_upload('spec/fixtures/files/piece_justificative_0.pdf', 'application/pdf') }
before do
expect(ClamavService).to receive(:safe_file?).and_return(true)
post :update, params: { id: avis_without_answer.id, procedure_id: procedure.id, avis: { answer: 'answer', piece_justificative_file: file } }
perform_enqueued_jobs
avis_without_answer.reload
end
it 'should be ok' do
expect(response).to redirect_to(instruction_expert_avis_path(avis_without_answer.procedure, avis_without_answer))
expect(avis_without_answer.answer).to eq('answer')
expect(avis_without_answer.piece_justificative_file).to be_attached
expect(flash.notice).to eq('Votre réponse est enregistrée.')
end
end
2022-11-29 10:30:35 +01:00
2022-02-28 14:24:41 +01:00
context 'with an avis that does not belongs to current_expert' do
2022-11-29 10:30:35 +01:00
before { sign_in(create(:expert).user) }
2022-02-28 14:24:41 +01:00
it "refuse l'accès au dossier" do
2022-11-29 10:30:35 +01:00
expect(subject).to redirect_to(expert_all_avis_path)
2022-02-28 14:24:41 +01:00
expect(flash.alert).to eq("Vous navez pas accès à cet avis.")
end
end
2021-02-28 22:20:24 +01:00
end
describe '#create_commentaire' do
let(:file) { nil }
let(:scan_result) { true }
let(:now) { Time.zone.parse("14/07/1789") }
subject { post :create_commentaire, params: { id: avis_without_answer.id, procedure_id: procedure.id, commentaire: { body: 'commentaire body', piece_jointe: file } } }
before do
allow(ClamavService).to receive(:safe_file?).and_return(scan_result)
Timecop.freeze(now)
end
after { Timecop.return }
it do
subject
expect(response).to redirect_to(messagerie_expert_avis_path(avis_without_answer.procedure, avis_without_answer))
expect(dossier.commentaires.map(&:body)).to match(['commentaire body'])
expect(dossier.reload.last_commentaire_updated_at).to eq(now)
end
context "with a file" do
let(:file) { fixture_file_upload('spec/fixtures/files/piece_justificative_0.pdf', 'application/pdf') }
it do
subject
expect(Commentaire.last.piece_jointe.filename).to eq("piece_justificative_0.pdf")
end
it { expect { subject }.to change(Commentaire, :count).by(1) }
end
end
describe '#expert_cannot_invite_another_expert' do
let!(:previous_avis) { create(:avis, dossier: dossier, claimant: claimant, experts_procedure: experts_procedure, confidentiel: previous_avis_confidentiel) }
2021-02-28 22:20:24 +01:00
let(:previous_avis_confidentiel) { false }
let(:asked_confidentiel) { false }
let(:intro) { 'introduction' }
2021-03-30 17:55:35 +02:00
let(:emails) { "[\"toto@totomail.com\"]" }
2021-02-28 22:20:24 +01:00
let(:invite_linked_dossiers) { nil }
before do
Flipper.enable_actor(:expert_not_allowed_to_invite, procedure)
post :create_avis, params: { id: previous_avis.id, procedure_id: procedure.id, avis: { emails: emails, introduction: intro, confidentiel: asked_confidentiel, invite_linked_dossiers: invite_linked_dossiers, introduction_file: @introduction_file } }
end
context 'when the expert cannot invite another expert' do
let(:asked_confidentiel) { false }
it { expect(flash.alert).to eq("Cette démarche ne vous permet pas de demander un avis externe") }
it { expect(response).to redirect_to(instruction_expert_avis_path(procedure, previous_avis)) }
end
end
describe '#create_avis' do
2022-11-28 21:21:57 +01:00
let(:previous_avis_confidentiel) { false }
let(:previous_revoked_at) { nil }
let!(:previous_avis) { create(:avis, dossier:, claimant:, experts_procedure:, confidentiel: previous_avis_confidentiel, revoked_at: previous_revoked_at) }
let(:emails) { '["a@b.com"]' }
2022-11-28 21:21:57 +01:00
let(:introduction) { 'introduction' }
2021-02-28 22:20:24 +01:00
let(:created_avis) { Avis.last }
let!(:old_avis_count) { Avis.count }
let(:invite_linked_dossiers) { nil }
2022-11-28 21:21:57 +01:00
let(:introduction_file) { fixture_file_upload('spec/fixtures/files/piece_justificative_0.pdf', 'application/pdf') }
let(:confidentiel) { false }
2021-02-28 22:20:24 +01:00
before do
Timecop.freeze(now)
2022-11-28 21:21:57 +01:00
post :create_avis, params: { id: previous_avis.id, procedure_id: procedure.id, avis: { emails:, introduction:, experts_procedure:, confidentiel:, invite_linked_dossiers:, introduction_file: } }
2021-02-28 22:20:24 +01:00
created_avis.reload
end
after { Timecop.return }
context 'from a revoked avis' do
let(:previous_revoked_at) { Time.zone.now }
it do
expect(response).to redirect_to(root_path)
expect(Avis.last).to eq(previous_avis)
end
end
2021-02-28 22:20:24 +01:00
context 'when an invalid email' do
2021-03-30 17:55:35 +02:00
let(:emails) { "[\"toto.fr\"]" }
2021-02-28 22:20:24 +01:00
2022-11-28 14:24:56 +01:00
it do
expect(response).to render_template :instruction
expect(flash.alert).to eq(["toto.fr : Email n'est pas valide"])
expect(Avis.last).to eq(previous_avis)
expect(dossier.last_avis_updated_at).to eq(nil)
end
2021-02-28 22:20:24 +01:00
end
context 'ask review with attachment' do
2021-03-30 17:55:35 +02:00
let(:emails) { "[\"toto@totomail.com\"]" }
2021-02-28 22:20:24 +01:00
2022-11-28 14:24:56 +01:00
it do
expect(created_avis.introduction_file).to be_attached
expect(created_avis.introduction_file.filename).to eq("piece_justificative_0.pdf")
expect(created_avis.dossier.reload.last_avis_updated_at).to eq(now)
expect(flash.notice).to eq("Une demande davis a été envoyée à toto@totomail.com")
end
2021-02-28 22:20:24 +01:00
end
context 'with multiple emails' do
2021-03-30 17:55:35 +02:00
let(:emails) { "[\"toto.fr\",\"titi@titimail.com\"]" }
2021-02-28 22:20:24 +01:00
2022-11-28 14:24:56 +01:00
it do
expect(response).to render_template :instruction
expect(flash.alert).to eq(["toto.fr : Email n'est pas valide"])
expect(flash.notice).to eq("Une demande davis a été envoyée à titi@titimail.com")
expect(Avis.count).to eq(old_avis_count + 1)
end
2021-02-28 22:20:24 +01:00
end
context 'when the previous avis is public' do
context 'when the user asked for a public avis' do
2022-11-28 14:24:56 +01:00
it do
expect(created_avis.confidentiel).to be(false)
2022-11-28 21:21:57 +01:00
expect(created_avis.introduction).to eq(introduction)
2022-11-28 14:24:56 +01:00
expect(created_avis.dossier).to eq(previous_avis.dossier)
expect(created_avis.claimant).to eq(expert)
expect(response).to redirect_to(instruction_expert_avis_path(previous_avis.procedure, previous_avis))
end
2021-02-28 22:20:24 +01:00
end
context 'when the user asked for a confidentiel avis' do
2022-11-28 21:21:57 +01:00
let(:confidentiel) { true }
2021-02-28 22:20:24 +01:00
it { expect(created_avis.confidentiel).to be(true) }
end
end
context 'when the preivous avis is confidentiel' do
let(:previous_avis_confidentiel) { true }
context 'when the user asked for a public avis' do
2022-11-28 21:21:57 +01:00
let(:confidentiel) { false }
2021-02-28 22:20:24 +01:00
it { expect(created_avis.confidentiel).to be(true) }
end
end
context 'with linked dossiers' do
let(:dossier) { create(:dossier, :en_construction, :with_dossier_link, procedure: procedure) }
context 'when the expert doesnt share linked dossiers' do
let(:invite_linked_dossiers) { false }
it 'sends a single avis for the main dossier, but doesnt give access to the linked dossiers' do
2021-05-26 15:16:30 +02:00
expect(flash.notice).to eq("Une demande davis a été envoyée à a@b.com")
2021-02-28 22:20:24 +01:00
expect(Avis.count).to eq(old_avis_count + 1)
expect(created_avis.dossier).to eq(dossier)
end
end
context 'when the expert also shares the linked dossiers' do
context 'and the expert can access the linked dossiers' do
let(:created_avis) { create(:avis, dossier: dossier, claimant: claimant, email: "toto3@gmail.com") }
let(:linked_dossier) { Dossier.find_by(id: dossier.reload.champs_public.filter(&:dossier_link?).filter_map(&:value)) }
let(:linked_avis) { create(:avis, dossier: linked_dossier, claimant: claimant) }
2021-02-28 22:20:24 +01:00
let(:invite_linked_dossiers) { true }
it 'sends one avis for the main dossier' do
2021-05-26 15:16:30 +02:00
expect(flash.notice).to eq("Une demande davis a été envoyée à a@b.com")
2021-02-28 22:20:24 +01:00
expect(created_avis.dossier).to eq(dossier)
end
it 'sends another avis for the linked dossiers' do
expect(Avis.count).to eq(old_avis_count + 2)
expect(linked_avis.dossier).to eq(linked_dossier)
end
end
context 'but the expert cant access the linked dossier' do
it 'sends a single avis for the main dossier, but doesnt give access to the linked dossiers' do
2021-05-26 15:16:30 +02:00
expect(flash.notice).to eq("Une demande davis a été envoyée à a@b.com")
2021-02-28 22:20:24 +01:00
expect(Avis.count).to eq(old_avis_count + 1)
expect(created_avis.dossier).to eq(dossier)
end
end
end
end
end
end
context 'without an expert signed in' do
let(:claimant) { create(:instructeur) }
let(:expert) { create(:expert) }
let(:experts_procedure) { create(:experts_procedure, expert: expert, procedure: procedure) }
let(:dossier) { create(:dossier) }
let(:avis) { create(:avis, dossier: dossier, experts_procedure: experts_procedure, claimant: claimant) }
let(:procedure) { dossier.procedure }
2021-02-28 22:20:24 +01:00
describe '#sign_up' do
subject do
get :sign_up, params: { id: avis.id, procedure_id: procedure.id, email: avis.expert.email }
end
context 'when the avis is revoked' do
before { avis.update(revoked_at: Time.zone.now) }
it { is_expected.to redirect_to(root_path) }
end
context 'when the expert hasnt signed up yet' do
before { expert.user.update(last_sign_in_at: nil) }
it { is_expected.to have_http_status(:success) }
end
context 'when the expert has already signed up' do
before { expert.user.update(last_sign_in_at: Time.zone.now) }
context 'and the expert belongs to the invitation' do
context 'and the expert is authenticated' do
before { sign_in(expert.user) }
it { is_expected.to redirect_to expert_avis_url(avis.procedure, avis) }
2021-02-28 22:20:24 +01:00
end
context 'and the expert is not authenticated' do
before { sign_out(expert.user) }
2021-02-28 22:20:24 +01:00
it { is_expected.to redirect_to new_user_session_url }
2021-02-28 22:20:24 +01:00
end
end
2021-02-28 22:20:24 +01:00
context 'and the expert does not belong to the invitation' do
let(:avis) { create(:avis, email: 'another_expert@avis.com', dossier: dossier, experts_procedure: experts_procedure) }
before { sign_in(expert.user) }
# redirected to dossier but then the instructeur gonna be banished !
it { is_expected.to redirect_to expert_avis_url(avis.procedure, avis) }
2021-02-28 22:20:24 +01:00
end
end
end
2021-02-28 22:20:24 +01:00
describe '#update_expert' do
subject do
post :update_expert, params: {
id: avis.id,
procedure_id: procedure.id,
email: avis.expert.email,
user: {
password: 'my-s3cure-p4ssword'
}
}
end
2021-02-28 22:20:24 +01:00
context 'when the avis is revoked' do
before { avis.update(revoked_at: Time.zone.now) }
it { is_expected.to redirect_to(root_path) }
end
context 'when the expert hasnt signed up yet' do
before { expert.user.update(last_sign_in_at: nil) }
2021-02-28 22:20:24 +01:00
it 'saves the expert new password' do
subject
expect(expert.user.reload.valid_password?('my-s3cure-p4ssword')).to be true
end
it { is_expected.to redirect_to expert_all_avis_path }
2021-02-28 22:20:24 +01:00
end
context 'when the expert has already signed up' do
before { expert.user.update(last_sign_in_at: Time.zone.now) }
it 'doesnt change the expert password' do
subject
expect(expert.user.reload.valid_password?('my-s3cure-p4ssword')).to be false
end
it { is_expected.to redirect_to new_user_session_url }
end
2021-02-28 22:20:24 +01:00
end
end
end