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"
}