From 3995ce9f51cf3b1245d11044f8e2dd7f3900644b Mon Sep 17 00:00:00 2001 From: Michael Moore <5983927+MichaelMakesGames@users.noreply.github.com> Date: Tue, 21 May 2024 21:53:18 -0500 Subject: [PATCH] Use alembic to automatically migrate game DBs (#153) Closes #151 --- requirements.txt | 1 + stellarisdashboard/datamodel.py | 32 +++++++++++++++++++++++++++++--- 2 files changed, 30 insertions(+), 3 deletions(-) diff --git a/requirements.txt b/requirements.txt index 7e5fcce..c959519 100644 --- a/requirements.txt +++ b/requirements.txt @@ -1,3 +1,4 @@ +alembic==1.13.1 Brotli==1.0.9 click==8.1.3 contourpy==1.0.6 diff --git a/stellarisdashboard/datamodel.py b/stellarisdashboard/datamodel.py index e7f6b6e..612d551 100644 --- a/stellarisdashboard/datamodel.py +++ b/stellarisdashboard/datamodel.py @@ -11,6 +11,11 @@ from sqlalchemy.ext.declarative import declarative_base from sqlalchemy.orm import sessionmaker, relationship, scoped_session +from alembic.autogenerate import produce_migrations +from alembic.migration import MigrationContext +from alembic.operations import Operations +from alembic.operations.ops import ModifyTableOps + from stellarisdashboard import config, game_info logger = logging.getLogger(__name__) @@ -29,7 +34,28 @@ def get_db_session(game_id) -> sqlalchemy.orm.Session: if not db_file.exists(): logger.info(f"Creating database for game {game_id} in file {db_file}.") engine = sqlalchemy.create_engine(f"sqlite:///{db_file}", echo=False) - Base.metadata.create_all(bind=engine) + + # auto-migrate the DB + # based on https://alembic.sqlalchemy.org/en/latest/cookbook.html#run-alembic-operation-objects-directly-as-in-from-autogenerate + with engine.connect() as connection: + mc = MigrationContext.configure(connection) + migrations = produce_migrations(mc, Base.metadata) + operations = Operations(mc) + use_batch = engine.name == "sqlite" + stack = [migrations.upgrade_ops] + while stack: + elem = stack.pop(0) + if use_batch and isinstance(elem, ModifyTableOps): + with operations.batch_alter_table( + elem.table_name, schema=elem.schema + ) as batch_ops: + for table_elem in elem.ops: + batch_ops.invoke(table_elem) + elif hasattr(elem, "ops"): + stack.extend(elem.ops) + else: + operations.invoke(elem) + _ENGINES[game_id] = engine _SESSIONMAKERS[game_id] = scoped_session(sessionmaker(bind=engine)) _DB_LOCKS[game_id] = threading.Lock() @@ -605,9 +631,9 @@ class Country(Base): __tablename__ = "country" country_id = Column(Integer, primary_key=True) game_id = Column(ForeignKey(Game.game_id)) - capital_planet_id = Column(ForeignKey("planet.planet_id")) + capital_planet_id = Column(ForeignKey("planet.planet_id", use_alter=True)) - ruler_id = Column(ForeignKey("leader.leader_id")) + ruler_id = Column(ForeignKey("leader.leader_id", use_alter=True)) is_player = Column(Boolean) is_other_player = Column(Boolean)