elmeg-demo/backend/routers/attendance.py
fullsizemalt bc804a666b feat: Gamification sprint complete
XP System:
- XP now awarded for attendance (+25), ratings (+10), reviews (+50)
- First-time bonuses for first rating (+25) and first review (+50)
- Streak bonuses (+10 per day, capped at 7x)
- Badge awards automatically grant XP

User Titles & Flair System (Tracker-style):
- Level-based free titles: Rookie → Immortal
- Purchasable titles with XP: Jam Connoisseur, Setlist Savant, etc.
- Username colors purchasable with XP (6 colors + Rainbow)
- Emoji flairs purchasable with XP
- Early adopter perks: exclusive titles, colors, 10% XP bonus

New Fields on User:
- custom_title, title_color, flair
- is_early_adopter, is_supporter
- joined_at

Shop API Endpoints:
- GET /gamification/shop/titles
- POST /gamification/shop/titles/purchase
- GET/POST for colors and flairs
- GET /gamification/user/{id}/display
- GET /gamification/early-adopter-perks

Frontend:
- XP Leaderboard added to home page
- LevelProgressCard shows on profile
2025-12-21 19:21:20 -08:00

84 lines
2.7 KiB
Python

from typing import List, Optional
from fastapi import APIRouter, Depends, HTTPException, Query
from sqlmodel import Session, select
from database import get_session
from models import Attendance, User, Show
from schemas import AttendanceCreate, AttendanceRead
from auth import get_current_user
from services.gamification import award_xp, check_and_award_badges, update_streak, XP_REWARDS
router = APIRouter(prefix="/attendance", tags=["attendance"])
@router.post("/", response_model=AttendanceRead)
def mark_attendance(
attendance: AttendanceCreate,
session: Session = Depends(get_session),
current_user: User = Depends(get_current_user)
):
# Check if already attended
existing = session.exec(
select(Attendance)
.where(Attendance.user_id == current_user.id)
.where(Attendance.show_id == attendance.show_id)
).first()
if existing:
# Update notes if provided, or just return existing
if attendance.notes:
existing.notes = attendance.notes
session.add(existing)
session.commit()
session.refresh(existing)
return existing
db_attendance = Attendance(**attendance.model_dump(), user_id=current_user.id)
session.add(db_attendance)
# Award XP for marking attendance
new_xp, level_up = award_xp(session, current_user, XP_REWARDS["attendance_add"], "attendance")
update_streak(session, current_user)
new_badges = check_and_award_badges(session, current_user)
session.commit()
session.refresh(db_attendance)
return db_attendance
@router.delete("/{show_id}")
def remove_attendance(
show_id: int,
session: Session = Depends(get_session),
current_user: User = Depends(get_current_user)
):
attendance = session.exec(
select(Attendance)
.where(Attendance.user_id == current_user.id)
.where(Attendance.show_id == show_id)
).first()
if not attendance:
raise HTTPException(status_code=404, detail="Attendance not found")
session.delete(attendance)
session.commit()
return {"ok": True}
@router.get("/me", response_model=List[AttendanceRead])
def get_my_attendance(
session: Session = Depends(get_session),
current_user: User = Depends(get_current_user)
):
return session.exec(select(Attendance).where(Attendance.user_id == current_user.id)).all()
@router.get("/show/{show_id}", response_model=List[AttendanceRead])
def get_show_attendance(
show_id: int,
session: Session = Depends(get_session),
offset: int = 0,
limit: int = 100
):
return session.exec(
select(Attendance)
.where(Attendance.show_id == show_id)
.offset(offset)
.limit(limit)
).all()