Code source de geonature.core.gn_permissions.tools

import logging

from itertools import groupby, permutations

import sqlalchemy as sa
from sqlalchemy.orm import joinedload
from flask import has_request_context, g

from geonature.core.gn_commons.models import TModules
from geonature.core.gn_permissions.models import (
    PermAction,
    PermObject,
    PermScope,
    Permission,
)
from geonature.utils.env import db

from pypnusershub.db.models import User

[docs] log = logging.getLogger()
[docs] def _get_user_permissions(id_role): return db.session.scalars( sa.select(Permission) .options( joinedload(Permission.module), joinedload(Permission.object), joinedload(Permission.action), ) .where( sa.or_( # direct permissions Permission.id_role == id_role, # permissions through group # FIXME : provoke a cartesian product warning (but ) Permission.role.has(User.members.any(User.id_role == id_role)), ), ) .order_by(Permission.id_module, Permission.id_object, Permission.id_action) .distinct( Permission.id_module, Permission.id_object, Permission.id_action, *Permission.filters_fields.values(), ) ).all()
[docs] def get_user_permissions(id_role=None): if id_role is None: id_role = g.current_user.id_role if has_request_context(): if id_role not in g._permissions_by_user: g._permissions_by_user[id_role] = _get_user_permissions(id_role) return g._permissions_by_user[id_role] else: return _get_user_permissions(id_role)
[docs] def _get_permissions(id_role, module_code, object_code, action_code): permissions = { p for p in get_user_permissions(id_role) if p.module.module_code == module_code and p.object.code_object == object_code and p.action.code_action == action_code } # Remove all permissions supersed by another permission for p1, p2 in permutations(permissions, 2): if p1 in permissions and p1 <= p2: permissions.remove(p1) return permissions
[docs] def get_permissions(action_code, id_role=None, module_code=None, object_code=None): """ This function returns a set of all the permissions that match (action_code, id_role, module_code, object_code). If module_code is None, it is set as the code of the current module or as "GEONATURE" if no current module found. If object_code is None, it is set as the code of the current object or as "ALL" if no current object found. :returns : the list of permissions that match, and an empty list if no match """ if module_code is None: if hasattr(g, "current_module"): module_code = g.current_module.module_code else: module_code = "GEONATURE" if object_code is None: if hasattr(g, "current_object"): object_code = g.current_object.code_object else: object_code = "ALL" ident = (id_role, module_code, object_code, action_code) if has_request_context(): if ident not in g._permissions: g._permissions[ident] = _get_permissions(*ident) return g._permissions[ident] else: return _get_permissions(*ident)
[docs] def get_scope(action_code, id_role=None, module_code=None, object_code=None, bypass_warning=False): """ This function gets the final scope permission. It takes the maximum for all the permissions that match (action_code, id_role, module_code, object_code) and with of a "SCOPE" filter type. :returns : (int) The scope computed for specified arguments """ permissions = get_permissions(action_code, id_role, module_code, object_code) max_scope = 0 for permission in permissions: if permission.has_other_filters_than("SCOPE") and not bypass_warning: log.warning( f"""WARNING : You are trying to get scope permission for a module ({module_code}) which implements other permissions type. Please use get_permission instead """ ) if permission.scope_value is None: max_scope = 3 else: max_scope = max(max_scope, permission.scope_value) return max_scope
[docs] def get_scopes_by_action(id_role=None, module_code=None, object_code=None): """ This function gets the scopes permissions for each one of the 6 actions in "CRUVED", that match (id_role, module_code, object_code) :returns : (dict) A dict of the scope for each one of the 6 actions (the char in "CRUVED") """ return { action_code: get_scope(action_code, id_role, module_code, object_code) for action_code in "CRUVED" }
[docs] def has_any_permissions(action_code, id_role=None, module_code=None, object_code=None) -> bool: """ This function return the scope for an action, a module and an object as a Boolean Use for frontend """ permissions = get_permissions(action_code, id_role, module_code, object_code) return len(permissions) > 0
[docs] def has_any_permissions_by_action(id_role=None, module_code=None, object_code=None): """ This function gets the scopes permissions for each one of the 6 actions in "CRUVED", that match (id_role, module_code, object_code) :returns : (dict) A dict of the boolean for each one of the 6 actions (the char in "CRUVED") """ return { action_code: has_any_permissions(action_code, id_role, module_code, object_code) for action_code in "CRUVED" }