178 lines
5.9 KiB
Python
178 lines
5.9 KiB
Python
from __future__ import annotations
|
|
|
|
import click
|
|
|
|
from .extensions import db
|
|
from .models import BadgeDefinition, User
|
|
from .services.monthly import archive_months_missing_up_to_previous
|
|
from .services.notifications import send_due_notifications, send_monthly_winner_notifications
|
|
|
|
|
|
DEFAULT_BADGES = [
|
|
{
|
|
"key": "first_wipe",
|
|
"name": "Erster Wisch",
|
|
"description": "Erledige deine erste Aufgabe.",
|
|
"icon_name": "sparkles",
|
|
"trigger_type": "first_task_completed",
|
|
"threshold": 1,
|
|
"bonus_points": 5,
|
|
},
|
|
{
|
|
"key": "warmed_up",
|
|
"name": "Warmgelaufen",
|
|
"description": "Erledige 10 Aufgaben insgesamt.",
|
|
"icon_name": "fire",
|
|
"trigger_type": "total_tasks_completed",
|
|
"threshold": 10,
|
|
"bonus_points": 10,
|
|
},
|
|
{
|
|
"key": "punctuality_pro",
|
|
"name": "Pünktlichkeitsprofi",
|
|
"description": "Erledige 10 Aufgaben pünktlich.",
|
|
"icon_name": "check-double",
|
|
"trigger_type": "on_time_tasks_completed",
|
|
"threshold": 10,
|
|
"bonus_points": 15,
|
|
},
|
|
{
|
|
"key": "early_bird",
|
|
"name": "Der frühe Vogel",
|
|
"description": "Erledige eine Aufgabe mindestens 1 Tag vor der Deadline.",
|
|
"icon_name": "calendar-day",
|
|
"trigger_type": "early_tasks_completed",
|
|
"threshold": 1,
|
|
"bonus_points": 8,
|
|
},
|
|
{
|
|
"key": "early_starter",
|
|
"name": "Frühstarter",
|
|
"description": "Erledige 5 Aufgaben mindestens 1 Tag vor der Deadline.",
|
|
"icon_name": "rocket-launch",
|
|
"trigger_type": "early_tasks_completed",
|
|
"threshold": 5,
|
|
"bonus_points": 15,
|
|
},
|
|
{
|
|
"key": "weekly_flow",
|
|
"name": "Wochenflow",
|
|
"description": "Erledige 7 Tage in Folge mindestens eine Aufgabe.",
|
|
"icon_name": "flag-checkered",
|
|
"trigger_type": "streak_days",
|
|
"threshold": 7,
|
|
"bonus_points": 20,
|
|
},
|
|
{
|
|
"key": "monthly_champion",
|
|
"name": "Monatssieger",
|
|
"description": "Gewinne einen Monat mit den meisten Punkten.",
|
|
"icon_name": "trophy",
|
|
"trigger_type": "monthly_win_count",
|
|
"threshold": 1,
|
|
"bonus_points": 25,
|
|
},
|
|
{
|
|
"key": "title_defender",
|
|
"name": "Titelverteidiger",
|
|
"description": "Gewinne 2 Monate in Folge.",
|
|
"icon_name": "crown",
|
|
"trigger_type": "consecutive_month_wins",
|
|
"threshold": 2,
|
|
"bonus_points": 30,
|
|
},
|
|
{
|
|
"key": "boss_battle",
|
|
"name": "Bosskampf",
|
|
"description": "Erledige eine Aufgabe mit besonders hohem Punktwert.",
|
|
"icon_name": "bolt",
|
|
"trigger_type": "high_point_task",
|
|
"threshold": 25,
|
|
"bonus_points": 12,
|
|
},
|
|
{
|
|
"key": "foreign_savior",
|
|
"name": "Fremdretter",
|
|
"description": "Erledige 5 Aufgaben, die ursprünglich jemand anderem zugewiesen waren.",
|
|
"icon_name": "users-crown",
|
|
"trigger_type": "foreign_tasks_completed",
|
|
"threshold": 5,
|
|
"bonus_points": 18,
|
|
},
|
|
{
|
|
"key": "white_glove",
|
|
"name": "Weiße Weste",
|
|
"description": "Bleibe einen ganzen Monat ohne überfällige Aufgabe.",
|
|
"icon_name": "shield",
|
|
"trigger_type": "clean_month",
|
|
"threshold": 1,
|
|
"bonus_points": 20,
|
|
},
|
|
{
|
|
"key": "comeback_kid",
|
|
"name": "Comeback Kid",
|
|
"description": "Gewinne nach einem verlorenen Monat den nächsten Monat.",
|
|
"icon_name": "award",
|
|
"trigger_type": "comeback_win",
|
|
"threshold": 1,
|
|
"bonus_points": 18,
|
|
},
|
|
]
|
|
|
|
|
|
def seed_badges() -> None:
|
|
wanted_keys = {payload["key"] for payload in DEFAULT_BADGES}
|
|
BadgeDefinition.query.filter(~BadgeDefinition.key.in_(wanted_keys)).delete(synchronize_session=False)
|
|
for payload in DEFAULT_BADGES:
|
|
badge = BadgeDefinition.query.filter_by(key=payload["key"]).first()
|
|
if not badge:
|
|
db.session.add(BadgeDefinition(**payload))
|
|
continue
|
|
badge.name = payload["name"]
|
|
badge.description = payload["description"]
|
|
badge.icon_name = payload["icon_name"]
|
|
badge.trigger_type = payload["trigger_type"]
|
|
badge.threshold = payload["threshold"]
|
|
badge.bonus_points = payload["bonus_points"]
|
|
badge.active = True
|
|
db.session.commit()
|
|
|
|
|
|
def register_cli(app) -> None:
|
|
@app.cli.command("init-db")
|
|
def init_db_command():
|
|
db.create_all()
|
|
seed_badges()
|
|
click.echo("Datenbank und Standard-Badges sind bereit.")
|
|
|
|
@app.cli.command("create-user")
|
|
@click.option("--name", prompt=True, help="Anzeigename")
|
|
@click.option("--email", prompt=True, help="E-Mail")
|
|
@click.option("--password", prompt=True, hide_input=True, confirmation_prompt=True, help="Passwort")
|
|
def create_user_command(name: str, email: str, password: str):
|
|
existing = User.query.filter_by(email=email.lower().strip()).first()
|
|
if existing:
|
|
click.echo("Es existiert bereits ein Nutzer mit dieser E-Mail.")
|
|
raise SystemExit(1)
|
|
|
|
user = User(name=name.strip(), email=email.lower().strip())
|
|
user.set_password(password)
|
|
db.session.add(user)
|
|
db.session.commit()
|
|
click.echo(f"Nutzer {user.email} wurde angelegt.")
|
|
|
|
@app.cli.command("archive-months")
|
|
def archive_months_command():
|
|
archive_months_missing_up_to_previous()
|
|
click.echo("Monatsarchiv wurde geprüft.")
|
|
|
|
@app.cli.command("notify-due")
|
|
def notify_due_command():
|
|
result = send_due_notifications()
|
|
click.echo(f"Due-Push: sent={result.sent} skipped={result.skipped} failed={result.failed}")
|
|
|
|
@app.cli.command("notify-monthly-winner")
|
|
def notify_monthly_winner_command():
|
|
result = send_monthly_winner_notifications()
|
|
click.echo(f"Winner-Push: sent={result.sent} skipped={result.skipped} failed={result.failed}")
|