oidc-fastapi-test/src/oidc-test/auth_utils.py

58 lines
1.7 KiB
Python

from typing import Union
from functools import wraps
from fastapi import HTTPException, Request
from .models import User
def get_current_user(request: Request) -> User:
auth_data = request.session.get("user")
if auth_data is None:
raise HTTPException(401, "Not authorized")
return User(**auth_data)
def get_current_user_or_none(request: Request) -> User | None:
try:
return get_current_user(request)
except HTTPException:
return None
def hasrole(
required_roles: Union[str, list[str]] = [],
roles_key: str = "roles",
realm: str | None = "realm_access", # Keycloak standard for realm defined roles
):
required_roles_set: set[str]
if isinstance(required_roles, str):
required_roles_set = set([required_roles])
else:
required_roles_set = set(required_roles)
def decorator(func):
@wraps(func)
async def wrapper(request=None, *args, **kwargs):
if request is None:
raise HTTPException(
500,
"Functions decorated with hasrole must have a request:Request argument",
)
if "user" not in request.session:
raise HTTPException(401, "Not authorized")
user = request.session["user"]
try:
if realm in user:
roles = user[realm][roles_key]
else:
roles = user[roles_key]
except KeyError:
raise HTTPException(401, "Not authorized")
if not any(required_roles_set.intersection(roles)):
raise HTTPException(401, "Not authorized")
return await func(request, *args, **kwargs)
return wrapper
return decorator