2026-05-25 09:05:17 +01:00
|
|
|
"""
|
2026-05-28 13:22:24 +01:00
|
|
|
Business logic reusable functions related to IAM
|
2026-05-25 09:05:17 +01:00
|
|
|
|
2026-05-28 13:22:24 +01:00
|
|
|
Exports:
|
|
|
|
|
- service_key_dependency: bool: verifies request headers contain the correct api key for the service
|
2026-05-25 09:05:17 +01:00
|
|
|
"""
|
2026-06-08 15:31:37 +01:00
|
|
|
|
2026-05-25 09:05:17 +01:00
|
|
|
from typing import Annotated
|
2026-06-09 16:52:22 +01:00
|
|
|
from datetime import datetime, timedelta, timezone
|
2026-06-16 16:09:17 +01:00
|
|
|
from fastapi import Request, Depends
|
2026-06-17 10:49:58 +01:00
|
|
|
from sqlalchemy.orm import Session
|
2026-05-25 09:05:17 +01:00
|
|
|
|
|
|
|
|
from src.database import db_dependency
|
2026-06-11 14:58:05 +01:00
|
|
|
from src.exceptions import UnauthorizedException
|
2026-06-09 16:52:22 +01:00
|
|
|
from src.utils import send_email, generate_jwt
|
2026-06-15 11:26:22 +01:00
|
|
|
from src.iam.models import Group
|
2026-06-17 10:49:58 +01:00
|
|
|
from src.organisation.models import Organisation as Org
|
|
|
|
|
from src.user.models import User
|
|
|
|
|
from src.iam.models import Permission as Perm
|
2026-06-15 11:26:22 +01:00
|
|
|
|
2026-06-16 16:09:17 +01:00
|
|
|
from src.service.models import Service
|
|
|
|
|
from src.service.schemas import HasServiceName
|
2026-05-25 09:05:17 +01:00
|
|
|
|
|
|
|
|
|
2026-06-10 09:32:02 +01:00
|
|
|
def valid_service_key(
|
2026-06-16 16:09:17 +01:00
|
|
|
db: db_dependency, request: Request, request_model: HasServiceName
|
2026-06-10 09:32:02 +01:00
|
|
|
) -> bool:
|
|
|
|
|
rn = request_model.rn
|
2026-05-25 09:05:17 +01:00
|
|
|
api_key = request.headers.get("X-API-Key", None)
|
|
|
|
|
if not api_key:
|
2026-06-04 14:53:35 +01:00
|
|
|
raise UnauthorizedException("Missing API key")
|
2026-05-25 09:05:17 +01:00
|
|
|
service = rn.service
|
2026-06-08 15:31:37 +01:00
|
|
|
result = (
|
|
|
|
|
db.query(Service)
|
|
|
|
|
.filter(Service.name == service)
|
|
|
|
|
.filter(Service.api_key == api_key)
|
|
|
|
|
.first()
|
|
|
|
|
)
|
2026-05-25 09:05:17 +01:00
|
|
|
if result is None:
|
2026-06-04 14:53:35 +01:00
|
|
|
raise UnauthorizedException("Invalid API key")
|
2026-05-25 09:05:17 +01:00
|
|
|
|
|
|
|
|
return True
|
|
|
|
|
|
2026-06-08 15:31:37 +01:00
|
|
|
|
2026-05-25 09:05:17 +01:00
|
|
|
service_key_dependency = Annotated[bool, Depends(valid_service_key)]
|
2026-06-09 16:52:22 +01:00
|
|
|
|
|
|
|
|
|
|
|
|
|
async def send_user_group_invitation(
|
|
|
|
|
user_email: str, org_name: str, org_id: int, group_id: int, group_name: str
|
|
|
|
|
):
|
|
|
|
|
expiry_delta = timedelta(hours=24)
|
|
|
|
|
expiry = datetime.now(timezone.utc) + expiry_delta
|
|
|
|
|
claims = {
|
|
|
|
|
"email": user_email,
|
|
|
|
|
"org_id": org_id,
|
|
|
|
|
"group_id": group_id,
|
|
|
|
|
"group_name": group_name,
|
|
|
|
|
"exp": expiry,
|
|
|
|
|
"type": "group_invite",
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
token = await generate_jwt(claims)
|
|
|
|
|
subject = f"You have been invited to join a group of {org_name}"
|
|
|
|
|
body = f"You have been invited to join {group_name}.\nClick the link to accept.\nfrontend.capture/send/to/endpoint/{token}"
|
|
|
|
|
|
|
|
|
|
await send_email(
|
|
|
|
|
recipient=user_email,
|
|
|
|
|
subject=subject,
|
|
|
|
|
body=body,
|
|
|
|
|
)
|
2026-06-15 11:26:22 +01:00
|
|
|
|
|
|
|
|
|
2026-06-17 10:49:58 +01:00
|
|
|
async def create_group_and_assign_perms(
|
|
|
|
|
db: Session, org_model: Org, group_name: str, perm_list: list[int]
|
|
|
|
|
):
|
|
|
|
|
new_group = Group(name=group_name, org_id=org_model.id)
|
2026-06-15 11:26:22 +01:00
|
|
|
db.add(new_group)
|
|
|
|
|
db.flush()
|
|
|
|
|
|
2026-06-17 10:49:58 +01:00
|
|
|
for permission in perm_list:
|
|
|
|
|
perm_model = db.get(Perm, permission)
|
2026-06-15 11:26:22 +01:00
|
|
|
|
2026-06-17 10:49:58 +01:00
|
|
|
if perm_model is None:
|
|
|
|
|
continue
|
2026-06-15 11:26:22 +01:00
|
|
|
|
2026-06-17 10:49:58 +01:00
|
|
|
new_group.permission_rel.append(perm_model)
|
|
|
|
|
db.flush()
|
2026-06-15 11:26:22 +01:00
|
|
|
|
|
|
|
|
return new_group
|
|
|
|
|
|
|
|
|
|
|
2026-06-17 10:49:58 +01:00
|
|
|
async def assign_default_group(
|
|
|
|
|
db: db_dependency,
|
|
|
|
|
org_model: Org,
|
|
|
|
|
user_model: User,
|
|
|
|
|
group_name: str,
|
|
|
|
|
perm_list: list[int],
|
|
|
|
|
):
|
|
|
|
|
group_model = (
|
|
|
|
|
db.query(Group)
|
|
|
|
|
.filter(Group.org_id == org_model.id)
|
|
|
|
|
.filter(Group.name == group_name)
|
|
|
|
|
.first()
|
|
|
|
|
)
|
2026-06-15 11:26:22 +01:00
|
|
|
|
|
|
|
|
if group_model is None:
|
2026-06-17 10:49:58 +01:00
|
|
|
group_model = await create_group_and_assign_perms(
|
|
|
|
|
db=db, group_name=group_name, org_model=org_model, perm_list=perm_list
|
|
|
|
|
)
|
2026-06-15 11:26:22 +01:00
|
|
|
|
|
|
|
|
user_model.group_rel.append(group_model)
|
|
|
|
|
db.flush()
|