diff --git a/atlas/atlasAPI.py b/atlas/atlasAPI.py index de661edce..ad4952423 100644 --- a/atlas/atlasAPI.py +++ b/atlas/atlasAPI.py @@ -20,7 +20,7 @@ def searchTaxonAPI(): session = db.session search = request.args.get("search", "") limit = request.args.get("limit", 50) - results = vmSearchTaxonRepository.listeTaxonsSearch(session, search, limit) + results = vmSearchTaxonRepository.searchTaxons(session, search, limit) session.close() return jsonify(results) diff --git a/atlas/modeles/entities/vmSearchTaxon.py b/atlas/modeles/entities/vmSearchTaxon.py index a6e4c3813..3fb9b2b9e 100644 --- a/atlas/modeles/entities/vmSearchTaxon.py +++ b/atlas/modeles/entities/vmSearchTaxon.py @@ -14,6 +14,7 @@ class VmSearchTaxon(Base): metadata, Column("cd_ref", Integer, primary_key=True, unique=True), Column("cd_nom", Integer), + Column("display_name", String), Column("search_name", String), schema="atlas", autoload=True, diff --git a/atlas/modeles/repositories/vmSearchTaxonRepository.py b/atlas/modeles/repositories/vmSearchTaxonRepository.py index 95c138153..c9798caac 100644 --- a/atlas/modeles/repositories/vmSearchTaxonRepository.py +++ b/atlas/modeles/repositories/vmSearchTaxonRepository.py @@ -19,7 +19,7 @@ def listeTaxons(session): return taxonList -def listeTaxonsSearch(session, search, limit=50): +def searchTaxons(session, search, limit=50): """ Recherche dans la VmSearchTaxon en ilike Utilisé pour l'autocomplétion de la recherche de taxon @@ -34,20 +34,20 @@ def listeTaxonsSearch(session, search, limit=50): label = search_name value = cd_ref """ - - req = session.query( - VmSearchTaxon.search_name, - VmSearchTaxon.cd_ref, - func.similarity(VmSearchTaxon.search_name, search).label("idx_trgm"), - ).distinct() - - search = search.replace(" ", "%") - req = ( - req.filter(VmSearchTaxon.search_name.ilike("%" + search + "%")) + like_search = "%" + search.replace(" ", "%") + "%" + + query = ( + session.query( + VmSearchTaxon.display_name, + VmSearchTaxon.cd_ref, + func.similarity(VmSearchTaxon.search_name, search).label("idx_trgm"), + ) + .distinct() + .filter(func.unaccent(VmSearchTaxon.search_name).ilike(func.unaccent(like_search))) .order_by(desc("idx_trgm")) .order_by(VmSearchTaxon.cd_ref == VmSearchTaxon.cd_nom) .limit(limit) ) - data = req.all() + results = query.all() - return [{"label": d[0], "value": d[1]} for d in data] + return [{"label": r[0], "value": r[1]} for r in results] diff --git a/data/atlas/5.atlas.vm_search_taxon.sql b/data/atlas/5.atlas.vm_search_taxon.sql index 083783573..68bbfa7b4 100644 --- a/data/atlas/5.atlas.vm_search_taxon.sql +++ b/data/atlas/5.atlas.vm_search_taxon.sql @@ -1,34 +1,62 @@ --- Taxons observés et de tous leurs synonymes (utilisés pour la recherche d'une espèce) +CREATE MATERIALIZED VIEW atlas.vm_search_taxon AS + WITH verna_names AS ( + SELECT DISTINCT + cd_nom, + lb_nom, + cd_ref, + STRING_TO_TABLE(nom_vern, ', ') AS nom_vern + FROM atlas.vm_taxref + WHERE nom_vern IS NOT NULL + AND cd_nom = cd_ref + AND nom_vern <> lb_nom + ), + names AS ( + -- Chosen scinames + SELECT + cd_nom, + cd_ref, + lb_nom AS search_name, + CONCAT('', REPLACE(nom_complet_html, lb_auteur, ''), ' ', lb_auteur) AS display_name + FROM atlas.vm_taxref + WHERE cd_nom = cd_ref -CREATE MATERIALIZED VIEW atlas.vm_search_taxon AS -SELECT row_number() OVER (ORDER BY t.cd_nom,t.cd_ref,t.search_name)::integer AS fid, - t.cd_nom, - t.cd_ref, - t.search_name, - t.nom_valide, - t.lb_nom -FROM ( - SELECT t_1.cd_nom, - t_1.cd_ref, - concat(t_1.lb_nom, ' = ', t_1.nom_valide, '') AS search_name, - t_1.nom_valide, - t_1.lb_nom - FROM atlas.vm_taxref t_1 + UNION - UNION - SELECT t_1.cd_nom, - t_1.cd_ref, - concat(t_1.nom_vern, ' = ', t_1.nom_valide, '' ) AS search_name, - t_1.nom_valide, - t_1.lb_nom - FROM atlas.vm_taxref t_1 - WHERE t_1.nom_vern IS NOT NULL AND t_1.cd_nom = t_1.cd_ref -) t -JOIN atlas.vm_taxons taxons ON taxons.cd_ref = t.cd_ref; + -- Synonym scinames + SELECT + t1.cd_nom, + t1.cd_ref, + t1.lb_nom AS search_name, + CONCAT(REPLACE(t1.nom_complet_html, t1.lb_auteur, ''), ' = ', REPLACE(t2.nom_complet_html, t2.lb_auteur, ''), ' ', t2.lb_auteur) AS display_name + FROM atlas.vm_taxref AS t1 + JOIN atlas.vm_taxref AS t2 + ON t1.cd_ref = t2.cd_nom + WHERE t1.cd_nom <> t1.cd_ref + + UNION + + -- Vernacular names + SELECT + v.cd_nom, + v.cd_ref, + v.nom_vern AS search_name, + CONCAT(v.nom_vern, ' = ', REPLACE(t.nom_complet_html, t.lb_auteur, ''), ' ', t.lb_auteur) AS display_name + FROM verna_names AS v + JOIN atlas.vm_taxref AS t + ON t.cd_nom = v.cd_ref + WHERE v.nom_vern <> v.lb_nom + ) + SELECT ROW_NUMBER() OVER (ORDER BY n.cd_nom, n.cd_ref, n.search_name)::integer AS fid, + n.cd_nom, + n.cd_ref, + n.search_name, + n.display_name + FROM atlas.vm_taxons AS t + JOIN names AS n + ON t.cd_ref = n.cd_ref ; CREATE UNIQUE INDEX ON atlas.vm_search_taxon(fid); CREATE INDEX ON atlas.vm_search_taxon(cd_nom); -create INDEX ON atlas.vm_search_taxon(cd_ref); - +CREATE INDEX ON atlas.vm_search_taxon(cd_ref); CREATE INDEX trgm_idx ON atlas.vm_search_taxon USING GIST (search_name gist_trgm_ops); -CREATE UNIQUE INDEX ON atlas.vm_search_taxon (cd_nom, search_name); \ No newline at end of file +CREATE UNIQUE INDEX ON atlas.vm_search_taxon (cd_nom, search_name); diff --git a/data/update/update_1.6.1to1.6.2.sql b/data/update/update_1.6.1to1.6.2.sql index 8841c9526..03c3eba9a 100644 --- a/data/update/update_1.6.1to1.6.2.sql +++ b/data/update/update_1.6.1to1.6.2.sql @@ -43,3 +43,72 @@ CREATE INDEX ON atlas.vm_observations_mailles CREATE INDEX ON atlas.vm_observations_mailles USING btree (id_maille, cd_ref); + + +-- ISSUE #531 & #532 +CREATE EXTENSION IF NOT EXISTS unaccent SCHEMA "public"; + +-- ISSUE #532 +DROP MATERIALIZED VIEW IF EXISTS atlas.vm_search_taxon ; +CREATE MATERIALIZED VIEW atlas.vm_search_taxon AS + WITH verna_names AS ( + SELECT DISTINCT + cd_nom, + lb_nom, + cd_ref, + STRING_TO_TABLE(nom_vern, ', ') AS nom_vern + FROM atlas.vm_taxref + WHERE nom_vern IS NOT NULL + AND cd_nom = cd_ref + AND nom_vern <> lb_nom + ), + names AS ( + -- Chosen scinames + SELECT + cd_nom, + cd_ref, + lb_nom AS search_name, + CONCAT('', REPLACE(nom_complet_html, lb_auteur, ''), ' ', lb_auteur) AS display_name + FROM atlas.vm_taxref + WHERE cd_nom = cd_ref + + UNION + + -- Synonym scinames + SELECT + t1.cd_nom, + t1.cd_ref, + t1.lb_nom AS search_name, + CONCAT(REPLACE(t1.nom_complet_html, t1.lb_auteur, ''), ' = ', REPLACE(t2.nom_complet_html, t2.lb_auteur, ''), ' ', t2.lb_auteur) AS display_name + FROM atlas.vm_taxref AS t1 + JOIN atlas.vm_taxref AS t2 + ON t1.cd_ref = t2.cd_nom + WHERE t1.cd_nom <> t1.cd_ref + + UNION + + -- Vernacular names + SELECT + v.cd_nom, + v.cd_ref, + v.nom_vern AS search_name, + CONCAT(v.nom_vern, ' = ', REPLACE(t.nom_complet_html, t.lb_auteur, ''), ' ', t.lb_auteur) AS display_name + FROM verna_names AS v + JOIN atlas.vm_taxref AS t + ON t.cd_nom = v.cd_ref + WHERE v.nom_vern <> v.lb_nom + ) + SELECT ROW_NUMBER() OVER (ORDER BY n.cd_nom, n.cd_ref, n.search_name)::integer AS fid, + n.cd_nom, + n.cd_ref, + n.search_name, + n.display_name + FROM atlas.vm_taxons AS t + JOIN names AS n + ON t.cd_ref = n.cd_ref ; + +CREATE UNIQUE INDEX ON atlas.vm_search_taxon(fid); +CREATE INDEX ON atlas.vm_search_taxon(cd_nom); +CREATE INDEX ON atlas.vm_search_taxon(cd_ref); +CREATE INDEX trgm_idx ON atlas.vm_search_taxon USING GIST (search_name gist_trgm_ops); +CREATE UNIQUE INDEX ON atlas.vm_search_taxon (cd_nom, search_name); diff --git a/docs/changelog.rst b/docs/changelog.rst index 2709dccf9..ac61947d7 100644 --- a/docs/changelog.rst +++ b/docs/changelog.rst @@ -2,6 +2,24 @@ CHANGELOG ========= +[Unreleased] +------------------ + +🐛 **Optimisations** + +- Amélioration de la "recherche par espèces" (#532 par @jpm-cbna) + + +⚠️ **Notes de version** + +Si vous mettez à jour GeoNature-atlas : + +- Exécutez le script SQL de mise à jour de la BDD : https://github.com/PnX-SI/GeoNature-atlas/blob/master/data/update/update_1.6.1to1.6.2.sql +- Donner les droits à l'utilisateur en lecture seule de l'Atlas (habituellement geonatatlas) : + :: + GRANT SELECT ON TABLE atlas.vm_search_taxon TO geonatatlas; + + 1.6.1 (2023-10-16) ------------------