Code source de geonature.core.gn_permissions.admin

from flask import url_for, has_app_context, request
from flask_admin.contrib.sqla import ModelView
from flask_admin.contrib.sqla.filters import FilterEqual
import sqlalchemy as sa
from flask_admin.contrib.sqla.tools import get_primary_key
from flask_admin.contrib.sqla.fields import QuerySelectField
from flask_admin.contrib.sqla.ajax import QueryAjaxModelLoader
from flask_admin.form.widgets import Select2Widget
from markupsafe import Markup
from sqlalchemy.orm import contains_eager, joinedload
from sqlalchemy import select
from markupsafe import Markup

from geonature.utils.env import db
from geonature.utils.config import config
from geonature.core.admin.admin import admin
from geonature.core.admin.utils import CruvedProtectedMixin, DynamicOptionsMixin
from geonature.core.gn_permissions.models import (
    PermObject,
    PermAction,
    PermScope,
    Permission,
    PermissionAvailable,
    PermFilter,
)
from geonature.core.gn_permissions.tools import get_permissions
from geonature.core.gn_commons.models.base import TModules

from pypnusershub.db.models import User


### Filters


[docs] class RoleFilter(DynamicOptionsMixin, FilterEqual):
[docs] def get_dynamic_options(self, view): if has_app_context(): yield from [(u.id_role, u.nom_complet) for u in db.session.scalars(select(User)).all()]
[docs] class ModuleFilter(DynamicOptionsMixin, FilterEqual):
[docs] def get_dynamic_options(self, view): if has_app_context(): modules = db.session.scalars(select(TModules).order_by(TModules.module_code)).all() yield from [(module.id_module, module.module_code) for module in modules]
[docs] class ObjectFilter(DynamicOptionsMixin, FilterEqual):
[docs] def get_dynamic_options(self, view): if has_app_context(): objects = db.session.scalars(select(PermObject)).all() yield from [(object.id_object, object.code_object) for object in objects]
[docs] class ActionFilter(DynamicOptionsMixin, FilterEqual):
[docs] def get_dynamic_options(self, view): if has_app_context(): actions = db.session.scalars(select(PermAction)).all() yield from [(action.id_action, action.code_action) for action in actions]
[docs] class ScopeFilter(DynamicOptionsMixin, FilterEqual):
[docs] def apply(self, query, value, alias=None): column = self.get_column(alias) if value: return query.where(column == value) else: return query.where(column.is_(None))
[docs] def get_dynamic_options(self, view): if has_app_context(): yield (None, "Sans restriction") scopes = db.session.scalars(select(PermScope)).all() yield from [(scope.value, scope.label) for scope in scopes]
### Formatters
[docs] def filters_formatter(v, c, m, p): filters = [] if m.scope: filters.append(m.scope.label) if m.sensitivity_filter: filters.append("Données non sensibles") return Markup("<ul>" + "".join(["<li>{}</li>".format(f) for f in filters]) + "</ul>")
[docs] def modules_formatter(view, context, model, name): modules = [ '<a href="{}">{}</a>'.format( url_for("tmodules.details_view", id=module.id_module), Markup.escape(module.module_code), ) for module in model.modules ] return Markup(", ".join(modules))
[docs] def groups_formatter(view, context, model, name): groups = [ '<a href="{}">{}</a>'.format( url_for("permissions/group.details_view", id=group.id_role), Markup.escape(group) ) for group in model.groups ] return Markup(", ".join(groups))
[docs] def role_formatter(view, context, model, name): role = model.role if role.groupe: url = url_for("permissions/group.details_view", id=role.id_role) nom = "<b>{}</b>".format(Markup.escape(role.nom_role)) else: url = url_for("permissions/user.details_view", id=role.id_role) nom = Markup.escape(role.nom_complet) return Markup('<a href="{}">{}</a>'.format(url, nom))
[docs] def permissions_formatter(view, context, model, name): available_permissions = db.session.scalars(PermissionAvailable.nice_order()).unique().all() o = "<table class='table'>" columns = ["Module", "Object", "Action", "Label"] if model.groupe: return_url = url_for("permissions/group.details_view", id=model.id_role) columns += ["Permissions"] else: return_url = url_for("permissions/user.details_view", id=model.id_role) columns += ["Permissions personnelles", "Permissions effectives"] o += "<thead><tr>" + "".join([f"<th>{col}</th>" for col in columns]) + "</tr></thead>" o += "<tbody>" for ap in available_permissions: own_permissions = list( filter( lambda p: p.module == ap.module and p.object == ap.object and p.action == ap.action, model.permissions, ) ) permissions = [(own_permissions, True)] if not model.groupe: effective_permissions = list( get_permissions( id_role=model.id_role, module_code=ap.module.module_code, object_code=ap.object.code_object, action_code=ap.action.code_action, ) ) permissions.append((effective_permissions, False)) o += ( "<tr>" if own_permissions or effective_permissions else "<tr class='text-muted alert alert-danger'>" ) else: o += "<tr>" if own_permissions else "<tr class='text-muted alert alert-danger'>" o += "".join( [ f"<td>{col}</td>" for col in [ ap.module.module_code, ap.object.code_object, ap.action.code_action, ap.label, ] ] ) for perms, managable in permissions: o += "<td>" if perms: if len(perms) > 1: o += f"{len(perms)} permissions :" o += '<table class="table table-bordered table-sm" style="border-collapse: separate; border-spacing:0 8px;">' for perm in perms: flts = perm.filters o += "<tr>" if not flts: o += '<td class="table-success">' else: o += '<td class="table-info">' o += """<div class="row"><div class="col">""" if not flts: o += """<i class="fa fa-check" aria-hidden="true"></i>""" else: o += """<ul class="list-group">""" for flt_name in perm.availability.filters: flt_field = Permission.filters_fields[flt_name] flt = PermFilter(flt_name, getattr(perm, flt_field.name)) o += f"""<li class="list-group-item">{flt}</li>""" o += "</ul>" o += """</div></div>""" if managable: o += """<div class="row"><div class="col text-right">""" edit_url = url_for( "permissions/permission.edit_view", id=perm.id_permission, url=return_url, ) delete_url = url_for( "permissions/permission.delete_view", id=perm.id_permission, url=return_url, ) o += f"""<form method="post" action="{delete_url}">""" if len(ap.filters) > 0: o += ( f"""<a class="btn btn-primary btn-sm" href="{edit_url}">""" """<i class="fa fa-pencil" aria-hidden="true"></i>""" """</a>""" ) o += ( """<button class="btn btn-danger btn-sm" onclick="return faHelpers.safeConfirm('Supprimer cette permission ?');">""" """<i class="fa fa-trash" aria-hidden="true"></i>""" "</button>" "</form>" ) o += """</div></div>""" o += "</td></tr>" o += "</table>" if managable and (not perms or len(ap.filters) > 1): add_url = url_for( "permissions/permission.create_view", id_role=model.id_role, module_code=ap.module.module_code, code_object=ap.object.code_object, code_action=ap.action.code_action, url=return_url, ) o += ( f"""<a class="btn btn-success btn-sm float-right" href="{add_url}">""" """<i class="fa fa-plus" aria-hidden="true"></i>""" """</a>""" ) o += "</td>" o += "</tr>" o += "</tbody>" o += "</table>" return Markup(o)
[docs] def permissions_count_formatter(view, context, model, name): url = url_for("permissions/permission.index_view", flt1_rle_equals=model.id_role) return Markup(f'<a href="{url}">{len(model.permissions)}</a>')
### Widgets
[docs] class OptionSelect2Widget(Select2Widget): @classmethod
[docs] def render_option(cls, value, label, options): return super().render_option(value, label, options.pop("selected"), **options)
### Fields
[docs] class OptionQuerySelectField(QuerySelectField): """ Overrides the QuerySelectField class from flask admin to allow other attributes on a select option. options_additional_values is added in form_args, it is a list of strings, each element is the name of the attribute in the model which will be added on the option """
[docs] widget = OptionSelect2Widget()
def __init__(self, *args, **kwargs):
[docs] self.options_additional_values = kwargs.pop("options_additional_values")
super().__init__(*args, **kwargs)
[docs] def iter_choices(self): if self.allow_blank: yield ("__None", self.blank_text, {"selected": self.data is None}) for pk, obj in self._get_object_list(): options = {k: getattr(obj, k) for k in self.options_additional_values} options["selected"] = obj == self.data yield (pk, self.get_label(obj), options)
### ModelLoader
[docs] class UserAjaxModelLoader(QueryAjaxModelLoader):
[docs] def format(self, user): if not user: return None def format_availability(availability): return ":".join( [str(getattr(availability, attr)) for attr in get_primary_key(PermissionAvailable)] ) def filter_availability(availability): filters_count = sum( [ getattr(availability, field.name) for field in PermissionAvailable.filters_fields.values() ] ) return filters_count < 2 availabilities = {p.availability for p in user.permissions if p.availability} excluded_availabilities = filter(filter_availability, availabilities) excluded_availabilities = map(format_availability, excluded_availabilities) return super().format(user) + (list(excluded_availabilities),)
[docs] def get_query(self): return ( super() .get_query() .options(joinedload(User.permissions).joinedload(Permission.availability)) .order_by(User.groupe.desc(), User.nom_role) )
### ModelViews
[docs] class ObjectAdmin(CruvedProtectedMixin, ModelView):
[docs] module_code = "ADMIN"
[docs] object_code = "PERMISSIONS"
[docs] can_create = False
[docs] can_edit = False
[docs] can_delete = False
[docs] column_list = ("code_object", "description_object", "modules")
[docs] column_labels = { "code_object": "Code", "description_object": "Description", }
[docs] column_default_sort = "id_object"
[docs] column_formatters = { "modules": modules_formatter, }
# self.allow_blank = (True,) # self.blank_test = "lalala"
[docs] class PermissionAdmin(CruvedProtectedMixin, ModelView):
[docs] module_code = "ADMIN"
[docs] object_code = "PERMISSIONS"
[docs] column_list = ("role", "module", "object", "action", "label", "filters")
[docs] column_labels = { "role": "Rôle", "filters": "Restriction(s)", "object": "Objet", "role.identifiant": "identifiant du rôle", "role.nom_complet": "nom du rôle", "availability": "Permission", "scope": "Filtre sur l'appartenance des données", "sensitivity_filter": ( "Flouter" if config["SYNTHESE"]["BLUR_SENSITIVE_OBSERVATIONS"] else "Exclure" ) + " les données sensibles", }
[docs] column_searchable_list = ("role.identifiant", "role.nom_complet")
[docs] column_formatters = { "role": role_formatter, "module": lambda v, c, m, p: m.module.module_code, "object": lambda v, c, m, p: m.object.code_object, "label": lambda v, c, m, p: m.availability.label if m.availability else None, "filters": filters_formatter, }
[docs] column_filters = ( RoleFilter(column=Permission.id_role, name="Rôle"), ModuleFilter(column=Permission.id_module, name="Module"), ObjectFilter(column=Permission.id_object, name="Objet"), ActionFilter(column=Permission.id_action, name="Action"), ScopeFilter(column=Permission.scope_value, name="Scope"), )
[docs] named_filter_urls = True
[docs] column_sortable_list = ( ("role", "role.nom_complet"), ("module", "module.module_code"), ("object", "object.code_object"), ("action", "action.code_action"), )
[docs] column_default_sort = [ ("role.nom_complet", False), ("module.module_code", False), ("object.code_object", False), ("id_action", False), ]
[docs] form_columns = ("role", "availability", "scope", "sensitivity_filter")
[docs] form_overrides = dict( availability=OptionQuerySelectField, )
[docs] form_args = dict( availability=dict( query_factory=lambda: PermissionAvailable.nice_order(), options_additional_values=["sensitivity_filter", "scope_filter"], ), )
[docs] create_template = "admin/hide_select2_options_create.html"
[docs] edit_template = "admin/hide_select2_options_edit.html"
[docs] form_ajax_refs = { "role": UserAjaxModelLoader( "role", db.session, User, fields=( "identifiant", "nom_role", "prenom_role", ), placeholder="Veuillez sélectionner un utilisateur ou un groupe", minimum_input_length=0, ), }
[docs] def render(self, template, **kwargs): self.extra_js = [url_for("static", filename="js/hide_unnecessary_filters.js")] return super().render(template, **kwargs)
[docs] def create_form(self): form = super().create_form() if request.method == "GET": # Set default values from request.args if "id_role" in request.args: form.role.data = db.session.get(User, request.args.get("id_role", type=int)) if {"module_code", "code_object", "code_action"}.issubset(request.args.keys()): form.availability.data = db.session.execute( select(PermissionAvailable) .join(PermissionAvailable.module) .join(PermissionAvailable.object) .join(PermissionAvailable.action) .where( TModules.module_code == request.args.get("module_code"), PermObject.code_object == request.args.get("code_object"), PermAction.code_action == request.args.get("code_action"), ) ).scalar_one_or_none() return form
[docs] class PermissionAvailableAdmin(CruvedProtectedMixin, ModelView):
[docs] module_code = "ADMIN"
[docs] object_code = "PERMISSIONS"
[docs] can_create = False
[docs] can_delete = False
[docs] can_export = False
[docs] column_labels = { "scope": "Portée", "object": "Objet", "scope_filter": "Filtre appartenance", "sensitivity_filter": "Filtre sensibilité", }
[docs] column_formatters = { "module": lambda v, c, m, p: m.module.module_code, "object": lambda v, c, m, p: m.object.code_object, }
[docs] column_sortable_list = ( ("module", "module.module_code"), ("object", "object.code_object"), ("action", "action.code_action"), )
[docs] column_filters = (ModuleFilter(column=PermissionAvailable.id_module, name="Module"),)
[docs] column_default_sort = [ ("module.module_code", False), ("object.code_object", False), ("id_action", False), ]
[docs] form_columns = ("scope_filter", "sensitivity_filter")
[docs] class RolePermAdmin(CruvedProtectedMixin, ModelView):
[docs] module_code = "ADMIN"
[docs] object_code = "PERMISSIONS"
[docs] can_create = False
[docs] can_edit = False
[docs] can_delete = False
[docs] can_export = False
[docs] can_view_details = True
[docs] details_template = "role_or_group_detail.html"
[docs] column_labels = { "nom_role": "Nom", "prenom_role": "Prénom", "groups": "Groupes", "permissions": "Permissions", "permissions_count": "Nombre de permissions", }
[docs] column_searchable_list = ("identifiant", "nom_complet")
[docs] column_formatters = { "groups": groups_formatter, "permissions_count": permissions_count_formatter, }
[docs] column_formatters_detail = { "groups": groups_formatter, "permissions": permissions_formatter, "permissions_count": permissions_count_formatter, }
[docs] class GroupPermAdmin(RolePermAdmin):
[docs] column_list = ( "nom_role", "permissions_count", )
[docs] column_details_list = ("nom_role", "permissions_count", "permissions")
[docs] def get_query(self): # TODO : change to sqla2.0 query when flask admin update to sqla2 return db.session.query(User).filter_by(groupe=True).where(User.filter_by_app())
[docs] def get_count_query(self): # TODO : change to sqla2.0 query when flask admin update to sqla2 return ( db.session.query(sa.func.count("*")) .select_from(User) .where(User.groupe == True) .where(User.filter_by_app()) )
[docs] class UserPermAdmin(RolePermAdmin):
[docs] column_list = ( "identifiant", "nom_role", "prenom_role", "groups", "permissions_count", )
[docs] column_labels = { **RolePermAdmin.column_labels, "permissions_count": "Nombre de permissions non héritées", }
[docs] column_details_list = ( "identifiant", "nom_role", "prenom_role", "groups", "permissions_count", "permissions", )
[docs] def get_query(self): # TODO : change to sqla2.0 query when flask admin update to sqla2 return db.session.query(User).filter_by(groupe=False).where(User.filter_by_app())
[docs] def get_count_query(self): # TODO : change to sqla2.0 query when flask admin update to sqla2 return ( db.session.query(sa.func.count("*")) .select_from(User) .where(User.groupe == False) .where(User.filter_by_app()) )
admin.add_view( GroupPermAdmin( User, db.session, name="Par groupes", category="Permissions", endpoint="permissions/group", ) ) admin.add_view( UserPermAdmin( User, db.session, name="Par utilisateurs", category="Permissions", endpoint="permissions/user", ) ) # Retirer pour plus de lisibilité de l'interface des permissions # admin.add_view( # ObjectAdmin( # PermObject, # db.session, # name="Objets", # category="Permissions", # endpoint="permissions/object", # ) # ) admin.add_view( PermissionAdmin( Permission, db.session, name="Permissions", category="Permissions", endpoint="permissions/permission", ) ) # Retirer pour plus de lisibilité de l'interface des permissions # admin.add_view( # PermissionAvailableAdmin( # PermissionAvailable, # db.session, # name="Permissions disponibles", # category="Permissions", # endpoint="permissions/availablepermission", # ) # )