Merge pull request #11107 from demarches-simplifiees/better_search
ETQ usager et instructeur, la recherche ne tient plus compte des accents
This commit is contained in:
commit
9fabef0ac7
2 changed files with 102 additions and 207 deletions
|
@ -6,7 +6,7 @@ class DossierSearchService
|
||||||
[]
|
[]
|
||||||
else
|
else
|
||||||
dossier_by_exact_id(dossiers, search_terms)
|
dossier_by_exact_id(dossiers, search_terms)
|
||||||
.presence || dossier_by_full_text(dossiers, search_terms, with_annotations)
|
.presence || dossier_ids_by_full_text(dossiers, search_terms, with_annotations)
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
|
@ -26,24 +26,23 @@ class DossierSearchService
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
def self.dossier_by_full_text(dossiers, search_terms, with_annotations)
|
def self.dossier_ids_by_full_text(dossiers, search_terms, with_annotations)
|
||||||
ts_vector = "to_tsvector('french', #{with_annotations ? 'dossiers.search_terms || dossiers.private_search_terms' : 'dossiers.search_terms'})"
|
dossier_by_full_text(dossiers.visible_by_administration, search_terms, with_annotations:)
|
||||||
ts_query = "to_tsquery('french', #{Dossier.connection.quote(to_tsquery(search_terms))})"
|
|
||||||
|
|
||||||
dossiers
|
|
||||||
.visible_by_administration
|
|
||||||
.where("#{ts_vector} @@ #{ts_query}")
|
|
||||||
.order(Arel.sql("COALESCE(ts_rank(#{ts_vector}, #{ts_query}), 0) DESC"))
|
|
||||||
.pluck('id')
|
.pluck('id')
|
||||||
.uniq
|
.uniq
|
||||||
end
|
end
|
||||||
|
|
||||||
def self.dossier_by_full_text_for_user(search_terms, dossiers)
|
def self.dossier_by_full_text_for_user(search_terms, dossiers)
|
||||||
ts_vector = "to_tsvector('french', search_terms)"
|
dossier_by_full_text(dossiers.visible_by_user, search_terms)
|
||||||
ts_query = "to_tsquery('french', #{Dossier.includes(:procedure).connection.quote(to_tsquery(search_terms))})"
|
end
|
||||||
|
|
||||||
|
def self.dossier_by_full_text(dossiers, search_terms, with_annotations: false)
|
||||||
|
columns = with_annotations ? 'search_terms || \' \' || private_search_terms' : 'search_terms'
|
||||||
|
|
||||||
|
ts_vector = "to_tsvector('french', unaccent(#{columns}))"
|
||||||
|
ts_query = "to_tsquery('french', unaccent(#{Dossier.connection.quote(to_tsquery(search_terms))}))"
|
||||||
|
|
||||||
dossiers
|
dossiers
|
||||||
.visible_by_user
|
|
||||||
.where("#{ts_vector} @@ #{ts_query}")
|
.where("#{ts_vector} @@ #{ts_query}")
|
||||||
.order(Arel.sql("COALESCE(ts_rank(#{ts_vector}, #{ts_query}), 0) DESC"))
|
.order(Arel.sql("COALESCE(ts_rank(#{ts_vector}, #{ts_query}), 0) DESC"))
|
||||||
end
|
end
|
||||||
|
|
|
@ -2,229 +2,125 @@
|
||||||
|
|
||||||
describe DossierSearchService do
|
describe DossierSearchService do
|
||||||
describe '#matching_dossiers' do
|
describe '#matching_dossiers' do
|
||||||
subject { liste_dossiers }
|
let!(:dossiers) { Dossier.where(id: dossier.id) }
|
||||||
|
|
||||||
let(:liste_dossiers) do
|
before { perform_enqueued_jobs(only: DossierIndexSearchTermsJob) }
|
||||||
described_class.matching_dossiers(instructeur_1.dossiers, terms)
|
|
||||||
|
def searching(terms, with_annotations: false)
|
||||||
|
described_class.matching_dossiers(dossiers, terms, with_annotations)
|
||||||
end
|
end
|
||||||
|
|
||||||
let(:administrateur_1) { administrateurs(:default_admin) }
|
describe 'ignores brouillon' do
|
||||||
let(:administrateur_2) { administrateurs(:default_admin) }
|
let(:dossier) { create(:dossier, state: :brouillon) }
|
||||||
|
|
||||||
let(:instructeur_1) { create(:instructeur, administrateurs: [administrateur_1]) }
|
it { expect(searching(dossier.id.to_s)).to eq([]) }
|
||||||
let(:instructeur_2) { create(:instructeur, administrateurs: [administrateur_2]) }
|
|
||||||
|
|
||||||
before do
|
|
||||||
instructeur_1.assign_to_procedure(procedure_1)
|
|
||||||
instructeur_2.assign_to_procedure(procedure_2)
|
|
||||||
|
|
||||||
# create dossier before performing jobs
|
|
||||||
# because let!() syntax is executed after "before" callback
|
|
||||||
dossier_0
|
|
||||||
dossier_1
|
|
||||||
dossier_2
|
|
||||||
dossier_3
|
|
||||||
dossier_archived
|
|
||||||
|
|
||||||
perform_enqueued_jobs(only: DossierIndexSearchTermsJob)
|
|
||||||
end
|
end
|
||||||
|
|
||||||
let(:procedure_1) { create(:procedure, :published, administrateur: administrateur_1) }
|
context 'with a dossier not in brouillon' do
|
||||||
let(:procedure_2) { create(:procedure, :published, administrateur: administrateur_2) }
|
let(:user) { create(:user, email: 'nicolas@email.com') }
|
||||||
|
let(:etablissement) { create(:etablissement, entreprise_raison_sociale: 'Direction Interministerielle Du Numérique', siret: '13002526500013') }
|
||||||
let(:dossier_0) { create(:dossier, state: Dossier.states.fetch(:brouillon), procedure: procedure_1, user: create(:user, email: 'brouillon@clap.fr')) }
|
let(:procedure) { create(:procedure, types_de_champ_public: [{ type: :text }], types_de_champ_private: [{ type: :text }]) }
|
||||||
|
let(:dossier) do
|
||||||
let(:etablissement_1) { create(:etablissement, entreprise_raison_sociale: 'OCTO Academy', siret: '41636169600051') }
|
create(:dossier, procedure:, state: :en_construction, user:, etablissement:).tap do |dossier|
|
||||||
let(:dossier_1) { create(:dossier, :en_construction, procedure: procedure_1, user: create(:user, email: 'contact@test.com'), etablissement: etablissement_1) }
|
dossier.project_champs_public.first.update!(value: 'Hélène mange des pommes')
|
||||||
|
dossier.project_champs_private.first.update!(value: 'annotations')
|
||||||
let(:etablissement_2) { create(:etablissement, entreprise_raison_sociale: 'Plop octo', siret: '41816602300012') }
|
end
|
||||||
let(:dossier_2) { create(:dossier, :en_construction, procedure: procedure_1, user: create(:user, email: 'plop@gmail.com'), etablissement: etablissement_2) }
|
|
||||||
|
|
||||||
let(:etablissement_3) { create(:etablissement, entreprise_raison_sociale: 'OCTO Technology', siret: '41816609600051') }
|
|
||||||
let(:dossier_3) { create(:dossier, :en_construction, procedure: procedure_2, user: create(:user, email: 'peace@clap.fr'), etablissement: etablissement_3) }
|
|
||||||
|
|
||||||
let(:dossier_archived) { create(:dossier, :en_construction, procedure: procedure_1, archived: true, user: create(:user, email: 'archived@clap.fr')) }
|
|
||||||
|
|
||||||
describe 'search is empty' do
|
|
||||||
let(:terms) { '' }
|
|
||||||
|
|
||||||
it { expect(subject.size).to eq(0) }
|
|
||||||
end
|
|
||||||
|
|
||||||
describe 'search brouillon file' do
|
|
||||||
let(:terms) { 'brouillon' }
|
|
||||||
|
|
||||||
it { expect(subject.size).to eq(0) }
|
|
||||||
end
|
|
||||||
|
|
||||||
describe 'search archived file' do
|
|
||||||
let(:terms) { 'archived' }
|
|
||||||
|
|
||||||
it { expect(subject.size).to eq(1) }
|
|
||||||
end
|
|
||||||
|
|
||||||
describe 'search on contact email' do
|
|
||||||
let(:terms) { 'clap' }
|
|
||||||
|
|
||||||
it { expect(subject.size).to eq(0) }
|
|
||||||
end
|
|
||||||
|
|
||||||
describe 'search on SIRET' do
|
|
||||||
context 'when is part of SIRET' do
|
|
||||||
let(:terms) { '4181' }
|
|
||||||
|
|
||||||
it { expect(subject.size).to eq(1) }
|
|
||||||
end
|
end
|
||||||
|
|
||||||
context 'when is a complet SIRET' do
|
it do
|
||||||
let(:terms) { '41816602300012' }
|
expect(searching('')).to eq([])
|
||||||
|
|
||||||
it { expect(subject.size).to eq(1) }
|
# by dossier id
|
||||||
|
expect(searching(dossier.id.to_s)).to eq([dossier.id])
|
||||||
|
|
||||||
|
# annotations is unsearchable by default
|
||||||
|
expect(searching('annotations')).to eq([])
|
||||||
|
# but can be searched with the with_annotations option
|
||||||
|
expect(searching('annotations', with_annotations: true)).to eq([dossier.id])
|
||||||
|
|
||||||
|
# by email
|
||||||
|
expect(searching('nicolas@email.com')).to eq([dossier.id])
|
||||||
|
expect(searching('nicolas')).to eq([dossier.id])
|
||||||
|
|
||||||
|
# by SIRET
|
||||||
|
expect(searching('13002526500013')).to eq([dossier.id])
|
||||||
|
expect(searching('1300')).to eq([dossier.id])
|
||||||
|
|
||||||
|
# by raison sociale
|
||||||
|
expect(searching('Direction Interministerielle Du Numérique')).to eq([dossier.id])
|
||||||
|
expect(searching('Direction')).to eq([dossier.id])
|
||||||
|
|
||||||
|
# with multiple terms
|
||||||
|
expect(searching('Direction nicolas')).to eq([dossier.id])
|
||||||
|
|
||||||
|
# with forbidden characters
|
||||||
|
expect(searching("'?\\:&!(Direction) <Interministerielle>")).to eq([dossier.id])
|
||||||
|
|
||||||
|
# with a single forbidden character should not crash postgres
|
||||||
|
expect(searching('? Direction')).to eq([dossier.id])
|
||||||
|
|
||||||
|
# with supirious spaces
|
||||||
|
expect(searching(" nicolas ")).to eq([dossier.id])
|
||||||
|
|
||||||
|
# with wrong case
|
||||||
|
expect(searching('direction')).to eq([dossier.id])
|
||||||
|
|
||||||
|
# by champ text
|
||||||
|
expect(searching('Hélène')).to eq([dossier.id])
|
||||||
|
|
||||||
|
# by singular
|
||||||
|
expect(searching('la pomme')).to eq([dossier.id])
|
||||||
|
|
||||||
|
# without accent
|
||||||
|
expect(searching('helene')).to eq([dossier.id])
|
||||||
|
|
||||||
|
# NOT WORKING YET
|
||||||
|
# with a single faulty character
|
||||||
|
expect(searching('des pammes')).to eq([])
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
describe 'search on raison social' do
|
describe 'does not ignore archived dossiers' do
|
||||||
let(:terms) { 'OCTO' }
|
let(:dossier) { create(:dossier, state: :en_construction, archived: true) }
|
||||||
|
|
||||||
it { expect(subject.size).to eq(2) }
|
it { expect(searching(dossier.id.to_s)).to eq([dossier.id]) }
|
||||||
end
|
|
||||||
|
|
||||||
describe 'search terms surrounded with spurious spaces' do
|
|
||||||
let(:terms) { ' OCTO ' }
|
|
||||||
|
|
||||||
it { expect(subject.size).to eq(2) }
|
|
||||||
end
|
|
||||||
|
|
||||||
describe 'search on multiple fields' do
|
|
||||||
let(:terms) { 'octo plop' }
|
|
||||||
|
|
||||||
it { expect(subject.size).to eq(1) }
|
|
||||||
end
|
|
||||||
|
|
||||||
describe 'search with characters disallowed by the tsquery parser' do
|
|
||||||
let(:terms) { "'?\\:&!(OCTO) <plop>" }
|
|
||||||
|
|
||||||
it { expect(subject.size).to eq(1) }
|
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
describe '#matching_dossiers_for_user' do
|
describe '#matching_dossiers_for_user' do
|
||||||
subject { liste_dossiers }
|
let(:user) { create(:user) }
|
||||||
|
let(:another_user) { create(:user) }
|
||||||
|
|
||||||
before do
|
before { perform_enqueued_jobs(only: DossierIndexSearchTermsJob) }
|
||||||
dossier_0
|
|
||||||
dossier_0b
|
|
||||||
dossier_1
|
|
||||||
dossier_2
|
|
||||||
dossier_3
|
|
||||||
dossier_archived
|
|
||||||
perform_enqueued_jobs(only: DossierIndexSearchTermsJob)
|
|
||||||
end
|
|
||||||
|
|
||||||
let(:liste_dossiers) do
|
def searching(terms, user) = described_class.matching_dossiers_for_user(terms, user)
|
||||||
described_class.matching_dossiers_for_user(terms, user_1)
|
|
||||||
end
|
|
||||||
|
|
||||||
let(:user_1) { create(:user, email: 'bidou@clap.fr') }
|
context 'when the dossier is brouillon' do
|
||||||
let(:user_2) { create(:user) }
|
let(:procedure) { create(:procedure, types_de_champ_private: [{ type: :text }]) }
|
||||||
|
let(:dossier) do
|
||||||
let(:procedure_1) { create(:procedure, :published) }
|
create(:dossier, procedure:, state: :brouillon, user:).tap do |dossier|
|
||||||
let(:procedure_2) { create(:procedure, :published) }
|
dossier.project_champs_private.first.update!(value: 'annotations')
|
||||||
|
end
|
||||||
let(:dossier_0) { create(:dossier, state: Dossier.states.fetch(:brouillon), procedure: procedure_1, user: user_1) }
|
|
||||||
let(:dossier_0b) { create(:dossier, state: Dossier.states.fetch(:brouillon), procedure: procedure_1, user: user_2) }
|
|
||||||
|
|
||||||
let(:etablissement_1) { create(:etablissement, entreprise_raison_sociale: 'OCTO Academy', siret: '41636169600051') }
|
|
||||||
let(:dossier_1) { create(:dossier, state: Dossier.states.fetch(:en_construction), procedure: procedure_1, user: user_1, etablissement: etablissement_1) }
|
|
||||||
|
|
||||||
let(:etablissement_2) { create(:etablissement, entreprise_raison_sociale: 'Plop octo', siret: '41816602300012') }
|
|
||||||
let(:dossier_2) { create(:dossier, state: Dossier.states.fetch(:en_construction), procedure: procedure_1, user: user_1, etablissement: etablissement_2) }
|
|
||||||
|
|
||||||
let(:etablissement_3) { create(:etablissement, entreprise_raison_sociale: 'OCTO Technology', siret: '41816609600051') }
|
|
||||||
let(:dossier_3) { create(:dossier, state: Dossier.states.fetch(:en_construction), procedure: procedure_2, user: user_1, etablissement: etablissement_3) }
|
|
||||||
|
|
||||||
let(:dossier_archived) { create(:dossier, state: Dossier.states.fetch(:en_construction), procedure: procedure_1, archived: true, user: user_1) }
|
|
||||||
|
|
||||||
describe 'search is empty' do
|
|
||||||
let(:terms) { '' }
|
|
||||||
|
|
||||||
it { expect(subject.size).to eq(0) }
|
|
||||||
end
|
|
||||||
|
|
||||||
describe 'search by dossier id' do
|
|
||||||
context 'when the user owns the dossier' do
|
|
||||||
let(:terms) { dossier_0.id.to_s }
|
|
||||||
|
|
||||||
it { expect(subject.map(&:id)).to include(dossier_0.id) }
|
|
||||||
end
|
end
|
||||||
|
|
||||||
context 'when the user does not own the dossier' do
|
it do
|
||||||
let(:terms) { dossier_0b.id.to_s }
|
# searching its own dossier by id
|
||||||
|
expect(searching(dossier.id.to_s, user)).to eq([dossier])
|
||||||
|
|
||||||
it { expect(subject.map(&:id)).not_to include(dossier_0b.id) }
|
# searching another dossier by id
|
||||||
|
expect(searching(dossier.id.to_s, another_user)).to eq([])
|
||||||
|
|
||||||
|
# annotations is unsearchable
|
||||||
|
expect(searching('annotations', user)).to eq([])
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
describe 'search brouillon file' do
|
context 'when the user is invited on the dossier' do
|
||||||
let(:terms) { 'brouillon' }
|
let(:dossier) { create(:dossier) }
|
||||||
|
|
||||||
it { expect(subject.size).to eq(0) }
|
before { create(:invite, dossier:, user:) }
|
||||||
end
|
|
||||||
|
|
||||||
describe 'search on contact email' do
|
it { expect(searching(dossier.id.to_s, user)).to eq([dossier]) }
|
||||||
let(:terms) { 'bidou@clap.fr' }
|
|
||||||
|
|
||||||
it { expect(subject.size).to eq(5) }
|
|
||||||
end
|
|
||||||
|
|
||||||
describe 'search on contact name' do
|
|
||||||
let(:terms) { 'bidou@clap.fr' }
|
|
||||||
|
|
||||||
it { expect(subject.size).to eq(5) }
|
|
||||||
end
|
|
||||||
|
|
||||||
describe 'search on SIRET' do
|
|
||||||
context 'when is part of SIRET' do
|
|
||||||
let(:terms) { '4181' }
|
|
||||||
|
|
||||||
it { expect(subject.size).to eq(2) }
|
|
||||||
end
|
|
||||||
|
|
||||||
context 'when is a complet SIRET' do
|
|
||||||
let(:terms) { '41816602300012' }
|
|
||||||
|
|
||||||
it { expect(subject.size).to eq(1) }
|
|
||||||
end
|
|
||||||
end
|
|
||||||
|
|
||||||
describe 'search on raison social' do
|
|
||||||
let(:terms) { 'OCTO' }
|
|
||||||
|
|
||||||
it { expect(subject.size).to eq(3) }
|
|
||||||
end
|
|
||||||
|
|
||||||
describe 'search terms surrounded with spurious spaces' do
|
|
||||||
let(:terms) { ' OCTO ' }
|
|
||||||
|
|
||||||
it { expect(subject.size).to eq(3) }
|
|
||||||
end
|
|
||||||
|
|
||||||
describe 'search on multiple fields' do
|
|
||||||
let(:terms) { 'octo plop' }
|
|
||||||
|
|
||||||
it { expect(subject.size).to eq(1) }
|
|
||||||
end
|
|
||||||
|
|
||||||
describe 'search with characters disallowed by the tsquery parser' do
|
|
||||||
let(:terms) { "'?\\:&!(OCTO) <plop>" }
|
|
||||||
|
|
||||||
it { expect(subject.size).to eq(1) }
|
|
||||||
end
|
|
||||||
|
|
||||||
describe 'search with a single forbidden character should not crash postgres' do
|
|
||||||
let(:terms) { '? OCTO' }
|
|
||||||
|
|
||||||
it { expect(subject.size).to eq(3) }
|
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
Loading…
Reference in a new issue