availabili.tf/backend-flask/models/event.py

143 lines
4.7 KiB
Python
Raw Normal View History

2024-11-19 10:46:45 -08:00
from datetime import datetime
import threading
2024-11-19 10:46:45 -08:00
from sqlalchemy.orm import mapped_column, relationship
from sqlalchemy.orm.attributes import Mapped
from sqlalchemy.orm.properties import ForeignKey
2024-11-24 10:47:45 -08:00
from sqlalchemy.schema import UniqueConstraint
2024-11-27 01:03:41 -08:00
from sqlalchemy.types import TIMESTAMP, BigInteger, Integer, String, Text
2024-11-19 10:46:45 -08:00
from sqlalchemy.sql import func
from sqlalchemy_utc import UtcDateTime
2024-11-27 01:03:41 -08:00
from discord_webhook import DiscordWebhook
2024-11-19 10:46:45 -08:00
import app_db
import spec
2024-11-19 10:46:45 -08:00
class Event(app_db.BaseModel):
__tablename__ = "events"
2024-11-24 10:47:45 -08:00
# surrogate key
2024-11-19 10:46:45 -08:00
id: Mapped[int] = mapped_column(Integer, autoincrement=True, primary_key=True)
2024-11-24 10:47:45 -08:00
# primary key
team_id: Mapped[int] = mapped_column(ForeignKey("teams.id"), nullable=False)
2024-11-19 10:46:45 -08:00
name: Mapped[str] = mapped_column(String(255), nullable=False)
start_time: Mapped[datetime] = mapped_column(UtcDateTime, nullable=False)
2024-11-24 10:47:45 -08:00
description: Mapped[str] = mapped_column(Text, nullable=True)
2024-11-19 10:46:45 -08:00
created_at: Mapped[datetime] = mapped_column(TIMESTAMP, server_default=func.now())
2024-11-27 01:03:41 -08:00
discord_message_id: Mapped[int | None] = mapped_column(BigInteger, nullable=True)
2024-11-19 10:46:45 -08:00
team: Mapped["Team"] = relationship("Team", back_populates="events")
2024-11-27 01:03:41 -08:00
players: Mapped[list["PlayerEvent"]] = relationship("PlayerEvent", back_populates="event")
2024-11-19 10:46:45 -08:00
2024-11-24 10:47:45 -08:00
__table_args__ = (
UniqueConstraint("team_id", "name", "start_time"),
)
2024-11-27 01:03:41 -08:00
def get_discord_content(self):
start_timestamp = int(self.start_time.timestamp())
players = list(self.players)
# players with a role should be sorted first
players.sort(key=lambda p: p.role is not None, reverse=True)
players_info = []
for player in players:
player_info = "- "
if player.role:
player_info += f"**{player.role.role.name}:** "
player_info += f"{player.player.username}"
if player.has_confirmed:
player_info += ""
else:
player_info += ""
players_info.append(player_info)
return "\n".join([
f"# {self.name}",
"",
self.description or "*No description.*",
"",
f"<t:{start_timestamp}:f>",
"\n".join(players_info),
"",
"[Confirm attendance here]" +
2024-11-27 01:03:41 -08:00
f"(https://availabili.tf/team/id/{self.team.id}/events/{self.id})",
])
def get_or_create_webhook(self):
integration = app_db.db.session.query(
TeamDiscordIntegration
).where(
TeamDiscordIntegration.team_id == self.team_id
).first()
if not integration:
return None
if self.discord_message_id:
return DiscordWebhook(
integration.webhook_url,
id=str(self.discord_message_id),
username=integration.webhook_bot_name,
avatar_url=integration.webhook_bot_profile_picture,
2024-11-27 01:03:41 -08:00
)
else:
return DiscordWebhook(
integration.webhook_url,
username=integration.webhook_bot_name,
avatar_url=integration.webhook_bot_profile_picture,
)
2024-11-27 01:03:41 -08:00
def update_discord_message(self):
webhook = self.get_or_create_webhook()
if webhook:
webhook.content = self.get_discord_content()
if webhook.id:
# fire and forget
threading.Thread(target=webhook.edit).start()
2024-11-27 01:03:41 -08:00
else:
webhook.execute()
if webhook_id := webhook.id:
self.discord_message_id = int(webhook_id)
app_db.db.session.commit()
else:
raise Exception("Failed to create webhook")
class EventSchema(spec.BaseModel):
id: int
team_id: int
name: str
2024-11-24 10:47:45 -08:00
description: str | None
start_time: datetime
created_at: datetime
@classmethod
def from_model(cls, model: Event) -> "EventSchema":
return cls(
id=model.id,
name=model.name,
description=model.description,
start_time=model.start_time,
team_id=model.team_id,
created_at=model.created_at,
)
2024-11-19 10:46:45 -08:00
2024-11-27 01:03:41 -08:00
class EventPlayersSchema(spec.BaseModel):
players: list["PlayerEventRolesSchema"]
@classmethod
def from_model(cls, model: Event) -> "EventPlayersSchema":
return cls(
players=[PlayerEventRolesSchema.from_model(player) for player in model.players],
roles=[RoleSchema.from_model(player.role.role) for player in model.players if player.role],
)
2024-11-24 10:47:45 -08:00
2024-11-19 10:46:45 -08:00
from models.team import Team
from models.player_event import PlayerEvent
2024-11-27 01:03:41 -08:00
from models.team_integration import TeamDiscordIntegration