Code source de geonature.core.gn_synthese.utils.blurring

from functools import lru_cache

from flask import g
import sqlalchemy as sa
from sqlalchemy.orm import aliased
from pypnnomenclature.models import BibNomenclaturesTypes, TNomenclatures
from ref_geo.models import BibAreasTypes, LAreas

from geonature.core.gn_synthese.models import CorAreaSynthese, Synthese, VSyntheseForWebApp
from geonature.core.sensitivity.models import cor_sensitivity_area_type
from geonature.core.gn_synthese.utils.query_select_sqla import SyntheseQuery

from geonature.utils.env import db


[docs] def split_blurring_precise_permissions(permissions): """ Return permissions respectively with and without sensitivity filter. """ return [p for p in permissions if p.sensitivity_filter], [ p for p in permissions if not p.sensitivity_filter ]
@lru_cache() # to retrive non sensitive nomenclature only on first call
[docs] def build_sensitive_unsensitive_filters(): """ Return where clauses for sensitive and non-sensitive observations. """ non_sensitive_nomenc = db.session.scalar( sa.select(TNomenclatures.id_nomenclature).where( TNomenclatures.nomenclature_type.has(BibNomenclaturesTypes.mnemonique == "SENSIBILITE"), TNomenclatures.cd_nomenclature == "0", ) ) return ( Synthese.id_nomenclature_sensitivity != non_sensitive_nomenc, Synthese.id_nomenclature_sensitivity == non_sensitive_nomenc, )
[docs] def build_blurred_precise_geom_queries( filters, where_clauses: list = [], select_size_hierarchy=False ): # Build 2 queries that will be UNIONed # The where_clauses list enables to add more conditions to the base query # Used in export query where_clauses.append(Synthese.the_geom_4326.isnot(None)) # Query precise geom, for use with unsensitive observations # and sensitive observations with precise permission columns = [ sa.literal(1).label("priority"), Synthese.id_synthese.label("id_synthese"), Synthese.the_geom_4326.label("geom"), ] # Size hierarchy can be used here to filter on it in # a mesh mode scenario. if select_size_hierarchy: # 0 since no blurring geometry is associated here and a point have a 0 size columns.append(sa.literal(0).label("size_hierarchy")) precise_geom_query = SyntheseQuery( Synthese, sa.select(*columns).where(sa.and_(*where_clauses)).order_by(Synthese.id_synthese.desc()), filters=dict(filters), # not to edit the actual filter object ) # In both queries, we applied all filters so that we do not need to query the # whole synthese table precise_geom_query.filter_taxonomy() precise_geom_query.filter_other_filters(g.current_user) precise_geom_query.build_query() # Query blurred geom, for use with sensitive observations CorAreaSyntheseAlias = aliased(CorAreaSynthese) LAreasAlias = aliased(LAreas) BibAreasTypesAlias = aliased(BibAreasTypes) # TODO@LAreas.geom_4326 geom = LAreasAlias.geom.st_transform(4326).label("geom") # In SyntheseQuery below : # - query_joins parameter is needed to bypass # "self.query_joins is not None" condition in the build_query() method below # - priority is used to prevail non blurred geom over blurred geom if the user # can access to the non blurred geom # - orderby needed to match the non blurred and the blurred observations columns = [ sa.literal(2).label("priority"), Synthese.id_synthese.label("id_synthese"), geom, ] # size hierarchy is the size of the joined blurring area if select_size_hierarchy: columns.append(BibAreasTypes.size_hierarchy.label("size_hierarchy")) blurred_geom_query = SyntheseQuery( Synthese, sa.select(*columns) .where( cor_sensitivity_area_type.c.id_nomenclature_sensitivity == Synthese.id_nomenclature_sensitivity ) .where(sa.and_(*where_clauses)) .order_by(Synthese.id_synthese.desc()), filters=dict(filters), query_joins=sa.join( Synthese, CorAreaSyntheseAlias, CorAreaSyntheseAlias.id_synthese == Synthese.id_synthese, ), geom_column=LAreas.geom_4326, ) # Joins here are needed to retrieve the blurred geometry blurred_geom_query.add_join(LAreasAlias, LAreasAlias.id_area, CorAreaSyntheseAlias.id_area) blurred_geom_query.add_join(BibAreasTypesAlias, BibAreasTypesAlias.id_type, LAreasAlias.id_type) blurred_geom_query.add_join( cor_sensitivity_area_type, cor_sensitivity_area_type.c.id_area_type, BibAreasTypesAlias.id_type, ) # Same for the first query => apply filter to avoid querying the whole table blurred_geom_query.filter_taxonomy() blurred_geom_query.filter_other_filters(g.current_user) blurred_geom_query.build_query() return blurred_geom_query, precise_geom_query
[docs] def build_allowed_geom_cte( blurring_permissions, precise_permissions, blurred_geom_query, precise_geom_query, limit, ): """ """ # The goal is to separate the blurring and precise permissions. # But in sensitive permissions there can be unsensitive observations so we need # to split them. # sensitive_where_clause and unsensitive_where_clause represents this split # See https://github.com/PnX-SI/GeoNature/issues/2558 sensitive_obs_filter, unsensitive_obs_filter = build_sensitive_unsensitive_filters() # Note: the used query is not important here, as it is only used to select the right Synthese model precise_perms_filter = precise_geom_query.build_permissions_filter( g.current_user, precise_permissions, ) blurring_perms_filter = precise_geom_query.build_permissions_filter( g.current_user, blurring_permissions, ) # Access precise geom for obs with precise perm and for unsensitive obs with blurring perm precise_geom_query = precise_geom_query.query.where( sa.or_( precise_perms_filter, sa.and_( blurring_perms_filter, unsensitive_obs_filter, ), ) ).limit(limit) # Access blurred geom for sensitive obs with blurring perms blurred_geom_query = blurred_geom_query.query.where( sa.and_( blurring_perms_filter, sensitive_obs_filter, ) ).limit(limit) return precise_geom_query.union(blurred_geom_query).cte("allowed_geom")
[docs] def build_synthese_obs_query(observations, allowed_geom_cte, limit): # Final observation query # orderby priority as explained in build_allowed_geom_cte() obs_query = ( sa.select(observations) .select_from( VSyntheseForWebApp.__table__.join( allowed_geom_cte, allowed_geom_cte.c.id_synthese == VSyntheseForWebApp.id_synthese ) ) .order_by(VSyntheseForWebApp.id_synthese, allowed_geom_cte.c.priority) .distinct(VSyntheseForWebApp.id_synthese) .limit(limit) ) return obs_query