commit893b229cc6
Author: zman <joshua.k.rzemien@gmail.com> Date: Fri Feb 7 22:14:08 2025 -0500 j commit06f539aea2
Author: zman <joshua.k.rzemien@gmail.com> Date: Fri Feb 7 21:55:30 2025 -0500 fk commitd0c2960ec9
Author: zman <joshua.k.rzemien@gmail.com> Date: Fri Feb 7 21:50:53 2025 -0500 frick commit6b1362c166
Author: zman <joshua.k.rzemien@gmail.com> Date: Fri Feb 7 21:49:40 2025 -0500 database commit8cadc6df4c
Author: zman <joshua.k.rzemien@gmail.com> Date: Fri Feb 7 21:38:09 2025 -0500 asdf commit1ca6f98684
Author: zman <joshua.k.rzemien@gmail.com> Date: Fri Feb 7 21:32:50 2025 -0500 fffff commit8bb337a9c3
Author: zman <joshua.k.rzemien@gmail.com> Date: Fri Feb 7 21:31:13 2025 -0500 ffff commit65aba280c5
Author: zman <joshua.k.rzemien@gmail.com> Date: Fri Feb 7 21:26:16 2025 -0500 aa commit59ef03a59e
Author: zman <joshua.k.rzemien@gmail.com> Date: Fri Feb 7 21:24:21 2025 -0500 asdf commitf44d5740fc
Author: zman <joshua.k.rzemien@gmail.com> Date: Fri Feb 7 21:23:32 2025 -0500 aaa commit13c96b1643
Author: zman <joshua.k.rzemien@gmail.com> Date: Fri Feb 7 21:18:54 2025 -0500 sdf commit949c795fd1
Author: zman <joshua.k.rzemien@gmail.com> Date: Fri Feb 7 21:17:53 2025 -0500 asdf commit8c3cd423fe
Author: zman <joshua.k.rzemien@gmail.com> Date: Fri Feb 7 20:56:01 2025 -0500 app2 commit78eafc739e
Author: zman <joshua.k.rzemien@gmail.com> Date: Fri Feb 7 20:54:55 2025 -0500 app commitdc47eced14
Author: zman <joshua.k.rzemien@gmail.com> Date: Fri Feb 7 20:43:15 2025 -0500 asdfasdfasdf commite24bcae88c
Author: zman <joshua.k.rzemien@gmail.com> Date: Fri Feb 7 20:39:44 2025 -0500 a commitc894451bfe
Author: zman <joshua.k.rzemien@gmail.com> Date: Fri Feb 7 20:38:20 2025 -0500 req commit3d09869562
Author: zman <joshua.k.rzemien@gmail.com> Date: Fri Feb 7 20:33:27 2025 -0500 wrong number = code dont work lol i love computers commit4c93a1271b
Author: zman <joshua.k.rzemien@gmail.com> Date: Fri Feb 7 20:29:39 2025 -0500 q commit1f5361da88
Author: zman <joshua.k.rzemien@gmail.com> Date: Fri Feb 7 18:27:20 2025 -0500 same as original code now -5 days of my life commit511b070cbb
Author: zman <joshua.k.rzemien@gmail.com> Date: Fri Feb 7 13:52:28 2025 -0500 pricey worky commit964fdd641b
Author: zman <joshua.k.rzemien@gmail.com> Date: Fri Feb 7 11:37:29 2025 -0500 prep for pricing service work commita78c3bcba3
Author: zman <joshua.k.rzemien@gmail.com> Date: Wed Feb 5 21:51:22 2025 -0500 more stuff yay commitbd9cfca7a9
Author: zman <joshua.k.rzemien@gmail.com> Date: Tue Feb 4 22:30:33 2025 -0500 GIGA FIXED EVERYTHING OMG commit85510a4671
Author: zman <joshua.k.rzemien@gmail.com> Date: Tue Feb 4 00:01:34 2025 -0500 data model change and some new services
116 lines
3.8 KiB
Python
116 lines
3.8 KiB
Python
from sqlalchemy import create_engine, text
|
|
from sqlalchemy.orm import sessionmaker, Session
|
|
from contextlib import contextmanager
|
|
from typing import Generator
|
|
import os
|
|
from sqlalchemy import inspect
|
|
from app.services.tcgplayer import TCGPlayerService
|
|
from app.services.pricing import PricingService
|
|
from app.services.file import FileService
|
|
from app.db.models import Price
|
|
from datetime import datetime
|
|
|
|
|
|
import logging
|
|
logger = logging.getLogger(__name__)
|
|
|
|
# Get database URL from environment variable with fallback
|
|
DATABASE_URL = os.getenv("DATABASE_URL", "sqlite:///omegacard.db")
|
|
|
|
# Create engine with proper configuration
|
|
engine = create_engine(
|
|
DATABASE_URL,
|
|
pool_pre_ping=True, # Enable connection health checks
|
|
pool_size=5, # Set reasonable pool size
|
|
max_overflow=10 # Allow some overflow connections
|
|
)
|
|
|
|
# Create session factory
|
|
SessionLocal = sessionmaker(
|
|
bind=engine,
|
|
autocommit=False,
|
|
autoflush=False
|
|
)
|
|
|
|
@contextmanager
|
|
def get_db_session() -> Generator[Session, None, None]:
|
|
"""Context manager for database sessions"""
|
|
session = SessionLocal()
|
|
try:
|
|
yield session
|
|
except Exception as e:
|
|
logger.error(f"Database session error: {str(e)}")
|
|
session.rollback()
|
|
raise
|
|
finally:
|
|
session.close()
|
|
|
|
def get_db() -> Generator[Session, None, None]:
|
|
"""Dependency for FastAPI to get database sessions"""
|
|
with get_db_session() as session:
|
|
yield session
|
|
|
|
def prepopulate_data(db: Session, db_exist: bool = False) -> None:
|
|
file_service = FileService(db)
|
|
tcgplayer_service = TCGPlayerService(db, file_service)
|
|
pricing_service = PricingService(db, file_service, tcgplayer_service)
|
|
if not db_exist:
|
|
tcgplayer_service.populate_tcgplayer_groups()
|
|
file = tcgplayer_service.load_tcgplayer_cards()
|
|
pricing_service.cron_load_prices(file)
|
|
else:
|
|
pricing_service.cron_load_prices()
|
|
|
|
def init_db() -> None:
|
|
"""Initialize database tables and run first-time setup if needed"""
|
|
from .models import Base
|
|
try:
|
|
inspector = inspect(engine)
|
|
tables_exist = all(
|
|
table in inspector.get_table_names()
|
|
for table in Base.metadata.tables.keys()
|
|
)
|
|
if tables_exist:
|
|
with get_db_session() as db:
|
|
# get date created of latest pricing record
|
|
latest_price = db.query(Price).order_by(Price.date_created.desc()).first()
|
|
if latest_price:
|
|
# check if it is greater than 1.5 hours old
|
|
if (datetime.now() - latest_price.date_created).total_seconds() > 5400:
|
|
prepopulate_data(db, db_exist=True)
|
|
else:
|
|
prepopulate_data(db, db_exist=True)
|
|
|
|
# Create tables if they don't exist
|
|
Base.metadata.create_all(bind=engine)
|
|
|
|
# Run first-time setup only if tables were just created
|
|
if not tables_exist:
|
|
with get_db_session() as db:
|
|
prepopulate_data(db)
|
|
logger.info("First-time database setup completed")
|
|
|
|
logger.info("Database initialization completed")
|
|
except Exception as e:
|
|
logger.error(f"Failed to initialize database: {str(e)}")
|
|
raise
|
|
|
|
def check_db_connection() -> bool:
|
|
"""Check if database connection is working"""
|
|
try:
|
|
with get_db_session() as session:
|
|
session.execute(text("SELECT 1"))
|
|
return True
|
|
except Exception as e:
|
|
logger.error(f"Database connection check failed: {str(e)}")
|
|
return False
|
|
|
|
def destroy_db() -> None:
|
|
"""Destroy all database tables"""
|
|
from .models import Base
|
|
try:
|
|
Base.metadata.drop_all(bind=engine)
|
|
logger.info("Database tables dropped successfully")
|
|
except Exception as e:
|
|
logger.error(f"Failed to destroy database: {str(e)}")
|
|
raise |