from __future__ import annotations
from onegov.agency import AgencyApp
from onegov.agency.collections import ExtendedAgencyCollection
from onegov.agency.models import AgencyMembershipMoveWithinAgency
from onegov.agency.models import AgencyMembershipMoveWithinPerson
from onegov.agency.models import AgencyMove
from onegov.agency.models.ticket import AgencyMutationTicket
from onegov.agency.models.ticket import PersonMutationTicket
from onegov.core.security.rules import has_permission_logged_in
from onegov.people import Agency
from onegov.people import AgencyCollection
from onegov.people import AgencyMembership
from onegov.people import Person
from onegov.user import RoleMapping
from typing import TYPE_CHECKING
if TYPE_CHECKING:
from morepath.authentication import Identity
from morepath.authentication import NoIdentity
from sqlalchemy.orm import Session
[docs]
def get_current_role(
session: Session,
identity: Identity | NoIdentity
) -> str | None:
""" Returns the current role of the identity. Elevates the role from member
to editor if any group role mapping with editor role is present.
"""
if identity.userid:
if identity.role == 'member' and identity.groupid:
role = session.query(RoleMapping).filter(
RoleMapping.role == 'editor',
RoleMapping.group_id == identity.groupid
).first()
if role:
return 'editor'
return identity.role
return None
[docs]
def has_permission(
app: AgencyApp,
identity: Identity,
model: object,
permission: object
) -> bool:
""" Global permission with elevated roles. """
role = get_current_role(app.session(), identity)
assert role is not None
if permission in getattr(app.settings.roles, role):
return True
return False
[docs]
def has_model_permission(
app: AgencyApp,
identity: Identity,
model: object,
permission: object
) -> bool:
""" Specific model permission with elevated roles for this model. """
# Check if the identity itself has the role
if permission in getattr(app.settings.roles, identity.role):
return True
# Check the role mappings of the model
if identity.role == 'member' and identity.groupid:
if hasattr(model, 'role_mappings'):
role = model.role_mappings.filter(
RoleMapping.role == 'editor',
RoleMapping.group_id == identity.groupid
).first()
if role and permission in getattr(
app.settings.roles, 'editor', []
):
return True
# Check the role mappings of the parent
if parent := getattr(model, 'parent', None):
return has_model_permission(app, identity, parent, permission)
return False
@AgencyApp.permission_rule(model=object, permission=object)
[docs]
def has_permission_all(
app: AgencyApp,
identity: Identity,
model: object,
permission: object
) -> bool:
if has_permission_logged_in(app, identity, model, permission):
return True
return has_permission(app, identity, model, permission)
@AgencyApp.permission_rule(model=Agency, permission=object)
[docs]
def has_permission_agency(
app: AgencyApp,
identity: Identity,
model: Agency,
permission: object
) -> bool:
return has_model_permission(app, identity, model, permission)
@AgencyApp.permission_rule(model=AgencyMembership, permission=object)
[docs]
def has_permission_agency_membership(
app: AgencyApp,
identity: Identity,
model: AgencyMembership,
permission: object
) -> bool:
if has_model_permission(app, identity, model, permission):
return True
if model.agency:
agency = model.agency
return has_model_permission(app, identity, agency, permission)
return False
@AgencyApp.permission_rule(model=Person, permission=object)
[docs]
def has_permission_person(
app: AgencyApp,
identity: Identity,
model: Person,
permission: object
) -> bool:
if has_model_permission(app, identity, model, permission):
return True
memberships = model.memberships.all()
if not memberships:
if has_permission(app, identity, model, permission):
return True
for membership in memberships:
if membership.agency:
agency = membership.agency
if has_model_permission(app, identity, agency, permission):
return True
return False
@AgencyApp.permission_rule(model=ExtendedAgencyCollection, permission=object)
@AgencyApp.permission_rule(model=AgencyCollection, permission=object)
[docs]
def has_permission_agency_collection(
app: AgencyApp,
identity: Identity,
model: AgencyCollection,
permission: object
) -> bool:
return has_permission_logged_in(app, identity, model, permission)
@AgencyApp.permission_rule(model=AgencyMove, permission=object)
[docs]
def has_permission_agency_move(
app: AgencyApp,
identity: Identity,
model: AgencyMove,
permission: object
) -> bool:
if model.subject:
agency = model.subject
if not has_permission_agency(app, identity, agency, permission):
return False
if model.target:
agency = model.target
if not has_permission_agency(app, identity, agency, permission):
return False
return True
@AgencyApp.permission_rule(
model=AgencyMembershipMoveWithinAgency, permission=object
)
[docs]
def has_permission_agency_membership_move_within_agency(
app: AgencyApp,
identity: Identity,
model: AgencyMembershipMoveWithinAgency,
permission: object
) -> bool:
if model.subject and model.subject.agency:
agency = model.subject.agency
if not has_permission_agency(app, identity, agency, permission):
return False
if model.target and model.target.agency:
agency = model.target.agency
if not has_permission_agency(app, identity, agency, permission):
return False
return True
@AgencyApp.permission_rule(
model=AgencyMembershipMoveWithinPerson, permission=object
)
[docs]
def has_permission_agency_membership_move_within_person(
app: AgencyApp,
identity: Identity,
model: AgencyMembershipMoveWithinPerson,
permission: object
) -> bool:
if model.subject and model.subject.agency:
agency = model.subject.agency
if not has_permission_agency(app, identity, agency, permission):
return False
if model.target and model.target.agency:
agency = model.target.agency
if not has_permission_agency(app, identity, agency, permission):
return False
return True
@AgencyApp.permission_rule(model=AgencyMutationTicket, permission=object)
[docs]
def has_permission_agency_mutation_ticket(
app: AgencyApp,
identity: Identity,
model: AgencyMutationTicket,
permission: object
) -> bool:
if model.handler and model.handler.agency:
agency = model.handler.agency
if not has_permission_agency(app, identity, agency, permission):
return False
return True
@AgencyApp.permission_rule(model=PersonMutationTicket, permission=object)
[docs]
def has_permission_person_mutation_ticket(
app: AgencyApp,
identity: Identity,
model: PersonMutationTicket,
permission: object
) -> bool:
if model.handler and model.handler.person:
person = model.handler.person
if not has_permission_person(app, identity, person, permission):
return False
return True