2023-02-17 07:02:21 +01:00
|
|
|
from typing import Optional, Dict, List
|
|
|
|
from sqlalchemy import Table, Column, UniqueConstraint, PrimaryKeyConstraint, and_, case
|
|
|
|
from sqlalchemy.types import Integer, String, TIMESTAMP, Boolean
|
|
|
|
from sqlalchemy.schema import ForeignKey
|
|
|
|
from sqlalchemy.sql import func, select, update, delete
|
|
|
|
from sqlalchemy.engine import Row
|
|
|
|
from sqlalchemy.dialects.mysql import insert
|
|
|
|
|
|
|
|
from core.data.schema import BaseData, metadata
|
|
|
|
|
|
|
|
item = Table(
|
|
|
|
"wacca_item",
|
|
|
|
metadata,
|
|
|
|
Column("id", Integer, primary_key=True, nullable=False),
|
2023-03-09 17:38:58 +01:00
|
|
|
Column(
|
|
|
|
"user",
|
|
|
|
ForeignKey("aime_user.id", ondelete="cascade", onupdate="cascade"),
|
|
|
|
nullable=False,
|
|
|
|
),
|
2023-02-17 07:02:21 +01:00
|
|
|
Column("item_id", Integer, nullable=False),
|
2023-03-09 17:38:58 +01:00
|
|
|
Column("type", Integer, nullable=False),
|
2023-02-17 07:02:21 +01:00
|
|
|
Column("acquire_date", TIMESTAMP, nullable=False, server_default=func.now()),
|
|
|
|
Column("use_count", Integer, server_default="0"),
|
|
|
|
UniqueConstraint("user", "item_id", "type", name="wacca_item_uk"),
|
2023-03-09 17:38:58 +01:00
|
|
|
mysql_charset="utf8mb4",
|
2023-02-17 07:02:21 +01:00
|
|
|
)
|
|
|
|
|
|
|
|
ticket = Table(
|
|
|
|
"wacca_ticket",
|
|
|
|
metadata,
|
|
|
|
Column("id", Integer, primary_key=True, nullable=False),
|
2023-03-09 17:38:58 +01:00
|
|
|
Column(
|
|
|
|
"user",
|
|
|
|
ForeignKey("aime_user.id", ondelete="cascade", onupdate="cascade"),
|
|
|
|
nullable=False,
|
|
|
|
),
|
2023-02-17 07:02:21 +01:00
|
|
|
Column("ticket_id", Integer, nullable=False),
|
|
|
|
Column("acquire_date", TIMESTAMP, nullable=False, server_default=func.now()),
|
|
|
|
Column("expire_date", TIMESTAMP),
|
2023-03-09 17:38:58 +01:00
|
|
|
mysql_charset="utf8mb4",
|
2023-02-17 07:02:21 +01:00
|
|
|
)
|
|
|
|
|
|
|
|
song_unlock = Table(
|
|
|
|
"wacca_song_unlock",
|
|
|
|
metadata,
|
|
|
|
Column("id", Integer, primary_key=True, nullable=False),
|
2023-03-09 17:38:58 +01:00
|
|
|
Column(
|
|
|
|
"user",
|
|
|
|
ForeignKey("aime_user.id", ondelete="cascade", onupdate="cascade"),
|
|
|
|
nullable=False,
|
|
|
|
),
|
2023-02-17 07:02:21 +01:00
|
|
|
Column("song_id", Integer, nullable=False),
|
2023-03-09 17:38:58 +01:00
|
|
|
Column("highest_difficulty", Integer, nullable=False),
|
2023-02-17 07:02:21 +01:00
|
|
|
Column("acquire_date", TIMESTAMP, nullable=False, server_default=func.now()),
|
|
|
|
UniqueConstraint("user", "song_id", name="wacca_song_unlock_uk"),
|
2023-03-09 17:38:58 +01:00
|
|
|
mysql_charset="utf8mb4",
|
2023-02-17 07:02:21 +01:00
|
|
|
)
|
|
|
|
|
|
|
|
trophy = Table(
|
|
|
|
"wacca_trophy",
|
|
|
|
metadata,
|
|
|
|
Column("id", Integer, primary_key=True, nullable=False),
|
2023-03-09 17:38:58 +01:00
|
|
|
Column(
|
|
|
|
"user",
|
|
|
|
ForeignKey("aime_user.id", ondelete="cascade", onupdate="cascade"),
|
|
|
|
nullable=False,
|
|
|
|
),
|
2023-02-17 07:02:21 +01:00
|
|
|
Column("trophy_id", Integer, nullable=False),
|
|
|
|
Column("season", Integer, nullable=False),
|
|
|
|
Column("progress", Integer, nullable=False, server_default="0"),
|
|
|
|
Column("badge_type", Integer, nullable=False, server_default="0"),
|
|
|
|
UniqueConstraint("user", "trophy_id", "season", name="wacca_trophy_uk"),
|
2023-03-09 17:38:58 +01:00
|
|
|
mysql_charset="utf8mb4",
|
2023-02-17 07:02:21 +01:00
|
|
|
)
|
|
|
|
|
2023-03-09 17:38:58 +01:00
|
|
|
|
|
|
|
class WaccaItemData(BaseData):
|
2023-02-17 07:02:21 +01:00
|
|
|
def get_song_unlocks(self, user_id: int) -> Optional[List[Row]]:
|
|
|
|
sql = song_unlock.select(song_unlock.c.user == user_id)
|
|
|
|
|
|
|
|
result = self.execute(sql)
|
2023-03-09 17:38:58 +01:00
|
|
|
if result is None:
|
|
|
|
return None
|
|
|
|
|
2023-02-17 07:02:21 +01:00
|
|
|
return result.fetchall()
|
|
|
|
|
|
|
|
def unlock_song(self, user_id: int, song_id: int, difficulty: int) -> Optional[int]:
|
|
|
|
sql = insert(song_unlock).values(
|
2023-03-09 17:38:58 +01:00
|
|
|
user=user_id, song_id=song_id, highest_difficulty=difficulty
|
2023-02-17 07:02:21 +01:00
|
|
|
)
|
|
|
|
|
|
|
|
conflict = sql.on_duplicate_key_update(
|
|
|
|
highest_difficulty=case(
|
2023-03-09 17:38:58 +01:00
|
|
|
(
|
|
|
|
song_unlock.c.highest_difficulty >= difficulty,
|
|
|
|
song_unlock.c.highest_difficulty,
|
|
|
|
),
|
2023-02-17 07:02:21 +01:00
|
|
|
(song_unlock.c.highest_difficulty < difficulty, difficulty),
|
|
|
|
)
|
|
|
|
)
|
|
|
|
|
|
|
|
result = self.execute(conflict)
|
|
|
|
if result is None:
|
2023-03-09 17:38:58 +01:00
|
|
|
self.logger.error(
|
|
|
|
f"{__name__} failed to unlock song! user: {user_id}, song_id: {song_id}, difficulty: {difficulty}"
|
|
|
|
)
|
2023-02-17 07:02:21 +01:00
|
|
|
return None
|
2023-03-09 17:38:58 +01:00
|
|
|
|
2023-02-17 07:02:21 +01:00
|
|
|
return result.lastrowid
|
|
|
|
|
|
|
|
def put_item(self, user_id: int, item_type: int, item_id: int) -> Optional[int]:
|
|
|
|
sql = insert(item).values(
|
2023-03-09 17:38:58 +01:00
|
|
|
user=user_id,
|
|
|
|
item_id=item_id,
|
|
|
|
type=item_type,
|
2023-02-17 07:02:21 +01:00
|
|
|
)
|
|
|
|
|
2023-03-09 17:38:58 +01:00
|
|
|
conflict = sql.on_duplicate_key_update(use_count=item.c.use_count + 1)
|
2023-02-17 07:02:21 +01:00
|
|
|
|
|
|
|
result = self.execute(conflict)
|
|
|
|
if result is None:
|
2023-03-09 17:38:58 +01:00
|
|
|
self.logger.error(
|
|
|
|
f"{__name__} failed to insert item! user: {user_id}, item_id: {item_id}, item_type: {item_type}"
|
|
|
|
)
|
2023-02-17 07:02:21 +01:00
|
|
|
return None
|
2023-03-09 17:38:58 +01:00
|
|
|
|
2023-02-17 07:02:21 +01:00
|
|
|
return result.lastrowid
|
2023-03-09 17:38:58 +01:00
|
|
|
|
|
|
|
def get_items(
|
|
|
|
self, user_id: int, item_type: int = None, item_id: int = None
|
|
|
|
) -> Optional[List[Row]]:
|
2023-02-17 07:02:21 +01:00
|
|
|
"""
|
|
|
|
A catch-all item lookup given a profile and option item type and ID specifiers
|
|
|
|
"""
|
|
|
|
sql = item.select(
|
2023-03-09 17:38:58 +01:00
|
|
|
and_(
|
|
|
|
item.c.user == user_id,
|
|
|
|
item.c.type == item_type if item_type is not None else True,
|
|
|
|
item.c.item_id == item_id if item_id is not None else True,
|
|
|
|
)
|
2023-02-17 07:02:21 +01:00
|
|
|
)
|
|
|
|
|
|
|
|
result = self.execute(sql)
|
2023-03-09 17:38:58 +01:00
|
|
|
if result is None:
|
|
|
|
return None
|
2023-02-17 07:02:21 +01:00
|
|
|
return result.fetchall()
|
2023-03-09 17:38:58 +01:00
|
|
|
|
2023-02-17 07:02:21 +01:00
|
|
|
def get_tickets(self, user_id: int) -> Optional[List[Row]]:
|
|
|
|
sql = select(ticket).where(ticket.c.user == user_id)
|
|
|
|
|
|
|
|
result = self.execute(sql)
|
2023-03-09 17:38:58 +01:00
|
|
|
if result is None:
|
|
|
|
return None
|
2023-02-17 07:02:21 +01:00
|
|
|
return result.fetchall()
|
|
|
|
|
|
|
|
def add_ticket(self, user_id: int, ticket_id: int) -> None:
|
2023-03-09 17:38:58 +01:00
|
|
|
sql = insert(ticket).values(user=user_id, ticket_id=ticket_id)
|
2023-02-17 07:02:21 +01:00
|
|
|
|
|
|
|
result = self.execute(sql)
|
|
|
|
if result is None:
|
2023-03-09 17:38:58 +01:00
|
|
|
self.logger.error(
|
|
|
|
f"add_ticket: Failed to insert wacca ticket! user_id: {user_id} ticket_id {ticket_id}"
|
|
|
|
)
|
2023-02-17 07:02:21 +01:00
|
|
|
return None
|
|
|
|
return result.lastrowid
|
2023-03-09 17:38:58 +01:00
|
|
|
|
2023-02-17 07:02:21 +01:00
|
|
|
def spend_ticket(self, id: int) -> None:
|
|
|
|
sql = delete(ticket).where(ticket.c.id == id)
|
|
|
|
|
|
|
|
result = self.execute(sql)
|
|
|
|
if result is None:
|
2023-08-08 16:17:56 +02:00
|
|
|
self.logger.warning(f"Failed to delete ticket id {id}")
|
2023-02-17 07:02:21 +01:00
|
|
|
return None
|
|
|
|
|
2023-03-09 17:38:58 +01:00
|
|
|
def get_trophies(self, user_id: int, season: int = None) -> Optional[List[Row]]:
|
2023-02-17 07:02:21 +01:00
|
|
|
if season is None:
|
|
|
|
sql = select(trophy).where(trophy.c.user == user_id)
|
|
|
|
else:
|
2023-03-09 17:38:58 +01:00
|
|
|
sql = select(trophy).where(
|
|
|
|
and_(trophy.c.user == user_id, trophy.c.season == season)
|
|
|
|
)
|
|
|
|
|
2023-02-17 07:02:21 +01:00
|
|
|
result = self.execute(sql)
|
2023-03-09 17:38:58 +01:00
|
|
|
if result is None:
|
|
|
|
return None
|
2023-02-17 07:02:21 +01:00
|
|
|
return result.fetchall()
|
2023-03-09 17:38:58 +01:00
|
|
|
|
|
|
|
def update_trophy(
|
|
|
|
self, user_id: int, trophy_id: int, season: int, progress: int, badge_type: int
|
|
|
|
) -> Optional[int]:
|
2023-02-17 07:02:21 +01:00
|
|
|
sql = insert(trophy).values(
|
2023-03-09 17:38:58 +01:00
|
|
|
user=user_id,
|
|
|
|
trophy_id=trophy_id,
|
|
|
|
season=season,
|
|
|
|
progress=progress,
|
|
|
|
badge_type=badge_type,
|
2023-02-17 07:02:21 +01:00
|
|
|
)
|
|
|
|
|
2023-03-09 17:38:58 +01:00
|
|
|
conflict = sql.on_duplicate_key_update(progress=progress)
|
2023-02-17 07:02:21 +01:00
|
|
|
|
|
|
|
result = self.execute(conflict)
|
|
|
|
if result is None:
|
2023-03-09 17:38:58 +01:00
|
|
|
self.logger.error(
|
|
|
|
f"update_trophy: Failed to insert wacca trophy! user_id: {user_id} trophy_id: {trophy_id} progress {progress}"
|
|
|
|
)
|
2023-02-17 07:02:21 +01:00
|
|
|
return None
|
|
|
|
return result.lastrowid
|