group_horse_racing: - settle_race: rewrite with 7 bug fixes (race condition, draw double-credit, empty participants, etc.) - models.py: reorder fields for correct defaults, add indexes - message_service: add logger import danding_points: - api.py: add finally blocks to 3 methods (add_points, get_history, get_leaderboard) - database.py: add finally block to get_user_balance chatai: - __init__.py: deprecated API→asyncio.to_thread, deduplicate logging, taskkill filter for safety - screenshot.py: XSS protection with bleach on HTML content - requirements.txt: add bleach dependency danding_qqpush: - api.py L13: fix self-referencing _renderer NameError crash - api.py: lazy singleton pattern via _get_renderer() instead of per-request ImageRenderer - __init__.py: mask Token in log output (security) All 34 tests pass.
107 lines
3.5 KiB
Python
107 lines
3.5 KiB
Python
import sqlite3
|
|
import os
|
|
from datetime import datetime
|
|
from typing import Optional, List, Dict, Any
|
|
from .config import Config
|
|
|
|
|
|
class PointsDatabase:
|
|
"""SQLite database handler for points system."""
|
|
|
|
def __init__(self, config: Config):
|
|
self.config = config
|
|
self.db_path = config.POINTS_DB_FILE
|
|
self._ensure_db_dir()
|
|
self._init_db()
|
|
|
|
def _ensure_db_dir(self):
|
|
"""Create database directory if it doesn't exist."""
|
|
db_dir = os.path.dirname(self.db_path)
|
|
if db_dir:
|
|
os.makedirs(db_dir, exist_ok=True)
|
|
|
|
def _init_db(self):
|
|
"""Initialize database tables."""
|
|
conn = sqlite3.connect(self.db_path, timeout=5.0)
|
|
cursor = conn.cursor()
|
|
|
|
# Create user_points table
|
|
cursor.execute(
|
|
"""
|
|
CREATE TABLE IF NOT EXISTS user_points (
|
|
user_id TEXT PRIMARY KEY,
|
|
points INTEGER NOT NULL DEFAULT 0 CHECK(points >= 0),
|
|
total_earned INTEGER NOT NULL DEFAULT 0,
|
|
total_spent INTEGER NOT NULL DEFAULT 0,
|
|
created_at TEXT NOT NULL,
|
|
updated_at TEXT NOT NULL
|
|
)
|
|
"""
|
|
)
|
|
|
|
# Create point_transactions table
|
|
cursor.execute(
|
|
"""
|
|
CREATE TABLE IF NOT EXISTS point_transactions (
|
|
id INTEGER PRIMARY KEY AUTOINCREMENT,
|
|
user_id TEXT NOT NULL,
|
|
amount INTEGER NOT NULL,
|
|
balance_after INTEGER NOT NULL,
|
|
source TEXT NOT NULL,
|
|
reason TEXT,
|
|
created_at TEXT NOT NULL
|
|
)
|
|
"""
|
|
)
|
|
|
|
# Create indexes
|
|
cursor.execute(
|
|
"CREATE INDEX IF NOT EXISTS idx_transactions_user_id ON point_transactions(user_id)"
|
|
)
|
|
cursor.execute(
|
|
"CREATE INDEX IF NOT EXISTS idx_transactions_source ON point_transactions(source)"
|
|
)
|
|
cursor.execute(
|
|
"CREATE INDEX IF NOT EXISTS idx_transactions_created_at ON point_transactions(created_at)"
|
|
)
|
|
|
|
conn.commit()
|
|
conn.close()
|
|
|
|
def get_connection(self) -> sqlite3.Connection:
|
|
"""Get a database connection."""
|
|
conn = sqlite3.connect(self.db_path, timeout=5.0)
|
|
conn.row_factory = sqlite3.Row
|
|
return conn
|
|
|
|
def get_user_balance(self, user_id: str) -> int:
|
|
"""Get user's current points balance."""
|
|
conn = self.get_connection()
|
|
try:
|
|
cursor = conn.cursor()
|
|
cursor.execute("SELECT points FROM user_points WHERE user_id = ?", (user_id,))
|
|
row = cursor.fetchone()
|
|
return row["points"] if row else 0
|
|
finally:
|
|
conn.close()
|
|
|
|
def ensure_user_exists(self, user_id: str, conn=None) -> None:
|
|
"""Create user account if it doesn't exist. Reuses provided conn if given."""
|
|
should_close = False
|
|
if conn is None:
|
|
conn = self.get_connection()
|
|
should_close = True
|
|
cursor = conn.cursor()
|
|
now = datetime.now().isoformat()
|
|
cursor.execute(
|
|
"""
|
|
INSERT OR IGNORE INTO user_points
|
|
(user_id, points, total_earned, total_spent, created_at, updated_at)
|
|
VALUES (?, 0, 0, 0, ?, ?)
|
|
""",
|
|
(user_id, now, now),
|
|
)
|
|
if should_close:
|
|
conn.commit()
|
|
conn.close()
|