2025-12-20 22:18:14 +01:00
|
|
|
"""Invite routes for public check, user invites, and admin management."""
|
|
|
|
|
|
2025-12-25 18:42:46 +01:00
|
|
|
from fastapi import APIRouter, Depends, Query
|
|
|
|
|
from sqlalchemy import select
|
2025-12-20 22:18:14 +01:00
|
|
|
from sqlalchemy.ext.asyncio import AsyncSession
|
|
|
|
|
|
|
|
|
|
from auth import require_permission
|
|
|
|
|
from database import get_db
|
2025-12-25 00:59:57 +01:00
|
|
|
from mappers import InviteMapper
|
2025-12-25 18:42:46 +01:00
|
|
|
from models import Permission, User
|
2025-12-20 22:18:14 +01:00
|
|
|
from schemas import (
|
2025-12-21 21:54:26 +01:00
|
|
|
AdminUserResponse,
|
2025-12-20 22:18:14 +01:00
|
|
|
InviteCheckResponse,
|
|
|
|
|
InviteCreate,
|
|
|
|
|
InviteResponse,
|
|
|
|
|
PaginatedInviteRecords,
|
2025-12-21 21:54:26 +01:00
|
|
|
UserInviteResponse,
|
2025-12-20 22:18:14 +01:00
|
|
|
)
|
2025-12-25 18:42:46 +01:00
|
|
|
from services.invite import InviteService
|
2025-12-20 22:18:14 +01:00
|
|
|
|
2025-12-20 22:38:39 +01:00
|
|
|
router = APIRouter(prefix="/api/invites", tags=["invites"])
|
|
|
|
|
admin_router = APIRouter(prefix="/api/admin", tags=["admin"])
|
2025-12-20 22:18:14 +01:00
|
|
|
|
|
|
|
|
|
2025-12-20 22:38:39 +01:00
|
|
|
@router.get("/{identifier}/check", response_model=InviteCheckResponse)
|
2025-12-20 22:18:14 +01:00
|
|
|
async def check_invite(
|
|
|
|
|
identifier: str,
|
|
|
|
|
db: AsyncSession = Depends(get_db),
|
|
|
|
|
) -> InviteCheckResponse:
|
|
|
|
|
"""Check if an invite is valid and can be used for signup."""
|
2025-12-25 18:42:46 +01:00
|
|
|
service = InviteService(db)
|
|
|
|
|
return await service.check_invite_validity(identifier)
|
2025-12-20 22:18:14 +01:00
|
|
|
|
|
|
|
|
|
2025-12-20 22:38:39 +01:00
|
|
|
@router.get("", response_model=list[UserInviteResponse])
|
2025-12-20 22:18:14 +01:00
|
|
|
async def get_my_invites(
|
|
|
|
|
db: AsyncSession = Depends(get_db),
|
|
|
|
|
current_user: User = Depends(require_permission(Permission.VIEW_OWN_INVITES)),
|
|
|
|
|
) -> list[UserInviteResponse]:
|
|
|
|
|
"""Get all invites owned by the current user."""
|
2025-12-25 18:42:46 +01:00
|
|
|
service = InviteService(db)
|
|
|
|
|
invites = await service.get_user_invites(current_user.id)
|
2025-12-20 22:18:14 +01:00
|
|
|
|
|
|
|
|
return [
|
|
|
|
|
UserInviteResponse(
|
|
|
|
|
id=invite.id,
|
|
|
|
|
identifier=invite.identifier,
|
|
|
|
|
status=invite.status.value,
|
|
|
|
|
used_by_email=invite.used_by.email if invite.used_by else None,
|
|
|
|
|
created_at=invite.created_at,
|
|
|
|
|
spent_at=invite.spent_at,
|
|
|
|
|
)
|
|
|
|
|
for invite in invites
|
|
|
|
|
]
|
|
|
|
|
|
|
|
|
|
|
2025-12-20 22:38:39 +01:00
|
|
|
@admin_router.get("/users", response_model=list[AdminUserResponse])
|
2025-12-20 22:18:14 +01:00
|
|
|
async def list_users_for_admin(
|
|
|
|
|
db: AsyncSession = Depends(get_db),
|
|
|
|
|
_current_user: User = Depends(require_permission(Permission.MANAGE_INVITES)),
|
|
|
|
|
) -> list[AdminUserResponse]:
|
|
|
|
|
"""List all users for admin dropdowns (invite creation, etc.)."""
|
2025-12-25 18:42:46 +01:00
|
|
|
# Note: UserRepository doesn't have list_all yet
|
|
|
|
|
# For now, keeping direct query for this specific use case
|
2025-12-20 22:18:14 +01:00
|
|
|
result = await db.execute(select(User.id, User.email).order_by(User.email))
|
|
|
|
|
users = result.all()
|
|
|
|
|
return [AdminUserResponse(id=u.id, email=u.email) for u in users]
|
|
|
|
|
|
|
|
|
|
|
2025-12-20 22:38:39 +01:00
|
|
|
@admin_router.post("/invites", response_model=InviteResponse)
|
2025-12-20 22:18:14 +01:00
|
|
|
async def create_invite(
|
|
|
|
|
data: InviteCreate,
|
|
|
|
|
db: AsyncSession = Depends(get_db),
|
|
|
|
|
_current_user: User = Depends(require_permission(Permission.MANAGE_INVITES)),
|
|
|
|
|
) -> InviteResponse:
|
|
|
|
|
"""Create a new invite for a specified godfather user."""
|
2025-12-25 18:42:46 +01:00
|
|
|
service = InviteService(db)
|
|
|
|
|
invite = await service.create_invite(data.godfather_id)
|
2025-12-25 00:59:57 +01:00
|
|
|
return InviteMapper.to_response(invite)
|
2025-12-20 22:18:14 +01:00
|
|
|
|
|
|
|
|
|
2025-12-20 22:38:39 +01:00
|
|
|
@admin_router.get("/invites", response_model=PaginatedInviteRecords)
|
2025-12-20 22:18:14 +01:00
|
|
|
async def list_all_invites(
|
|
|
|
|
page: int = Query(1, ge=1),
|
|
|
|
|
per_page: int = Query(10, ge=1, le=100),
|
2025-12-21 21:54:26 +01:00
|
|
|
status_filter: str | None = Query(
|
|
|
|
|
None, alias="status", description="Filter by status: ready, spent, revoked"
|
|
|
|
|
),
|
2025-12-20 22:18:14 +01:00
|
|
|
godfather_id: int | None = Query(None, description="Filter by godfather user ID"),
|
|
|
|
|
db: AsyncSession = Depends(get_db),
|
|
|
|
|
_current_user: User = Depends(require_permission(Permission.MANAGE_INVITES)),
|
|
|
|
|
) -> PaginatedInviteRecords:
|
|
|
|
|
"""List all invites with optional filtering and pagination."""
|
2025-12-25 18:42:46 +01:00
|
|
|
service = InviteService(db)
|
|
|
|
|
return await service.list_invites(
|
|
|
|
|
page=page,
|
|
|
|
|
per_page=per_page,
|
|
|
|
|
status_filter=status_filter,
|
|
|
|
|
godfather_id=godfather_id,
|
|
|
|
|
)
|
2025-12-20 22:18:14 +01:00
|
|
|
|
|
|
|
|
|
2025-12-20 22:38:39 +01:00
|
|
|
@admin_router.post("/invites/{invite_id}/revoke", response_model=InviteResponse)
|
2025-12-20 22:18:14 +01:00
|
|
|
async def revoke_invite(
|
|
|
|
|
invite_id: int,
|
|
|
|
|
db: AsyncSession = Depends(get_db),
|
|
|
|
|
_current_user: User = Depends(require_permission(Permission.MANAGE_INVITES)),
|
|
|
|
|
) -> InviteResponse:
|
|
|
|
|
"""Revoke an invite. Only READY invites can be revoked."""
|
2025-12-25 18:42:46 +01:00
|
|
|
service = InviteService(db)
|
|
|
|
|
invite = await service.revoke_invite(invite_id)
|
2025-12-25 00:59:57 +01:00
|
|
|
return InviteMapper.to_response(invite)
|
2025-12-22 09:10:26 +01:00
|
|
|
|
|
|
|
|
|
|
|
|
|
# All routers from this module for easy registration
|
|
|
|
|
routers = [router, admin_router]
|