8e6930d257
The `joins` are declared explicitely in order to associate a predictable name to the joined table. Otherwise, when the query is joined with `:users`, ActiveRecord will alias the join automatically to solve the conflict. Unfortunately, the automatic resolution means that the table name becomes unpredictable, and thus unsuitable to perform queries on.
866 lines
37 KiB
Ruby
866 lines
37 KiB
Ruby
require 'spec_helper'
|
||
|
||
describe ProcedurePresentation do
|
||
let(:procedure) { create(:procedure, :with_type_de_champ, :with_type_de_champ_private) }
|
||
let(:assign_to) { create(:assign_to, procedure: procedure) }
|
||
let(:first_type_de_champ_id) { assign_to.procedure.types_de_champ.first.id.to_s }
|
||
let (:procedure_presentation_id) {
|
||
ProcedurePresentation.create(
|
||
assign_to: assign_to,
|
||
displayed_fields: [
|
||
{ "label" => "test1", "table" => "user", "column" => "email" },
|
||
{ "label" => "test2", "table" => "type_de_champ", "column" => first_type_de_champ_id }
|
||
],
|
||
sort: { "table" => "user", "column" => "email", "order" => "asc" },
|
||
filters: { "a-suivre" => [], "suivis" => [{ "label" => "label1", "table" => "self", "column" => "created_at" }] }
|
||
).id
|
||
}
|
||
let (:procedure_presentation) { ProcedurePresentation.find(procedure_presentation_id) }
|
||
|
||
describe "#displayed_fields" do
|
||
it { expect(procedure_presentation.displayed_fields).to eq([{ "label" => "test1", "table" => "user", "column" => "email" }, { "label" => "test2", "table" => "type_de_champ", "column" => first_type_de_champ_id }]) }
|
||
end
|
||
|
||
describe "#sort" do
|
||
it { expect(procedure_presentation.sort).to eq({ "table" => "user", "column" => "email", "order" => "asc" }) }
|
||
end
|
||
|
||
describe "#filters" do
|
||
it { expect(procedure_presentation.filters).to eq({ "a-suivre" => [], "suivis" => [{ "label" => "label1", "table" => "self", "column" => "created_at" }] }) }
|
||
end
|
||
|
||
describe 'validation' do
|
||
it { expect(build(:procedure_presentation)).to be_valid }
|
||
|
||
context 'of displayed fields' do
|
||
it { expect(build(:procedure_presentation, displayed_fields: [{ "table" => "user", "column" => "reset_password_token", "order" => "asc" }])).to be_invalid }
|
||
end
|
||
|
||
context 'of sort' do
|
||
it { expect(build(:procedure_presentation, sort: { "table" => "notifications", "column" => "notifications", "order" => "asc" })).to be_valid }
|
||
it { expect(build(:procedure_presentation, sort: { "table" => "self", "column" => "id", "order" => "asc" })).to be_valid }
|
||
it { expect(build(:procedure_presentation, sort: { "table" => "self", "column" => "state", "order" => "asc" })).to be_valid }
|
||
it { expect(build(:procedure_presentation, sort: { "table" => "user", "column" => "reset_password_token", "order" => "asc" })).to be_invalid }
|
||
end
|
||
|
||
context 'of filters' do
|
||
it { expect(build(:procedure_presentation, filters: { "suivis" => [{ "table" => "user", "column" => "reset_password_token", "order" => "asc" }] })).to be_invalid }
|
||
end
|
||
end
|
||
|
||
describe "#fields" do
|
||
context 'when the procedure can have a SIRET number' do
|
||
let(:procedure) { create(:procedure, :with_type_de_champ, :with_type_de_champ_private, :types_de_champ_count => 4, :types_de_champ_private_count => 4) }
|
||
let(:tdc_1) { procedure.types_de_champ[0] }
|
||
let(:tdc_2) { procedure.types_de_champ[1] }
|
||
let(:tdc_private_1) { procedure.types_de_champ_private[0] }
|
||
let(:tdc_private_2) { procedure.types_de_champ_private[1] }
|
||
let(:expected) {
|
||
[
|
||
{ "label" => 'Créé le', "table" => 'self', "column" => 'created_at' },
|
||
{ "label" => 'En construction le', "table" => 'self', "column" => 'en_construction_at' },
|
||
{ "label" => 'Mis à jour le', "table" => 'self', "column" => 'updated_at' },
|
||
{ "label" => 'Demandeur', "table" => 'user', "column" => 'email' },
|
||
{ "label" => 'Email instructeur', "table" => 'followers_instructeurs', "column" => 'email' },
|
||
{ "label" => 'Groupe instructeur', "table" => 'groupe_instructeur', "column" => 'label' },
|
||
{ "label" => 'SIREN', "table" => 'etablissement', "column" => 'entreprise_siren' },
|
||
{ "label" => 'Forme juridique', "table" => 'etablissement', "column" => 'entreprise_forme_juridique' },
|
||
{ "label" => 'Nom commercial', "table" => 'etablissement', "column" => 'entreprise_nom_commercial' },
|
||
{ "label" => 'Raison sociale', "table" => 'etablissement', "column" => 'entreprise_raison_sociale' },
|
||
{ "label" => 'SIRET siège social', "table" => 'etablissement', "column" => 'entreprise_siret_siege_social' },
|
||
{ "label" => 'Date de création', "table" => 'etablissement', "column" => 'entreprise_date_creation' },
|
||
{ "label" => 'SIRET', "table" => 'etablissement', "column" => 'siret' },
|
||
{ "label" => 'Libellé NAF', "table" => 'etablissement', "column" => 'libelle_naf' },
|
||
{ "label" => 'Code postal', "table" => 'etablissement', "column" => 'code_postal' },
|
||
{ "label" => tdc_1.libelle, "table" => 'type_de_champ', "column" => tdc_1.id.to_s },
|
||
{ "label" => tdc_2.libelle, "table" => 'type_de_champ', "column" => tdc_2.id.to_s },
|
||
{ "label" => tdc_private_1.libelle, "table" => 'type_de_champ_private', "column" => tdc_private_1.id.to_s },
|
||
{ "label" => tdc_private_2.libelle, "table" => 'type_de_champ_private', "column" => tdc_private_2.id.to_s }
|
||
]
|
||
}
|
||
|
||
before do
|
||
procedure.types_de_champ[2].update_attribute(:type_champ, TypeDeChamp.type_champs.fetch(:header_section))
|
||
procedure.types_de_champ[3].update_attribute(:type_champ, TypeDeChamp.type_champs.fetch(:explication))
|
||
procedure.types_de_champ_private[2].update_attribute(:type_champ, TypeDeChamp.type_champs.fetch(:header_section))
|
||
procedure.types_de_champ_private[3].update_attribute(:type_champ, TypeDeChamp.type_champs.fetch(:explication))
|
||
end
|
||
|
||
subject { create(:procedure_presentation, assign_to: create(:assign_to, procedure: procedure)) }
|
||
|
||
it { expect(subject.fields).to eq(expected) }
|
||
end
|
||
|
||
context 'when the procedure is for individuals' do
|
||
let(:name_field) { { "label" => "Prénom", "table" => "individual", "column" => "prenom" } }
|
||
let(:surname_field) { { "label" => "Nom", "table" => "individual", "column" => "nom" } }
|
||
let(:gender_field) { { "label" => "Civilité", "table" => "individual", "column" => "gender" } }
|
||
let(:procedure) { create(:procedure, :for_individual) }
|
||
let(:procedure_presentation) { create(:procedure_presentation, assign_to: create(:assign_to, procedure: procedure)) }
|
||
|
||
subject { procedure_presentation.fields }
|
||
|
||
it { is_expected.to include(name_field, surname_field, gender_field) }
|
||
end
|
||
end
|
||
|
||
describe "#fields_for_select" do
|
||
subject { create(:procedure_presentation) }
|
||
|
||
before do
|
||
allow(subject).to receive(:fields).and_return([
|
||
{
|
||
"label" => "label1",
|
||
"table" => "table1",
|
||
"column" => "column1"
|
||
},
|
||
{
|
||
"label" => "label2",
|
||
"table" => "table2",
|
||
"column" => "column2"
|
||
}
|
||
])
|
||
end
|
||
|
||
it { expect(subject.fields_for_select).to eq([["label1", "table1/column1"], ["label2", "table2/column2"]]) }
|
||
end
|
||
|
||
describe '#get_value' do
|
||
let(:procedure_presentation) { ProcedurePresentation.create(assign_to: assign_to, displayed_fields: [{ 'table' => table, 'column' => column }]) }
|
||
|
||
subject { procedure_presentation.displayed_field_values(dossier).first }
|
||
|
||
context 'for self table' do
|
||
let(:table) { 'self' }
|
||
|
||
context 'for created_at column' do
|
||
let(:column) { 'created_at' }
|
||
let(:dossier) { Timecop.freeze(Time.zone.local(1992, 3, 22)) { create(:dossier, procedure: procedure) } }
|
||
|
||
it { is_expected.to eq('22/03/1992') }
|
||
end
|
||
|
||
context 'for en_construction_at column' do
|
||
let(:column) { 'en_construction_at' }
|
||
let(:dossier) { create(:dossier, :en_construction, procedure: procedure, en_construction_at: Time.zone.local(2018, 10, 17)) }
|
||
|
||
it { is_expected.to eq('17/10/2018') }
|
||
end
|
||
|
||
context 'for updated_at column' do
|
||
let(:column) { 'updated_at' }
|
||
let(:dossier) { create(:dossier, procedure: procedure) }
|
||
|
||
before { dossier.touch(time: Time.zone.local(2018, 9, 25)) }
|
||
|
||
it { is_expected.to eq('25/09/2018') }
|
||
end
|
||
end
|
||
|
||
context 'for user table' do
|
||
let(:table) { 'user' }
|
||
let(:column) { 'email' }
|
||
|
||
let(:dossier) { create(:dossier, procedure: procedure, user: create(:user, email: 'bla@yopmail.com')) }
|
||
|
||
it { is_expected.to eq('bla@yopmail.com') }
|
||
end
|
||
|
||
context 'for individual table' do
|
||
let(:table) { 'individual' }
|
||
let(:dossier) { create(:dossier, procedure: procedure, individual: create(:individual, nom: 'Martin', prenom: 'Jacques', gender: 'M.')) }
|
||
|
||
context 'for prenom column' do
|
||
let(:column) { 'prenom' }
|
||
|
||
it { is_expected.to eq('Jacques') }
|
||
end
|
||
|
||
context 'for nom column' do
|
||
let(:column) { 'nom' }
|
||
|
||
it { is_expected.to eq('Martin') }
|
||
end
|
||
|
||
context 'for gender column' do
|
||
let(:column) { 'gender' }
|
||
|
||
it { is_expected.to eq('M.') }
|
||
end
|
||
end
|
||
|
||
context 'for etablissement table' do
|
||
let(:table) { 'etablissement' }
|
||
let(:column) { 'code_postal' } # All other columns work the same, no extra test required
|
||
|
||
let!(:dossier) { create(:dossier, procedure: procedure, etablissement: create(:etablissement, code_postal: '75008')) }
|
||
|
||
it { is_expected.to eq('75008') }
|
||
end
|
||
|
||
context 'for groupe_instructeur table' do
|
||
let(:table) { 'groupe_instructeur' }
|
||
let(:column) { 'label' }
|
||
|
||
let!(:dossier) { create(:dossier, procedure: procedure) }
|
||
|
||
it { is_expected.to eq('défaut') }
|
||
end
|
||
|
||
context 'for followers_instructeurs table' do
|
||
let(:table) { 'followers_instructeurs' }
|
||
let(:column) { 'email' }
|
||
|
||
let(:dossier) { create(:dossier, procedure: procedure) }
|
||
let!(:follow1) { create(:follow, dossier: dossier, instructeur: create(:instructeur, email: 'user1@host')) }
|
||
let!(:follow2) { create(:follow, dossier: dossier, instructeur: create(:instructeur, email: 'user2@host')) }
|
||
|
||
it { is_expected.to eq('user1@host, user2@host') }
|
||
end
|
||
|
||
context 'for type_de_champ table' do
|
||
let(:table) { 'type_de_champ' }
|
||
let(:column) { procedure.types_de_champ.first.id.to_s }
|
||
|
||
let(:dossier) { create(:dossier, procedure: procedure) }
|
||
|
||
before { dossier.champs.first.update(value: 'kale') }
|
||
|
||
it { is_expected.to eq('kale') }
|
||
end
|
||
|
||
context 'for type_de_champ_private table' do
|
||
let(:table) { 'type_de_champ_private' }
|
||
let(:column) { procedure.types_de_champ_private.first.id.to_s }
|
||
|
||
let(:dossier) { create(:dossier, procedure: procedure) }
|
||
|
||
before { dossier.champs_private.first.update(value: 'quinoa') }
|
||
|
||
it { is_expected.to eq('quinoa') }
|
||
end
|
||
end
|
||
|
||
describe '#sorted_ids' do
|
||
let(:instructeur) { create(:instructeur) }
|
||
let(:assign_to) { create(:assign_to, procedure: procedure, instructeur: instructeur) }
|
||
let(:sort) { { 'table' => table, 'column' => column, 'order' => order } }
|
||
let(:procedure_presentation) { ProcedurePresentation.create(assign_to: assign_to, sort: sort) }
|
||
|
||
subject { procedure_presentation.sorted_ids(procedure.dossiers, instructeur) }
|
||
|
||
context 'for notifications table' do
|
||
let(:table) { 'notifications' }
|
||
let(:column) { 'notifications' }
|
||
|
||
let!(:notified_dossier) { create(:dossier, :en_construction, procedure: procedure) }
|
||
let!(:recent_dossier) { create(:dossier, :en_construction, procedure: procedure) }
|
||
let!(:older_dossier) { create(:dossier, :en_construction, procedure: procedure) }
|
||
|
||
before do
|
||
notified_dossier.champs.first.touch(time: Time.zone.local(2018, 9, 20))
|
||
create(:follow, instructeur: instructeur, dossier: notified_dossier, demande_seen_at: Time.zone.local(2018, 9, 10))
|
||
notified_dossier.touch(time: Time.zone.local(2018, 9, 20))
|
||
recent_dossier.touch(time: Time.zone.local(2018, 9, 25))
|
||
older_dossier.touch(time: Time.zone.local(2018, 5, 13))
|
||
end
|
||
|
||
context 'in ascending order' do
|
||
let(:order) { 'asc' }
|
||
|
||
it { is_expected.to eq([older_dossier, recent_dossier, notified_dossier].map(&:id)) }
|
||
end
|
||
|
||
context 'in descending order' do
|
||
let(:order) { 'desc' }
|
||
|
||
it { is_expected.to eq([notified_dossier, recent_dossier, older_dossier].map(&:id)) }
|
||
end
|
||
|
||
context 'with a dossier terminé' do
|
||
let!(:notified_dossier) { create(:dossier, :accepte, procedure: procedure) }
|
||
let(:order) { 'desc' }
|
||
|
||
it { is_expected.to eq([notified_dossier, recent_dossier, older_dossier].map(&:id)) }
|
||
end
|
||
end
|
||
|
||
context 'for self table' do
|
||
let(:table) { 'self' }
|
||
let(:order) { 'asc' } # Desc works the same, no extra test required
|
||
|
||
context 'for created_at column' do
|
||
let(:column) { 'created_at' }
|
||
let!(:recent_dossier) { Timecop.freeze(Time.zone.local(2018, 10, 17)) { create(:dossier, procedure: procedure) } }
|
||
let!(:older_dossier) { Timecop.freeze(Time.zone.local(2003, 11, 11)) { create(:dossier, procedure: procedure) } }
|
||
|
||
it { is_expected.to eq([older_dossier, recent_dossier].map(&:id)) }
|
||
end
|
||
|
||
context 'for en_construction_at column' do
|
||
let(:column) { 'en_construction_at' }
|
||
let!(:recent_dossier) { create(:dossier, :en_construction, procedure: procedure, en_construction_at: Time.zone.local(2018, 10, 17)) }
|
||
let!(:older_dossier) { create(:dossier, :en_construction, procedure: procedure, en_construction_at: Time.zone.local(2013, 1, 1)) }
|
||
|
||
it { is_expected.to eq([older_dossier, recent_dossier].map(&:id)) }
|
||
end
|
||
|
||
context 'for updated_at column' do
|
||
let(:column) { 'updated_at' }
|
||
let(:recent_dossier) { create(:dossier, procedure: procedure) }
|
||
let(:older_dossier) { create(:dossier, procedure: procedure) }
|
||
|
||
before do
|
||
recent_dossier.touch(time: Time.zone.local(2018, 9, 25))
|
||
older_dossier.touch(time: Time.zone.local(2018, 5, 13))
|
||
end
|
||
|
||
it { is_expected.to eq([older_dossier, recent_dossier].map(&:id)) }
|
||
end
|
||
end
|
||
|
||
context 'for type_de_champ table' do
|
||
let(:table) { 'type_de_champ' }
|
||
let(:column) { procedure.types_de_champ.first.id.to_s }
|
||
let(:order) { 'desc' } # Asc works the same, no extra test required
|
||
|
||
let(:beurre_dossier) { create(:dossier, procedure: procedure) }
|
||
let(:tartine_dossier) { create(:dossier, procedure: procedure) }
|
||
|
||
before do
|
||
beurre_dossier.champs.first.update(value: 'beurre')
|
||
tartine_dossier.champs.first.update(value: 'tartine')
|
||
end
|
||
|
||
it { is_expected.to eq([tartine_dossier, beurre_dossier].map(&:id)) }
|
||
end
|
||
|
||
context 'for type_de_champ_private table' do
|
||
let(:table) { 'type_de_champ_private' }
|
||
let(:column) { procedure.types_de_champ_private.first.id.to_s }
|
||
let(:order) { 'asc' } # Desc works the same, no extra test required
|
||
|
||
let(:biere_dossier) { create(:dossier, procedure: procedure) }
|
||
let(:vin_dossier) { create(:dossier, procedure: procedure) }
|
||
|
||
before do
|
||
biere_dossier.champs_private.first.update(value: 'biere')
|
||
vin_dossier.champs_private.first.update(value: 'vin')
|
||
end
|
||
|
||
it { is_expected.to eq([biere_dossier, vin_dossier].map(&:id)) }
|
||
end
|
||
|
||
context 'for individual table' do
|
||
let(:table) { 'individual' }
|
||
let(:order) { 'asc' } # Desc works the same, no extra test required
|
||
|
||
let(:procedure) { create(:procedure, :for_individual) }
|
||
|
||
let!(:first_dossier) { create(:dossier, procedure: procedure, individual: create(:individual, gender: 'M', prenom: 'Alain', nom: 'Antonelli')) }
|
||
let!(:last_dossier) { create(:dossier, procedure: procedure, individual: create(:individual, gender: 'Mme', prenom: 'Zora', nom: 'Zemmour')) }
|
||
|
||
context 'for gender column' do
|
||
let(:column) { 'gender' }
|
||
|
||
it { is_expected.to eq([first_dossier, last_dossier].map(&:id)) }
|
||
end
|
||
|
||
context 'for prenom column' do
|
||
let(:column) { 'prenom' }
|
||
|
||
it { is_expected.to eq([first_dossier, last_dossier].map(&:id)) }
|
||
end
|
||
|
||
context 'for nom column' do
|
||
let(:column) { 'nom' }
|
||
|
||
it { is_expected.to eq([first_dossier, last_dossier].map(&:id)) }
|
||
end
|
||
end
|
||
|
||
context 'for followers_instructeurs table' do
|
||
let(:table) { 'followers_instructeurs' }
|
||
let(:order) { 'asc' } # Desc works the same, no extra test required
|
||
|
||
let!(:dossier_a) { create(:dossier, :en_construction, procedure: procedure) }
|
||
let!(:dossier_z) { create(:dossier, :en_construction, procedure: procedure) }
|
||
let!(:dossier_without_instructeur) { create(:dossier, :en_construction, procedure: procedure) }
|
||
|
||
before do
|
||
create(:follow, dossier: dossier_a, instructeur: create(:instructeur, email: 'abaca@exemple.fr'))
|
||
create(:follow, dossier: dossier_z, instructeur: create(:instructeur, email: 'zythum@exemple.fr'))
|
||
end
|
||
|
||
context 'for email column' do
|
||
let(:column) { 'email' }
|
||
|
||
it { is_expected.to eq([dossier_a, dossier_z, dossier_without_instructeur].map(&:id)) }
|
||
end
|
||
end
|
||
|
||
context 'for other tables' do
|
||
# All other columns and tables work the same so it’s ok to test only one
|
||
let(:table) { 'etablissement' }
|
||
let(:column) { 'code_postal' }
|
||
let(:order) { 'asc' } # Desc works the same, no extra test required
|
||
|
||
let!(:huitieme_dossier) { create(:dossier, procedure: procedure, etablissement: create(:etablissement, code_postal: '75008')) }
|
||
let!(:vingtieme_dossier) { create(:dossier, procedure: procedure, etablissement: create(:etablissement, code_postal: '75020')) }
|
||
|
||
it { is_expected.to eq([huitieme_dossier, vingtieme_dossier].map(&:id)) }
|
||
end
|
||
end
|
||
|
||
describe '#filtered_ids' do
|
||
let(:procedure_presentation) { create(:procedure_presentation, assign_to: create(:assign_to, procedure: procedure), filters: { "suivis" => filter }) }
|
||
|
||
subject { procedure_presentation.filtered_ids(procedure.dossiers.joins(:user), 'suivis') }
|
||
|
||
context 'for self table' do
|
||
context 'for created_at column' do
|
||
let(:filter) { [{ 'table' => 'self', 'column' => 'created_at', 'value' => '18/9/2018' }] }
|
||
|
||
let!(:kept_dossier) { create(:dossier, procedure: procedure, created_at: Time.zone.local(2018, 9, 18, 14, 28)) }
|
||
let!(:discarded_dossier) { create(:dossier, procedure: procedure, created_at: Time.zone.local(2018, 9, 17, 23, 59)) }
|
||
|
||
it { is_expected.to contain_exactly(kept_dossier.id) }
|
||
end
|
||
|
||
context 'for en_construction_at column' do
|
||
let(:filter) { [{ 'table' => 'self', 'column' => 'en_construction_at', 'value' => '17/10/2018' }] }
|
||
|
||
let!(:kept_dossier) { create(:dossier, :en_construction, procedure: procedure, en_construction_at: Time.zone.local(2018, 10, 17)) }
|
||
let!(:discarded_dossier) { create(:dossier, :en_construction, procedure: procedure, en_construction_at: Time.zone.local(2013, 1, 1)) }
|
||
|
||
it { is_expected.to contain_exactly(kept_dossier.id) }
|
||
end
|
||
|
||
context 'for updated_at column' do
|
||
let(:filter) { [{ 'table' => 'self', 'column' => 'updated_at', 'value' => '18/9/2018' }] }
|
||
|
||
let(:kept_dossier) { create(:dossier, procedure: procedure) }
|
||
let(:discarded_dossier) { create(:dossier, procedure: procedure) }
|
||
|
||
before do
|
||
kept_dossier.touch(time: Time.zone.local(2018, 9, 18, 14, 28))
|
||
discarded_dossier.touch(time: Time.zone.local(2018, 9, 17, 23, 59))
|
||
end
|
||
|
||
it { is_expected.to contain_exactly(kept_dossier.id) }
|
||
end
|
||
|
||
context 'ignore time of day' do
|
||
let(:filter) { [{ 'table' => 'self', 'column' => 'en_construction_at', 'value' => '17/10/2018 19:30' }] }
|
||
|
||
let!(:kept_dossier) { create(:dossier, :en_construction, procedure: procedure, en_construction_at: Time.zone.local(2018, 10, 17, 15, 56)) }
|
||
let!(:discarded_dossier) { create(:dossier, :en_construction, procedure: procedure, en_construction_at: Time.zone.local(2018, 10, 18, 5, 42)) }
|
||
|
||
it { is_expected.to contain_exactly(kept_dossier.id) }
|
||
end
|
||
|
||
context 'for a malformed date' do
|
||
context 'when its a string' do
|
||
let(:filter) { [{ 'table' => 'self', 'column' => 'updated_at', 'value' => 'malformed date' }] }
|
||
|
||
it { is_expected.to match([]) }
|
||
end
|
||
|
||
context 'when its a number' do
|
||
let(:filter) { [{ 'table' => 'self', 'column' => 'updated_at', 'value' => '177500' }] }
|
||
|
||
it { is_expected.to match([]) }
|
||
end
|
||
end
|
||
|
||
context 'with multiple search values' do
|
||
let(:filter) do
|
||
[
|
||
{ 'table' => 'self', 'column' => 'en_construction_at', 'value' => '17/10/2018' },
|
||
{ 'table' => 'self', 'column' => 'en_construction_at', 'value' => '19/10/2018' }
|
||
]
|
||
end
|
||
|
||
let!(:kept_dossier) { create(:dossier, :en_construction, procedure: procedure, en_construction_at: Time.zone.local(2018, 10, 17)) }
|
||
let!(:other_kept_dossier) { create(:dossier, :en_construction, procedure: procedure, en_construction_at: Time.zone.local(2018, 10, 19)) }
|
||
let!(:discarded_dossier) { create(:dossier, :en_construction, procedure: procedure, en_construction_at: Time.zone.local(2013, 1, 1)) }
|
||
|
||
it 'returns every dossier that matches any of the search criteria for a given column' do
|
||
is_expected.to contain_exactly(kept_dossier.id, other_kept_dossier.id)
|
||
end
|
||
end
|
||
end
|
||
|
||
context 'for type_de_champ table' do
|
||
let(:filter) { [{ 'table' => 'type_de_champ', 'column' => type_de_champ.id.to_s, 'value' => 'keep' }] }
|
||
|
||
let(:kept_dossier) { create(:dossier, procedure: procedure) }
|
||
let(:discarded_dossier) { create(:dossier, procedure: procedure) }
|
||
let(:type_de_champ) { procedure.types_de_champ.first }
|
||
|
||
before do
|
||
type_de_champ.champ.create(dossier: kept_dossier, value: 'keep me')
|
||
type_de_champ.champ.create(dossier: discarded_dossier, value: 'discard me')
|
||
end
|
||
|
||
it { is_expected.to contain_exactly(kept_dossier.id) }
|
||
|
||
context 'with multiple search values' do
|
||
let(:filter) do
|
||
[
|
||
{ 'table' => 'type_de_champ', 'column' => type_de_champ.id.to_s, 'value' => 'keep' },
|
||
{ 'table' => 'type_de_champ', 'column' => type_de_champ.id.to_s, 'value' => 'and' }
|
||
]
|
||
end
|
||
|
||
let(:other_kept_dossier) { create(:dossier, procedure: procedure) }
|
||
|
||
before do
|
||
type_de_champ.champ.create(dossier: other_kept_dossier, value: 'and me too')
|
||
end
|
||
|
||
it 'returns every dossier that matches any of the search criteria for a given column' do
|
||
is_expected.to contain_exactly(kept_dossier.id, other_kept_dossier.id)
|
||
end
|
||
end
|
||
end
|
||
|
||
context 'for type_de_champ_private table' do
|
||
let(:filter) { [{ 'table' => 'type_de_champ_private', 'column' => type_de_champ_private.id.to_s, 'value' => 'keep' }] }
|
||
|
||
let(:kept_dossier) { create(:dossier, procedure: procedure) }
|
||
let(:discarded_dossier) { create(:dossier, procedure: procedure) }
|
||
let(:type_de_champ_private) { procedure.types_de_champ_private.first }
|
||
|
||
before do
|
||
type_de_champ_private.champ.create(dossier: kept_dossier, value: 'keep me')
|
||
type_de_champ_private.champ.create(dossier: discarded_dossier, value: 'discard me')
|
||
end
|
||
|
||
it { is_expected.to contain_exactly(kept_dossier.id) }
|
||
|
||
context 'with multiple search values' do
|
||
let(:filter) do
|
||
[
|
||
{ 'table' => 'type_de_champ_private', 'column' => type_de_champ_private.id.to_s, 'value' => 'keep' },
|
||
{ 'table' => 'type_de_champ_private', 'column' => type_de_champ_private.id.to_s, 'value' => 'and' }
|
||
]
|
||
end
|
||
|
||
let(:other_kept_dossier) { create(:dossier, procedure: procedure) }
|
||
|
||
before do
|
||
type_de_champ_private.champ.create(dossier: other_kept_dossier, value: 'and me too')
|
||
end
|
||
|
||
it 'returns every dossier that matches any of the search criteria for a given column' do
|
||
is_expected.to contain_exactly(kept_dossier.id, other_kept_dossier.id)
|
||
end
|
||
end
|
||
end
|
||
|
||
context 'for etablissement table' do
|
||
context 'for entreprise_date_creation column' do
|
||
let(:filter) { [{ 'table' => 'etablissement', 'column' => 'entreprise_date_creation', 'value' => '21/6/2018' }] }
|
||
|
||
let!(:kept_dossier) { create(:dossier, procedure: procedure, etablissement: create(:etablissement, entreprise_date_creation: Time.zone.local(2018, 6, 21))) }
|
||
let!(:discarded_dossier) { create(:dossier, procedure: procedure, etablissement: create(:etablissement, entreprise_date_creation: Time.zone.local(2008, 6, 21))) }
|
||
|
||
it { is_expected.to contain_exactly(kept_dossier.id) }
|
||
|
||
context 'with multiple search values' do
|
||
let(:filter) do
|
||
[
|
||
{ 'table' => 'etablissement', 'column' => 'entreprise_date_creation', 'value' => '21/6/2016' },
|
||
{ 'table' => 'etablissement', 'column' => 'entreprise_date_creation', 'value' => '21/6/2018' }
|
||
]
|
||
end
|
||
|
||
let!(:other_kept_dossier) { create(:dossier, procedure: procedure, etablissement: create(:etablissement, entreprise_date_creation: Time.zone.local(2016, 6, 21))) }
|
||
|
||
it 'returns every dossier that matches any of the search criteria for a given column' do
|
||
is_expected.to contain_exactly(kept_dossier.id, other_kept_dossier.id)
|
||
end
|
||
end
|
||
end
|
||
|
||
context 'for code_postal column' do
|
||
# All columns except entreprise_date_creation work exacly the same, just testing one
|
||
|
||
let(:filter) { [{ 'table' => 'etablissement', 'column' => 'code_postal', 'value' => '75017' }] }
|
||
|
||
let!(:kept_dossier) { create(:dossier, procedure: procedure, etablissement: create(:etablissement, code_postal: '75017')) }
|
||
let!(:discarded_dossier) { create(:dossier, procedure: procedure, etablissement: create(:etablissement, code_postal: '25000')) }
|
||
|
||
it { is_expected.to contain_exactly(kept_dossier.id) }
|
||
|
||
context 'with multiple search values' do
|
||
let(:filter) do
|
||
[
|
||
{ 'table' => 'etablissement', 'column' => 'code_postal', 'value' => '75017' },
|
||
{ 'table' => 'etablissement', 'column' => 'code_postal', 'value' => '88100' }
|
||
]
|
||
end
|
||
|
||
let!(:other_kept_dossier) { create(:dossier, procedure: procedure, etablissement: create(:etablissement, code_postal: '88100')) }
|
||
|
||
it 'returns every dossier that matches any of the search criteria for a given column' do
|
||
is_expected.to contain_exactly(kept_dossier.id, other_kept_dossier.id)
|
||
end
|
||
end
|
||
end
|
||
end
|
||
|
||
context 'for user table' do
|
||
let(:filter) { [{ 'table' => 'user', 'column' => 'email', 'value' => 'keepmail' }] }
|
||
|
||
let!(:kept_dossier) { create(:dossier, procedure: procedure, user: create(:user, email: 'me@keepmail.com')) }
|
||
let!(:discarded_dossier) { create(:dossier, procedure: procedure, user: create(:user, email: 'me@discard.com')) }
|
||
|
||
it { is_expected.to contain_exactly(kept_dossier.id) }
|
||
|
||
context 'with multiple search values' do
|
||
let(:filter) do
|
||
[
|
||
{ 'table' => 'user', 'column' => 'email', 'value' => 'keepmail' },
|
||
{ 'table' => 'user', 'column' => 'email', 'value' => 'beta.gouv.fr' }
|
||
]
|
||
end
|
||
|
||
let!(:other_kept_dossier) { create(:dossier, procedure: procedure, user: create(:user, email: 'bazinga@beta.gouv.fr')) }
|
||
|
||
it 'returns every dossier that matches any of the search criteria for a given column' do
|
||
is_expected.to contain_exactly(kept_dossier.id, other_kept_dossier.id)
|
||
end
|
||
end
|
||
end
|
||
|
||
context 'for individual table' do
|
||
let(:procedure) { create(:procedure, :for_individual) }
|
||
let!(:kept_dossier) { create(:dossier, procedure: procedure, individual: create(:individual, gender: 'Mme', prenom: 'Josephine', nom: 'Baker')) }
|
||
let!(:discarded_dossier) { create(:dossier, procedure: procedure, individual: create(:individual, gender: 'M', prenom: 'Jean', nom: 'Tremblay')) }
|
||
|
||
context 'for gender column' do
|
||
let(:filter) { [{ 'table' => 'individual', 'column' => 'gender', 'value' => 'Mme' }] }
|
||
|
||
it { is_expected.to contain_exactly(kept_dossier.id) }
|
||
end
|
||
|
||
context 'for prenom column' do
|
||
let(:filter) { [{ 'table' => 'individual', 'column' => 'prenom', 'value' => 'Josephine' }] }
|
||
|
||
it { is_expected.to contain_exactly(kept_dossier.id) }
|
||
end
|
||
|
||
context 'for nom column' do
|
||
let(:filter) { [{ 'table' => 'individual', 'column' => 'nom', 'value' => 'Baker' }] }
|
||
|
||
it { is_expected.to contain_exactly(kept_dossier.id) }
|
||
end
|
||
|
||
context 'with multiple search values' do
|
||
let(:filter) do
|
||
[
|
||
{ 'table' => 'individual', 'column' => 'prenom', 'value' => 'Josephine' },
|
||
{ 'table' => 'individual', 'column' => 'prenom', 'value' => 'Romuald' }
|
||
]
|
||
end
|
||
|
||
let!(:other_kept_dossier) { create(:dossier, procedure: procedure, individual: create(:individual, gender: 'M', prenom: 'Romuald', nom: 'Pistis')) }
|
||
|
||
it 'returns every dossier that matches any of the search criteria for a given column' do
|
||
is_expected.to contain_exactly(kept_dossier.id, other_kept_dossier.id)
|
||
end
|
||
end
|
||
end
|
||
|
||
context 'for followers_instructeurs table' do
|
||
let(:filter) { [{ 'table' => 'followers_instructeurs', 'column' => 'email', 'value' => 'keepmail' }] }
|
||
|
||
let!(:kept_dossier) { create(:dossier, procedure: procedure) }
|
||
let!(:discarded_dossier) { create(:dossier, procedure: procedure) }
|
||
|
||
before do
|
||
create(:follow, dossier: kept_dossier, instructeur: create(:instructeur, email: 'me@keepmail.com'))
|
||
create(:follow, dossier: discarded_dossier, instructeur: create(:instructeur, email: 'me@discard.com'))
|
||
end
|
||
|
||
it { is_expected.to contain_exactly(kept_dossier.id) }
|
||
|
||
context 'with multiple search values' do
|
||
let(:filter) do
|
||
[
|
||
{ 'table' => 'followers_instructeurs', 'column' => 'email', 'value' => 'keepmail' },
|
||
{ 'table' => 'followers_instructeurs', 'column' => 'email', 'value' => 'beta.gouv.fr' }
|
||
]
|
||
end
|
||
|
||
let(:other_kept_dossier) { create(:dossier, procedure: procedure) }
|
||
|
||
before do
|
||
create(:follow, dossier: other_kept_dossier, instructeur: create(:instructeur, email: 'bazinga@beta.gouv.fr'))
|
||
end
|
||
|
||
it 'returns every dossier that matches any of the search criteria for a given column' do
|
||
is_expected.to contain_exactly(kept_dossier.id, other_kept_dossier.id)
|
||
end
|
||
end
|
||
end
|
||
|
||
context 'for groupe_instructeur table' do
|
||
let(:filter) { [{ 'table' => 'groupe_instructeur', 'column' => 'label', 'value' => 'défaut' }] }
|
||
|
||
let!(:gi_2) { procedure.groupe_instructeurs.create(label: '2') }
|
||
let!(:gi_3) { procedure.groupe_instructeurs.create(label: '3') }
|
||
|
||
let!(:kept_dossier) { create(:dossier, procedure: procedure) }
|
||
let!(:discarded_dossier) { create(:dossier, groupe_instructeur: gi_2) }
|
||
|
||
it { is_expected.to contain_exactly(kept_dossier.id) }
|
||
|
||
context 'with multiple search values' do
|
||
let(:filter) do
|
||
[
|
||
{ 'table' => 'groupe_instructeur', 'column' => 'label', 'value' => 'défaut' },
|
||
{ 'table' => 'groupe_instructeur', 'column' => 'label', 'value' => '3' }
|
||
]
|
||
end
|
||
|
||
let!(:other_kept_dossier) { create(:dossier, groupe_instructeur: gi_3) }
|
||
|
||
it 'returns every dossier that matches any of the search criteria for a given column' do
|
||
is_expected.to contain_exactly(kept_dossier.id, other_kept_dossier.id)
|
||
end
|
||
end
|
||
end
|
||
end
|
||
|
||
describe '#eager_load_displayed_fields' do
|
||
let(:procedure_presentation) { ProcedurePresentation.create(assign_to: assign_to, displayed_fields: [{ 'table' => table, 'column' => column }]) }
|
||
let!(:dossier) { create(:dossier, :en_construction, procedure: procedure) }
|
||
let(:displayed_dossier) { procedure_presentation.eager_load_displayed_fields(procedure.dossiers).first }
|
||
|
||
context 'for type de champ' do
|
||
let(:table) { 'type_de_champ' }
|
||
let(:column) { procedure.types_de_champ.first.id }
|
||
|
||
it 'preloads the champs relation' do
|
||
# Ideally, we would only preload the champs for the matching column
|
||
|
||
expect(displayed_dossier.association(:champs)).to be_loaded
|
||
expect(displayed_dossier.association(:champs_private)).not_to be_loaded
|
||
expect(displayed_dossier.association(:user)).not_to be_loaded
|
||
expect(displayed_dossier.association(:individual)).not_to be_loaded
|
||
expect(displayed_dossier.association(:etablissement)).not_to be_loaded
|
||
expect(displayed_dossier.association(:followers_instructeurs)).not_to be_loaded
|
||
expect(displayed_dossier.association(:groupe_instructeur)).not_to be_loaded
|
||
end
|
||
end
|
||
|
||
context 'for type de champ private' do
|
||
let(:table) { 'type_de_champ_private' }
|
||
let(:column) { procedure.types_de_champ_private.first.id }
|
||
|
||
it 'preloads the champs relation' do
|
||
# Ideally, we would only preload the champs for the matching column
|
||
|
||
expect(displayed_dossier.association(:champs)).not_to be_loaded
|
||
expect(displayed_dossier.association(:champs_private)).to be_loaded
|
||
expect(displayed_dossier.association(:user)).not_to be_loaded
|
||
expect(displayed_dossier.association(:individual)).not_to be_loaded
|
||
expect(displayed_dossier.association(:etablissement)).not_to be_loaded
|
||
expect(displayed_dossier.association(:followers_instructeurs)).not_to be_loaded
|
||
expect(displayed_dossier.association(:groupe_instructeur)).not_to be_loaded
|
||
end
|
||
end
|
||
|
||
context 'for user' do
|
||
let(:table) { 'user' }
|
||
let(:column) { 'email' }
|
||
|
||
it 'preloads the user relation' do
|
||
expect(displayed_dossier.association(:champs)).not_to be_loaded
|
||
expect(displayed_dossier.association(:champs_private)).not_to be_loaded
|
||
expect(displayed_dossier.association(:user)).to be_loaded
|
||
expect(displayed_dossier.association(:individual)).not_to be_loaded
|
||
expect(displayed_dossier.association(:etablissement)).not_to be_loaded
|
||
expect(displayed_dossier.association(:followers_instructeurs)).not_to be_loaded
|
||
expect(displayed_dossier.association(:groupe_instructeur)).not_to be_loaded
|
||
end
|
||
end
|
||
|
||
context 'for individual' do
|
||
let(:table) { 'individual' }
|
||
let(:column) { 'nom' }
|
||
|
||
it 'preloads the individual relation' do
|
||
expect(displayed_dossier.association(:champs)).not_to be_loaded
|
||
expect(displayed_dossier.association(:champs_private)).not_to be_loaded
|
||
expect(displayed_dossier.association(:user)).not_to be_loaded
|
||
expect(displayed_dossier.association(:individual)).to be_loaded
|
||
expect(displayed_dossier.association(:etablissement)).not_to be_loaded
|
||
expect(displayed_dossier.association(:followers_instructeurs)).not_to be_loaded
|
||
expect(displayed_dossier.association(:groupe_instructeur)).not_to be_loaded
|
||
end
|
||
end
|
||
|
||
context 'for etablissement' do
|
||
let(:table) { 'etablissement' }
|
||
let(:column) { 'siret' }
|
||
|
||
it 'preloads the etablissement relation' do
|
||
expect(displayed_dossier.association(:champs)).not_to be_loaded
|
||
expect(displayed_dossier.association(:champs_private)).not_to be_loaded
|
||
expect(displayed_dossier.association(:user)).not_to be_loaded
|
||
expect(displayed_dossier.association(:individual)).not_to be_loaded
|
||
expect(displayed_dossier.association(:etablissement)).to be_loaded
|
||
expect(displayed_dossier.association(:followers_instructeurs)).not_to be_loaded
|
||
expect(displayed_dossier.association(:groupe_instructeur)).not_to be_loaded
|
||
end
|
||
end
|
||
|
||
context 'for followers_instructeurs' do
|
||
let(:table) { 'followers_instructeurs' }
|
||
let(:column) { 'email' }
|
||
|
||
it 'preloads the followers_instructeurs relation' do
|
||
expect(displayed_dossier.association(:champs)).not_to be_loaded
|
||
expect(displayed_dossier.association(:champs_private)).not_to be_loaded
|
||
expect(displayed_dossier.association(:user)).not_to be_loaded
|
||
expect(displayed_dossier.association(:individual)).not_to be_loaded
|
||
expect(displayed_dossier.association(:etablissement)).not_to be_loaded
|
||
expect(displayed_dossier.association(:followers_instructeurs)).to be_loaded
|
||
expect(displayed_dossier.association(:groupe_instructeur)).not_to be_loaded
|
||
end
|
||
end
|
||
|
||
context 'for groupe_instructeur' do
|
||
let(:table) { 'groupe_instructeur' }
|
||
let(:column) { 'email' }
|
||
|
||
it 'preloads the followers_instructeurs relation' do
|
||
expect(displayed_dossier.association(:champs)).not_to be_loaded
|
||
expect(displayed_dossier.association(:champs_private)).not_to be_loaded
|
||
expect(displayed_dossier.association(:user)).not_to be_loaded
|
||
expect(displayed_dossier.association(:individual)).not_to be_loaded
|
||
expect(displayed_dossier.association(:etablissement)).not_to be_loaded
|
||
expect(displayed_dossier.association(:followers_instructeurs)).not_to be_loaded
|
||
expect(displayed_dossier.association(:groupe_instructeur)).to be_loaded
|
||
end
|
||
end
|
||
|
||
context 'for self' do
|
||
let(:table) { 'self' }
|
||
let(:column) { 'created_at' }
|
||
|
||
it 'does not preload anything' do
|
||
expect(displayed_dossier.association(:champs)).not_to be_loaded
|
||
expect(displayed_dossier.association(:champs_private)).not_to be_loaded
|
||
expect(displayed_dossier.association(:user)).not_to be_loaded
|
||
expect(displayed_dossier.association(:individual)).not_to be_loaded
|
||
expect(displayed_dossier.association(:etablissement)).not_to be_loaded
|
||
expect(displayed_dossier.association(:followers_instructeurs)).not_to be_loaded
|
||
end
|
||
end
|
||
end
|
||
end
|